wip 0.3.7-beta

This commit is contained in:
Frede Hundewadt 2022-06-08 18:52:00 +02:00
parent e5e3b1024a
commit 91a8e6f346
42 changed files with 727 additions and 574 deletions

View file

@ -1,13 +1,15 @@
@if (Enabled)
@if (Enabled == 1)
{
<a type="button" class="btn btn-primary" href="/company/@CompanyId/activity">Opret Besøg</a>
<a type="button" class="btn btn-success" href="/company/@CompanyId/activity">Opret Besøg</a>
}
else
{
<a type="button" class="btn btn-primary disabled" aria-disabled="true">Opret Besøg</a>
<a type="button" class="btn btn-outline-secondary disabled" aria-disabled="true">Opret Besøg</a>
}
@code {
[Parameter] public string? CompanyId { get; set; }
[Parameter] public bool Enabled { get; set; }
[Parameter] public string CompanyId { get; set; } = "";
[Parameter]public int Enabled { get; set; }
}

View file

@ -16,9 +16,13 @@
// along with this program. If not, see [https://www.gnu.org/licenses/agpl-3.0.en.html]
//
*@
<span class="version">@Version</span>@if(IsBeta){<span class="version">-beta</span>}
<span class="version">@_app?.Version</span>@if(_app!.IsBeta){<span class="version">-beta</span>}
@code
{
private const string Version = "0.3.1";
private const bool IsBeta = true;
[Inject] IOptions<AppInfo?>? AppInfo { get; set; }
private AppInfo? _app;
protected override void OnInitialized()
{
_app = AppInfo?.Value;
}
}

View file

@ -10,39 +10,39 @@
<td class="color-code">
<img class="state the-good rounded-circle me-1" src="state.png" alt="state"/>
</td>
<td>
<td class="align-middle">
Ok
</td>
</tr>
<tr>
<td>
<td class="align-middle">
<img class="state the-bad rounded-circle me-1" src="state.png" alt="state"/>
</td>
<td>
<td class="align-middle">
Planlæg besøg
</td>
</tr>
<tr>
<td>
<td class="align-middle">
<img class="state the-ugly rounded-circle me-1" src="state.png" alt="state"/>
</td>
<td>
Inteval overskredet
<td class="align-middle">
Interval overskredet
</td>
</tr>
<tr>
<td>
<td class="align-middle">
<img class="state the-draw rounded-circle me-1" src="state.png" alt="state"/>
</td>
<td>
<td class="align-middle">
Opdatering nødvendig
</td>
</tr>
<tr>
<td>
<td class="align-middle">
<img class="state the-dead rounded-circle me-1" src="state.png" alt="state"/>
</td>
<td>
<td class="align-middle">
Virksomhed ophørt
</td>
</tr>

View file

@ -34,13 +34,23 @@
@foreach (var company in Companies)
{
<tr>
<td><DisplayStateComponent StateClass="@(company.HasFolded == 1 ? "the-dead" : Utils.GetVisitState(company.NextVisit))"></DisplayStateComponent></td>
<td>@company.Name</td>
<td>@company.Account</td>
<td>@company.City</td>
<td>
<a class="btn btn-info" href="/company/@company.CompanyId">Vis</a>
<CreateActivityButton CompanyId="@company.CompanyId" Enabled="@company.ValidVat"></CreateActivityButton>
<td class="align-middle">
<DisplayStateComponent StateClass="@(company.HasFolded == 1
? "the-dead" : Utils.GetVisitState(company.NextVisit))">
</DisplayStateComponent>
</td>
<td class="align-middle">
@company.Name
</td>
<td class="align-middle">
@company.Account
</td>
<td class="align-middle">
@company.City
</td>
<td class="align-middle">
<a class="btn btn-edit" href="/company/@company.CompanyId/update">Rediger</a>
<ActivityButton CompanyId="@company.CompanyId" Enabled="@company.ValidVat"></ActivityButton>
</td>
</tr>
}

View file

@ -29,17 +29,13 @@ namespace Wonky.Client.Components
[Parameter] public List<CompanyDto> Companies { get; set; } = new();
[Parameter] public EventCallback<string> OnDelete { get; set; }
[Parameter] public EventCallback<string> OnSelect { get; set; }
[Inject] public NavigationManager NavManager { get; set; }
[Inject] public VatUtils VatUtils { get; set; }
[Inject] public ILocalStorageService Storage { get; set; }
private Confirmation _confirmation = new ();
private string _companyId = string.Empty;
private UserInfoView _user = new();
private string _companyId = "";
protected override async Task OnInitializedAsync()
{
_user = await Storage.GetItemAsync<UserInfoView>("_xu").ConfigureAwait(true);
await base.OnInitializedAsync();
}
private void CallConfirmationModal(string companyId)
{
@ -52,10 +48,5 @@ namespace Wonky.Client.Components
_confirmation.Hide();
await OnDelete.InvokeAsync(_companyId);
}
private void NavigateCompany(string companyId)
{
NavManager.NavigateTo($"/company/{companyId}");
}
}
}

View file

@ -17,5 +17,5 @@
<img class="img-fluid float-start rounded-circle state @StateClass me-1" src="state.png" alt="state"/>
@code{
[Parameter] public string StateClass { get; set; } = "the-dead";
[Parameter] public string StateClass { get; set; } = "";
}

View file

@ -48,7 +48,7 @@ public partial class Home
_workDate = DateTime.Parse(_prefs.WorkDate).ToLongDateString();
Activities = await ActivityRepo.GetActivities(_prefs.WorkDate).ConfigureAwait(true);
Activities = await ActivityRepo.GetActivities(_prefs.WorkDate);
}

View file

@ -30,14 +30,14 @@
@foreach (var salesItem in SalesItems)
{
<tr>
<td>
<td class="align-middle">
@salesItem.Name
</td>
<td>@salesItem.ShortName</td>
<td>
<td class="align-middle">@salesItem.ShortName</td>
<td class="align-middle">
@salesItem.Sku
</td>
<td>
<td class="align-middle">
<ul class="list-group">
@foreach (var rate in salesItem.Rates)
{

View file

@ -19,7 +19,7 @@
<DataAnnotationsValidator/>
<div class="row">
<div class="col">
<InputText id="vatNumber" class="form-control" placeholder="reg.nr. eks. 26991765"
<InputText id="vatNumber" class="form-control" placeholder="26991765"
@bind-Value="@VatNumber"/>
<ValidationMessage For="@(() => VatNumber)"/>
</div>

View file

@ -27,11 +27,14 @@ public static class Utils
public static string GetVisitState(string dtNextVisit)
{
if (dtNextVisit == "1970-01-01" || !DateTime.TryParse(dtNextVisit, out var dtNext))
if (dtNextVisit is "0001-01-01" or "1970-01-01")
return "the-draw";
var dtNow = DateTime.Now;
if (dtNow >= dtNext)
var dtNext = DateTime.Parse(dtNextVisit);
if (dtNow > dtNext)
return "the-ugly";
return (dtNow > dtNext.AddDays(-14)) ? "the-bad" : "the-good";
}
}

View file

@ -5,7 +5,7 @@ public class VatUtils
// https://ec.europa.eu/taxation_customs/vies/faqvies.do#item_11
// https://ec.europa.eu/taxation_customs/vies/
public bool ValidateFormat(string countryCode, string vatNumber)
public static bool ValidateFormat(string countryCode, string vatNumber)
{
if (string.IsNullOrWhiteSpace(vatNumber) || string.IsNullOrWhiteSpace(countryCode))
return false;
@ -19,7 +19,7 @@ public class VatUtils
};
}
private bool ValidateFormatDk(string vatNumber)
private static bool ValidateFormatDk(string vatNumber)
{
// https://wiki.scn.sap.com/wiki/display/CRM/Denmark
// 8 digits 0 to 9
@ -35,7 +35,7 @@ public class VatUtils
return ValidateMod11(vatToCheck);
}
private bool ValidateFormatNo(string vatNumber)
private static bool ValidateFormatNo(string vatNumber)
{
// https://wiki.scn.sap.com/wiki/display/CRM/Norway
// 12 digits
@ -48,7 +48,7 @@ public class VatUtils
return long.Parse(vatToCheck) != 0 && ValidateMod11(vatNumber);
}
private bool ValidateFormatSe(string vatNumber)
private static bool ValidateFormatSe(string vatNumber)
{
// https://wiki.scn.sap.com/wiki/display/CRM/Sweden
// 12 digits 0 to 9
@ -73,7 +73,7 @@ public class VatUtils
return $"{vatToCheck[..9]}{c10}01" == vatNumber;
}
private bool ValidateMod11(string number)
private static bool ValidateMod11(string number)
{
if (long.Parse(number) == 0)
return false;
@ -87,7 +87,7 @@ public class VatUtils
return sum % 11 == 0;
}
private string SanitizeVatNumber(string vatNumber)
private static string SanitizeVatNumber(string vatNumber)
{
vatNumber = vatNumber.ToUpperInvariant();
return vatNumber

View file

@ -16,6 +16,7 @@
using System.Net;
using System.Net.Http.Headers;
using System.Threading.Tasks;
using Blazored.LocalStorage;
using Blazored.Toast.Services;
using Wonky.Client.Services;
using Microsoft.AspNetCore.Components;
@ -31,16 +32,19 @@ namespace Wonky.Client.HttpInterceptors
private readonly IToastService _toast;
private readonly RefreshTokenService _refreshTokenService;
private ILogger<HttpInterceptorService> _logger;
private ILocalStorageService _storage;
public HttpInterceptorService(HttpClientInterceptor interceptor,
NavigationManager navigation, IToastService toast,
RefreshTokenService refreshTokenService, ILogger<HttpInterceptorService> logger)
RefreshTokenService refreshTokenService, ILogger<HttpInterceptorService> logger,
ILocalStorageService storage)
{
_interceptor = interceptor;
_navigation = navigation;
_toast = toast;
_refreshTokenService = refreshTokenService;
_logger = logger;
_storage = storage;
}
public void RegisterEvent()
@ -66,13 +70,11 @@ namespace Wonky.Client.HttpInterceptors
{
// call TryRefreshToken
var token = await _refreshTokenService.TryRefreshToken();
if (!string.IsNullOrEmpty(token))
{
// set new token
e.Request.Headers.Authorization = new AuthenticationHeaderValue("bearer", token);
}
}
}
@ -88,15 +90,17 @@ namespace Wonky.Client.HttpInterceptors
{
case HttpStatusCode.NotFound:
//_navigation.NavigateTo("/404");
message = "404 - Siden blev ikke fundet.";
message = "Der er ingen data ...";
_toast.ShowInfo(message);
break;
case HttpStatusCode.BadRequest:
ClearInfo();
_navigation.NavigateTo($"/login/{currDoc}");
message = "Verifikation nødvendig ...";
_toast.ShowInfo(message);
break;
case HttpStatusCode.Unauthorized:
ClearInfo();
_navigation.NavigateTo($"/login/{currDoc}");
message = "Verifikation nødvendig ...";
_toast.ShowInfo(message);
@ -107,6 +111,11 @@ namespace Wonky.Client.HttpInterceptors
}
throw new HttpResponseException(message);
}
private async void ClearInfo()
{
await _storage.RemoveItemsAsync(new List<string> {"_xa", "_xr", "_xe", "_xu"});
}
}
}

View file

@ -57,7 +57,7 @@ public class ActivityHttpRepository : IActivityHttpRepository
{
var response = await _client
.GetAsync($"{_apiConfig.ActivityEndpoint}/date/{activityDate}")
.ConfigureAwait(true);
;
var content = await response.Content.ReadAsStringAsync();
//Console.WriteLine(content);
return string.IsNullOrWhiteSpace(content)
@ -91,7 +91,7 @@ public class ActivityHttpRepository : IActivityHttpRepository
public async Task<ActivityResponseView> CreateActivity(ActivityDto model)
{
Console.WriteLine(JsonSerializer.Serialize(model));
var response = await _client.PostAsJsonAsync($"{_apiConfig.ActivityEndpoint}", model).ConfigureAwait(true);
var response = await _client.PostAsJsonAsync($"{_apiConfig.ActivityEndpoint}", model);
var content = await response.Content.ReadAsStringAsync();
var result = JsonSerializer.Deserialize<ActivityResponseView>(content);
return result!;
@ -100,14 +100,14 @@ public class ActivityHttpRepository : IActivityHttpRepository
public async Task<ActivityDto> GetActivity(string id)
{
var salesItem = await _client
.GetFromJsonAsync<ActivityDto>($"{_apiConfig.ActivityEndpoint}/{id}").ConfigureAwait(true);
.GetFromJsonAsync<ActivityDto>($"{_apiConfig.ActivityEndpoint}/{id}");
return salesItem ?? new ActivityDto();
}
public async Task<ActivityResponseView> AcceptOffer(string id)
{
var response = await _client.PostAsJsonAsync($"{_apiConfig.ActivityEndpoint}/{id}/accept", id)
.ConfigureAwait(true);
;
var content = await response.Content.ReadAsStringAsync();
var result = JsonSerializer.Deserialize<ActivityResponseView>(content);
return result!;

View file

@ -27,6 +27,7 @@ using Microsoft.Extensions.Options;
using Wonky.Entity.Configuration;
using Wonky.Entity.DTO;
using Wonky.Entity.Requests;
using Wonky.Entity.Views;
#pragma warning disable CS8601
@ -90,22 +91,23 @@ public class CompanyHttpRepository : ICompanyHttpRepository
return company ?? new CompanyDto();
}
public async Task<string> CreateCompany(CompanyDto companyDto)
public async Task<string> CreateCompany(CompanyDto model)
{
var response = await _client.PostAsJsonAsync($"{_apiConfig.CustomerEndpoint}", companyDto);
var response = await _client.PostAsJsonAsync($"{_apiConfig.CustomerEndpoint}", model);
var content = await response.Content.ReadAsStringAsync();
var result = JsonSerializer.Deserialize<CompanyDto>(content);
Console.WriteLine(content);
return result.CompanyId;
}
public async Task UpdateCompany(CompanyDto companyDto)
public async Task<bool> UpdateCompany(string companyId, CompanyDto model)
{
await _client.PutAsJsonAsync($"{_apiConfig.CustomerEndpoint}/{companyDto.CompanyId}", companyDto);
var response = await _client.PutAsJsonAsync($"{_apiConfig.CustomerEndpoint}/{companyId}", model);
return response.IsSuccessStatusCode;
}
public async Task DeleteCompany(string companyId)
public async Task<bool> DeleteCompany(string companyId)
{
await _client.DeleteAsync($"{_apiConfig.CustomerEndpoint}/{companyId}");
var response = await _client.DeleteAsync($"{_apiConfig.CustomerEndpoint}/{companyId}");
return response.IsSuccessStatusCode;
}
}

View file

@ -17,6 +17,7 @@ using System.Threading.Tasks;
using Wonky.Client.Features;
using Wonky.Entity.DTO;
using Wonky.Entity.Requests;
using Wonky.Entity.Views;
namespace Wonky.Client.HttpRepository;
@ -25,7 +26,7 @@ public interface ICompanyHttpRepository
Task<PagingResponse<CompanyDto>> GetCompaniesPaged(CompanyPagingParams pagingParameters);
Task<CompanyDto> GetCompanyByAccount(string accountNumber);
Task<CompanyDto> GetCompanyById(string companyId);
Task<string> CreateCompany(CompanyDto companyDto);
Task UpdateCompany(CompanyDto companyDto);
Task DeleteCompany(string companyId);
Task<string> CreateCompany(CompanyDto model);
Task<bool> UpdateCompany(string companyId, CompanyDto model);
Task<bool> DeleteCompany(string companyId);
}

View file

@ -4,7 +4,7 @@ namespace Wonky.Client.Models;
public class VatAddress
{
[Required] public string StreetName { get; set; } = "";
[Required] public string HouseNumber { get; set; } = "";
[Required] public string ZipCode { get; set; } = "";
[Required(ErrorMessage = "Vejnavn skal angives")] public string StreetName { get; set; } = "";
[Required(ErrorMessage = "Husnummer skal angives")] public string HouseNumber { get; set; } = "";
[Required(ErrorMessage = "Postnr skal angives")] public string ZipCode { get; set; } = "";
}

View file

@ -18,131 +18,174 @@
@page "/company/create"
@using Microsoft.AspNetCore.Authorization
@using Wonky.Client.Components
@using System.Xml
@attribute [Authorize(Roles = "Adviser")]
<h2>Opret firma</h2>
<div class="card bg-light mb-2">
<div class="card-header">
CVR data
</div>
<div class="card-body">
<VatNumberInputComponent OnValidSubmit="GetInfoFromVat"/>
</div>
<div class="card-body">
<VatAddressInputComponent OnValidSubmit="GetInfoFromAddress"/>
</div>
@if (VInfos.Any())
{
<div class="card-body">
@foreach (var info in VInfos)
<h2>Opret kunde</h2>
<table class="table">
<thead>
<tr>
<th colspan="2">Virksomhedsopslag</th>
</tr>
<tr>
<th>Reg.nr.</th>
<th>Adresse</th>
</tr>
</thead>
<tbody>
<tr>
<td class="align-middle">
<VatNumberInputComponent OnValidSubmit="GetInfoFromVat"/>
</td>
@if (_countryCode == "dk")
{
<div class="row mb-2">
<div class="col">
@info.VatNumber
</div>
<div class="col">
@info.Name
</div>
<div class="col">
@info.States[^1].State
</div>
<div class="col">
<button class="btn btn-primary" @onclick="@(() => SelectCompany(info.VatNumber))">OVERFØR</button>
</div>
</div>
<td class="align-middle">
<VatAddressInputComponent Address="_vatAddress" OnValidSubmit="GetInfoFromAddress"/>
</td>
}
</div>
</tr>
@if (_vInfos.Any())
{
<tr>
<td colspan="2">
<table class="table">
<thead>
<tr>
<th scope="col">CVR ORG</th>
<th scope="col">Navn</th>
<th scope="col">Status</th>
<th scope="col"></th>
<th scope="col"></th>
</tr>
</thead>
<tbody>
@foreach (var info in _vInfos)
{
<tr>
<td class="align-middle">@info.VatNumber</td>
<td class="align-middle">@info.Name</td>
<td class="align-middle">@info.States[^1].State</td>
<td class="align-middle"></td>
<td class="align-middle"><button class="btn btn-primary" @onclick="@(() => SelectCompany(info.VatNumber))">OVERFØR</button></td>
</tr>
}
</tbody>
</table>
</td>
</tr>
}
</div>
<div class="card bg-light">
<EditForm EditContext="_createCompany" OnValidSubmit="Create" class="card-body">
</tbody>
</table>
<EditForm EditContext="_createCompany" OnValidSubmit="Create">
<DataAnnotationsValidator />
<InputText type="hidden" id="salesRepId" @bind-Value="_companyDto.SalesRepId"/>
<InputText type="hidden" id="salesRepId" @bind-Value="_createDto.SalesRepId"/>
<table class="table">
<thead>
<tr>
<th>
Besøgt
</th>
<th>
Næste besøg
</th>
<th>
Interval (uger)
</th>
</tr>
</thead>
<tbody>
<tr>
<th scope="row">Reg.nr.</th>
<td class="state"><DisplayStateComponent StateClass="@RegState"></DisplayStateComponent></td>
<td>
<InputText id="vatNumber" class="form-control" @bind-Value="_companyDto.VatNumber"/>
<InputDate id="lastVisit" class="form-control" @bind-Value="@_lastVisit"/>
</td>
</tr>
<tr>
<th scope="row">Firmanavn</th>
<td></td>
<td>
<InputText id="name" class="form-control" @bind-Value="_companyDto.Name"/>
<ValidationMessage For="@(() => _companyDto.Name)"></ValidationMessage>
<InputDate id="nextVisit" class="form-control" @bind-Value="@(_nextVisit)"/>
<ValidationMessage For="@(() => _createDto.NextVisit)"></ValidationMessage>
</td>
</tr>
<tr>
<th scope="row">Adresse</th>
<td></td>
<td>
<InputText id="address1" class="form-control" @bind-Value="_companyDto.Address1"/>
</td>
</tr>
<tr>
<th scope="row">Conavn</th>
<td></td>
<td>
<InputText id="address2" class="form-control" @bind-Value="_companyDto.Address2"/>
</td>
</tr>
<tr>
<th scope="row">Postnr</th>
<td></td>
<td>
<InputText id="zipCode" class="form-control" @bind-Value="_companyDto.ZipCode"/>
<ValidationMessage For="@(() => _companyDto.ZipCode)"></ValidationMessage>
</td>
</tr>
<tr>
<th scope="row">Bynavn</th>
<td></td>
<td>
<InputText id="city" class="form-control" @bind-Value="_companyDto.City"/>
<ValidationMessage For="@(() => _companyDto.City)"></ValidationMessage>
</td>
</tr>
<tr>
<th scope="row">Telefon</th>
<td></td>
<td>
<InputText id="phone" class="form-control" @bind-Value="_companyDto.Phone"/>
</td>
</tr>
<tr>
<th scope="row">Mobil</th>
<td></td>
<td>
<InputText id="mobile" class="form-control" @bind-Value="_companyDto.Mobile"/>
</td>
</tr>
<tr>
<th scope="row">Email</th>
<td></td>
<td>
<InputText id="email" class="form-control" @bind-Value="_companyDto.Email"/>
</td>
</tr>
<tr>
<th scope="row">Attention</th>
<td></td>
<td>
<InputText id="attention" class="form-control" @bind-Value="_companyDto.Attention"/>
<InputNumber id="interval" class="form-control" @bind-Value="_createDto.Interval"/>
<ValidationMessage For="@(() => _createDto.Interval)"></ValidationMessage>
</td>
</tr>
</tbody>
</table>
<table class="table">
<thead>
<tr>
<th colspan="5">
Stamdata
</th>
</tr>
</thead>
<tbody>
<tr>
<td class="align-middle">Reg.nr.</td>
<td class="align-middle state"><DisplayStateComponent StateClass="@RegState"></DisplayStateComponent></td>
<td class="align-middle">
<InputText id="vatNumber" class="form-control" @bind-Value="_createDto.VatNumber"/>
</td>
<td class="align-middle">Telefon</td>
<td class="align-middle">
<InputText id="phone" class="form-control" @bind-Value="_createDto.Phone"/>
</td>
</tr>
<tr>
<td class="align-middle">Firmanavn</td>
<td class="align-middle"></td>
<td colspan="3">
<InputText id="name" class="form-control" @bind-Value="_createDto.Name"/>
<ValidationMessage For="@(() => _createDto.Name)"></ValidationMessage>
</td>
</tr>
<tr>
<td class="align-middle">Attention</td>
<td class="align-middle"></td>
<td colspan="3">
<InputText id="attention" class="form-control" @bind-Value="_createDto.Attention"/>
</td>
</tr>
<tr>
<td class="align-middle">Adresse</td>
<td class="align-middle"></td>
<td class="align-middle">
<InputText id="address1" class="form-control" @bind-Value="_createDto.Address1"/>
</td>
<td class="align-middle">Adresse</td>
<td class="align-middle">
<InputText id="address2" class="form-control" @bind-Value="_createDto.Address2"/>
</td>
</tr>
<tr>
<td class="align-middle">Postnr</td>
<td class="align-middle"></td>
<td class="align-middle">
<InputText id="zipCode" class="form-control" @bind-Value="_createDto.ZipCode"/>
<ValidationMessage For="@(() => _createDto.ZipCode)"></ValidationMessage>
</td>
<td class="align-middle">Bynavn</td>
<td class="align-middle">
<InputText id="city" class="form-control" @bind-Value="_createDto.City"/>
<ValidationMessage For="@(() => _createDto.City)"></ValidationMessage>
</td>
</tr>
<tr>
<td class="align-middle">Mobil</td>
<td class="align-middle"></td>
<td class="align-middle">
<InputText id="mobile" class="form-control" @bind-Value="_createDto.Mobile"/>
</td>
<td class="align-middle">Email</td>
<td class="align-middle">
<InputText id="email" class="form-control" @bind-Value="_createDto.Email"/>
</td>
</tr>
</tbody>
</table>
<div class="row mb-2">
<div class="col-md-12 text-right">
<div class="col-md-8"></div>
<div class="col-md-4">
<button type="submit" class="btn btn-success" disabled="@_formInvalid">Opret</button>
</div>
</div>
</EditForm>
</div>

View file

@ -27,6 +27,7 @@ using Microsoft.AspNetCore.Components.Forms;
using Microsoft.Extensions.Logging;
using Wonky.Client.Helpers;
using Wonky.Client.Models;
using Wonky.Client.Shared;
using Wonky.Entity.DTO;
using Wonky.Entity.Models;
using Wonky.Entity.Requests;
@ -43,77 +44,92 @@ namespace Wonky.Client.Pages
[Inject] public ICompanyHttpRepository CompanyRepo { get; set; }
[Inject] public HttpInterceptorService Interceptor { get; set; }
[Inject] public VatInfoLookupService VatInfoLookupService { get; set; }
private List<VirkRegInfo> VInfos { get; set; } = new();
private CompanyDto _companyDto = new();
private List<VirkRegInfo> _vInfos { get; set; } = new();
private CompanyDto _createDto = new();
private VirkRegInfo _virkRegInfo = new();
private EditContext _createCompany;
private bool _formInvalid = true;
private VatUtils _vatUtils { get; set; }
private string RegState { get; set; } = "";
private VatAddress _vatAddress = new();
private string RegState = "";
private DateTime _lastVisit { get; set; }
private DateTime _nextVisit { get; set; }
private string _countryCode;
protected override async Task OnInitializedAsync()
{
_vatUtils = new VatUtils();
_createCompany = new EditContext(_companyDto);
_createCompany.OnFieldChanged += HandleFieldChanged;
var ux = await StorageService.GetItemAsync<UserInfoView>("_xu");
_companyDto.SalesRepId = ux.Id;
_companyDto.CountryCode = ux.CountryCode;
Interceptor.RegisterEvent();
Interceptor.RegisterBeforeSendEvent();
var ux = await StorageService.GetItemAsync<UserInfoView>("_xu");
_countryCode = ux.CountryCode;
_createDto.SalesRepId = ux.Id;
_createDto.CountryCode = ux.CountryCode;
_lastVisit = DateTime.Now;
_nextVisit = DateTime.Now.AddDays(_createDto.Interval * 7);
_createDto.LastVisit = $"{_lastVisit:yyyy-MM-dd}";
_createDto.LastVisit = $"{_nextVisit:yyyy-MM-dd}";
_createCompany = new EditContext(_createDto);
_createCompany.OnFieldChanged += HandleFieldChanged;
_createCompany.OnValidationStateChanged += ValidationChanged;
}
private async Task GetInfoFromAddress(VatAddress address)
{
VInfos = await VatInfoLookupService.QueryVirkRegistry(
ToastService.ShowInfo("Vent for adresse info ...");
_vInfos = await VatInfoLookupService.QueryVirkRegistry(
new VirkParams
{
StreetName = address.StreetName,
HouseNumber = address.HouseNumber,
ZipCode = address.ZipCode
});
if (!VInfos.Any())
if (!_vInfos.Any())
{
ToastService.ShowError($"Ingen virksomheder fundet.");
ToastService.ShowInfo($"Ingen data fundet ...");
}
}
private async Task GetInfoFromVat(string vatNumber)
{
VInfos = await VatInfoLookupService
ToastService.ShowInfo("Vent for firma info ...");
_vInfos = await VatInfoLookupService
.QueryVirkRegistry(new VirkParams {VatNumber = vatNumber});
if (!VInfos.Any())
if (!_vInfos.Any())
{
ToastService.ShowError($"Firma med CVR '{vatNumber}' findes ikke.");
return;
}
ToastService.ShowSuccess($"Data for '{vatNumber}' er hentet.");
//SelectCompany(vatNumber);
}
private void SelectCompany(string vatNumber)
{
_virkRegInfo = (from x in VInfos where x.VatNumber == vatNumber select x).First();
_virkRegInfo = (from x in _vInfos where x.VatNumber == vatNumber select x).First();
RegState = _virkRegInfo.States[^1].State == "NORMAL" ? "the-good" : "the-ugly";
_companyDto.Name = _virkRegInfo.Name;
_companyDto.Address1 = _virkRegInfo.Address;
_companyDto.Address2 = _virkRegInfo.CoName;
_companyDto.ZipCode = _virkRegInfo.ZipCode;
_companyDto.City = _virkRegInfo.City;
_companyDto.VatNumber = _virkRegInfo.VatNumber;
_createDto.Name = _virkRegInfo.Name;
_createDto.Address1 = _virkRegInfo.CoName;
_createDto.Address2 = _virkRegInfo.Address;
_createDto.ZipCode = _virkRegInfo.ZipCode;
_createDto.City = _virkRegInfo.City;
_createDto.VatNumber = _virkRegInfo.VatNumber;
_createDto.ValidVat = 1;
}
private async Task Create()
{
var newId = await CompanyRepo.CreateCompany(_companyDto);
Console.WriteLine(newId);
ToastService.ShowSuccess($"Godt så! '{_companyDto.Name}' er oprettet i CRM.");
Navigation.NavigateTo($"/company/id/{newId}");
var newId = await CompanyRepo.CreateCompany(_createDto);
if (!string.IsNullOrWhiteSpace(newId))
{
ToastService.ShowSuccess($"'{_createDto.Name}' er oprettet i CRM.");
Navigation.NavigateTo($"/company/id/{newId}");
}
}
private void HandleFieldChanged(object sender, FieldChangedEventArgs e)
{
if (!_vatUtils.ValidateFormat(_companyDto.CountryCode, _companyDto.VatNumber))
_createDto.LastVisit = $"{_lastVisit:yyyy-MM-dd}";
_createDto.NextVisit = $"{_nextVisit:yyyy-MM-dd}";
if (!VatUtils.ValidateFormat(_createDto.CountryCode, _createDto.VatNumber)
|| !_createDto.ValidDateSpan())
{
_formInvalid = false;
_formInvalid = true;
}
else
{
@ -124,8 +140,11 @@ namespace Wonky.Client.Pages
private void ValidationChanged(object sender, ValidationStateChangedEventArgs e)
{
_formInvalid = true;
_createCompany.OnFieldChanged -= HandleFieldChanged;
_createCompany = new EditContext(_companyDto);
_createCompany = new EditContext(_createDto);
_createCompany.OnFieldChanged += HandleFieldChanged;
_createCompany.OnValidationStateChanged -= ValidationChanged;
}

View file

@ -21,147 +21,161 @@
@attribute [Authorize(Roles = "Adviser")]
@page "/company/{CompanyId}/update"
@if (CompanyDto != null)
@if (!string.IsNullOrWhiteSpace(_companyView.Name))
{
<div class="card">
<div class="card-header">
<div class="h2">@CompanyDto.Account - @CompanyDto.Name</div>
</div>
<div class="card-body">
<VatNumberInputComponent OnValidSubmit="GetInfoFromVat"/>
<h5>@_companyView.Account - @_companyView.Name</h5>
</div>
<div class="card-body">
<VatAddressInputComponent Address="vatAddress" OnValidSubmit="GetInfoFromAddress"/>
</div>
@if (VInfos.Any())
@if (_vInfos.Any())
{
<div class="card-body">
@foreach (var info in VInfos)
<table class="table">
<thead>
<tr>
<th scope="col">CVR ORG</th>
<th scope="col">Navn</th>
<th scope="col">Status</th>
<th scope="col"></th>
<th scope="col"></th>
</tr>
</thead>
<tbody>
@foreach (var info in _vInfos)
{
<div class="row mb-1">
<div class="col">
@info.VatNumber
</div>
<div class="col">
@info.Name
</div>
<div class="col">
@info.States[^1].State
</div>
<div class="col">
<button class="btn btn-primary" @onclick="@(() => SelectCompany(info.VatNumber))">OVERFØR</button>
</div>
</div>
<tr>
<td class="align-middle">@info.VatNumber</td>
<td class="align-middle">@info.Name</td>
<td class="align-middle">@info.States[^1].State</td>
<td class="align-middle">
<button class="btn btn-primary" @onclick="@(() => SelectCompany(info.VatNumber, true))">OVERRØR</button>
</td>
<td class="align-middle">
<button class="btn btn-primary" @onclick="@(() => SelectCompany(info.VatNumber, false))">CVR/VAT</button>
</td>
</tr>
}
</div>
</tbody>
</table>
}
<div class="card-body">
<EditForm EditContext="_updateCompany" OnValidSubmit="Update">
<DataAnnotationsValidator/>
<EditForm EditContext="_updateCompany" OnValidSubmit="SubmitUpdate">
<DataAnnotationsValidator/>
<div class="card-body">
<div class="form-group row mb-1">
<label for="vatNumber" class="col-md-2 col-form-label">
<DisplayStateComponent StateClass="@_vatState"></DisplayStateComponent>
CVR/ORG
</label>
<div class="col-md-10">
<InputText id="vatNumber" class="form-control" @bind-Value="CompanyDto.VatNumber"/>
<ValidationMessage For="@(() => CompanyDto.VatNumber)"></ValidationMessage>
<InputText id="vatNumber" class="form-control" @bind-Value="_companyView.VatNumber"/>
<ValidationMessage For="@(() => _companyView.VatNumber)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="name" class="col-md-2 col-form-label">Firmanavn</label>
<div class="col-md-10">
<InputText id="name" class="form-control" @bind-Value="CompanyDto.Name"/>
<ValidationMessage For="@(() => CompanyDto.Name)"></ValidationMessage>
<InputText id="name" class="form-control" @bind-Value="_companyView.Name"/>
<ValidationMessage For="@(() => _companyView.Name)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="address1" class="col-md-2 col-form-label">Adresse</label>
<div class="col-md-10">
<InputText id="address1" class="form-control" @bind-Value="CompanyDto.Address1"/>
<ValidationMessage For="@(() => CompanyDto.Address1)"></ValidationMessage>
<InputText id="address1" class="form-control" @bind-Value="_companyView.Address1"/>
<ValidationMessage For="@(() => _companyView.Address1)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="address2" class="col-md-2 col-form-label">Conavn</label>
<label for="address2" class="col-md-2 col-form-label">Adresse</label>
<div class="col-md-10">
<InputText id="address2" class="form-control" @bind-Value="CompanyDto.Address2"/>
<InputText id="address2" class="form-control" @bind-Value="_companyView.Address2"/>
</div>
</div>
<div class="form-group row mb-1">
<label for="zipCode" class="col-md-2 col-form-label">Postnr</label>
<div class="col-md-10">
<InputText id="zipCode" class="form-control" @bind-Value="CompanyDto.ZipCode"/>
<ValidationMessage For="@(() => CompanyDto.ZipCode)"></ValidationMessage>
<InputText id="zipCode" class="form-control" @bind-Value="_companyView.ZipCode"/>
<ValidationMessage For="@(() => _companyView.ZipCode)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="city" class="col-md-2 col-form-label">Bynavn</label>
<div class="col-md-10">
<InputText id="city" class="form-control" @bind-Value="CompanyDto.City"/>
<ValidationMessage For="@(() => CompanyDto.City)"></ValidationMessage>
<InputText id="city" class="form-control" @bind-Value="_companyView.City"/>
<ValidationMessage For="@(() => _companyView.City)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="phone" class="col-md-2 col-form-label">Telefon</label>
<div class="col-md-10">
<InputText id="phone" class="form-control" @bind-Value="CompanyDto.Phone"/>
<InputText id="phone" class="form-control" @bind-Value="_companyView.Phone"/>
<ValidationMessage For="@(() => _companyView.Phone)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="mobile" class="col-md-2 col-form-label">Mobil</label>
<div class="col-md-10">
<InputText id="mobile" class="form-control" @bind-Value="CompanyDto.Mobile"/>
<InputText id="mobile" class="form-control" @bind-Value="_companyView.Mobile"/>
<ValidationMessage For="@(() => _companyView.Mobile)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="email" class="col-md-2 col-form-label">Email</label>
<div class="col-md-10">
<InputText id="email" class="form-control" @bind-Value="CompanyDto.Email"/>
<InputText id="email" class="form-control" @bind-Value="_companyView.Email"/>
<ValidationMessage For="@(() => _companyView.Email)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="attention" class="col-md-2 col-form-label">Attention</label>
<div class="col-md-10">
<InputText id="attention" class="form-control" @bind-Value="CompanyDto.Attention"/>
<InputText id="attention" class="form-control" @bind-Value="_companyView.Attention"/>
<ValidationMessage For="@(() => _companyView.Attention)"></ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="lastVisit" class="col-form-label col-md-2">
<DisplayStateComponent StateClass="@(_hasFolded ? "the-dead" : Utils.GetVisitState(CompanyDto.NextVisit))"></DisplayStateComponent>
Sidste besøg
<DisplayStateComponent
StateClass="@(_hasFolded ? "the-dead" : Utils.GetVisitState($"{_companyView.NextVisit}"))">
</DisplayStateComponent>
Besøgt
</label>
<div class="col-md-4">
<InputDate id="lastVisit" class="form-control" @bind-Value="@LastVisit"/>
<InputDate id="lastVisit" class="form-control" @bind-Value="@_lastVisit"/>
</div>
<label for="nextVisit" class="col-form-label col-md-2">Næste besøg</label>
<div class="col-md-4">
<InputDate id="nextVisit" class="form-control" @bind-Value="@(NextVisit)"/>
<InputDate id="nextVisit" class="form-control" @bind-Value="@(_nextVisit)"/>
<ValidationMessage For="@(() => _companyView.NextVisit)">Dato kan ikke vær før sidste besøg</ValidationMessage>
</div>
</div>
<div class="form-group row mb-1">
<label for="interval" class="col-form-label col-md-2">Interval (uger)</label>
<div class="col-md-4">
<InputNumber id="interval" class="form-control" @bind-Value="CompanyDto.Interval"/>
<InputNumber id="interval" class="form-control" @bind-Value="_companyView.Interval"/>
<ValidationMessage For="@(() => _companyView.Interval)"></ValidationMessage>
</div>
</div>
<div class="row mb-1">
<div class="col-sm-2 col-md-2 text-right">
<button type="button" class="btn btn-danger">SLET</button>
</div>
<div class="col-sm-2 col-md-2 text-right">
<button type="submit" class="btn btn-success">GEM</button>
</div>
</div>
</EditForm>
</div>
<div class="card-footer">
<div class="row">
<div class="col">
<a class="btn btn-primary" href="/company/@CompanyDto.CompanyId">Tilbage</a>
</div>
</div>
</div>
<div class="card-footer">
<div class="row">
<div class="col col-md-2">
<button type="button" class="btn btn-warning">SKJUL</button>
</div>
<div class="col col-md-2">
<button type="button" class="btn btn-danger">SLET</button>
</div>
<div class="col col-md-2">
<button type="submit" class="btn btn-success">GEM</button>
</div>
<div class="col col-md-2">
<a class="btn btn-primary" href="/company/@CompanyId">Tilbage</a>
</div>
</div>
</div>
</EditForm>
</div>
}
else

View file

@ -43,137 +43,181 @@ public partial class CompanyUpdate : IDisposable
[Inject] public ILocalStorageService StorageService { get; set; }
[Parameter] public string Account { get; set; } = "";
[Parameter] public string CompanyId { get; set; } = "";
private CompanyDto CompanyDto { get; set; }
private CompanyDto _companyView { get; set; } = new();
private EditContext _updateCompany { get; set; }
private List<VirkRegInfo> VInfos { get; set; } = new();
private List<VirkRegInfo> _vInfos { get; set; } = new();
private VirkRegInfo _virkRegInfo { get; set; } = new();
private DateTime LastVisit { get; set; }
private DateTime NextVisit { get; set; }
private DateTime _lastVisit { get; set; }
private DateTime _nextVisit { get; set; }
private string _vatState { get; set; } = "the-ugly";
private VatUtils _vatUtils { get; set; }
private VatAddress vatAddress = new();
private bool validVat;
private bool _validVat;
private bool _hasFolded;
private bool _formInvalid = true;
private string _orgVat;
private readonly JsonSerializerOptions _options = new ()
{
PropertyNameCaseInsensitive = true
};
protected override async Task OnInitializedAsync()
{
_vatUtils = new VatUtils();
Interceptor.RegisterEvent();
Interceptor.RegisterBeforeSendEvent();
CompanyDto = await CompanyRepo.GetCompanyById(CompanyId);
LastVisit = DateTime.Parse(CompanyDto.LastVisit);
NextVisit = DateTime.Parse(CompanyDto.NextVisit);
_updateCompany = new EditContext(CompanyDto);
var ux = await StorageService.GetItemAsync<UserInfoView>("_xu");
if (string.IsNullOrWhiteSpace(CompanyDto.CountryCode))
{
CompanyDto.CountryCode = ux.CountryCode;
}
Interceptor.RegisterEvent();
Interceptor.RegisterBeforeSendEvent();
if(CompanyDto.HasFolded == 1)
_companyView = await CompanyRepo.GetCompanyById(CompanyId);
_orgVat = _companyView.VatNumber;
_companyView.CountryCode = ux.CountryCode.ToLower();
vatAddress = PrepareVatAddress(_companyView);
if (_companyView.Interval == 0)
_companyView.Interval = 8;
_lastVisit = DateTime.Parse(_companyView.LastVisit);
_nextVisit = DateTime.Parse(_companyView.NextVisit);
if (!_companyView.ValidDateSpan())
_nextVisit = _lastVisit.AddDays(_companyView.Interval * 7);
if(_companyView.HasFolded == 1)
{
_hasFolded = true;
_vatState = "the-dead";
}
else
{
_vatState = _vatUtils.ValidateFormat(CompanyDto.CountryCode, CompanyDto.VatNumber) ? "the-good" : "the-draw";
if (_companyView.ValidVat == 0 && !string.IsNullOrWhiteSpace(_companyView.VatNumber))
_companyView.ValidVat = VatUtils.ValidateFormat(_companyView.CountryCode, _companyView.VatNumber) ? 1 : 0;
_validVat = _companyView.ValidVat == 1;
_vatState = _companyView.ValidVat == 1 ? "the-good" : "the-draw";
}
vatAddress = PrepareVatAddress();
_updateCompany = new EditContext(_companyView);
_updateCompany.OnFieldChanged += HandleFieldChanged;
_updateCompany.OnValidationStateChanged += ValidationChanged;
}
private VatAddress PrepareVatAddress()
private void HandleFieldChanged(object sender, FieldChangedEventArgs e)
{
var digits = "123456789".ToCharArray();
var pos1 = CompanyDto.Address1.IndexOfAny(digits);
if (pos1 > 0)
_nextVisit = _lastVisit.AddDays(_companyView.Interval * 7);
_companyView.LastVisit = $"{_lastVisit:yyyy-MM-dd}";
_companyView.NextVisit = $"{_nextVisit:yyyy-MM-dd}";
if (!VatUtils.ValidateFormat(_companyView.CountryCode, _companyView.VatNumber) ||
!_companyView.ValidDateSpan())
{
return new VatAddress
{
ZipCode = CompanyDto.ZipCode,
StreetName = CompanyDto.Address1[..pos1],
HouseNumber = Regex.Replace(CompanyDto.Address1[pos1..], "[^0-9]", "")
};
_formInvalid = true;
}
var pos2 = CompanyDto.Address2.IndexOfAny(digits);
if (pos2 > 0)
else
{
return new VatAddress
{
ZipCode = CompanyDto.ZipCode,
StreetName = CompanyDto.Address2[..pos2],
HouseNumber = Regex.Replace(CompanyDto.Address2[pos2..], "[^0-9]", "")
};
_formInvalid = !_updateCompany.Validate();
}
return vatAddress;
StateHasChanged();
}
private async Task Update()
private void ValidationChanged(object sender, ValidationStateChangedEventArgs e)
{
if (!string.IsNullOrWhiteSpace(CompanyDto.VatNumber) && !_vatUtils.ValidateFormat(CompanyDto.CountryCode, CompanyDto.VatNumber))
_formInvalid = true;
_updateCompany.OnFieldChanged -= HandleFieldChanged;
_updateCompany = new EditContext(_companyView);
_updateCompany.OnFieldChanged += HandleFieldChanged;
_updateCompany.OnValidationStateChanged -= ValidationChanged;
}
private async Task SubmitUpdate()
{
if (!VatUtils.ValidateFormat(_companyView.CountryCode, _companyView.VatNumber))
{
ToastService.ShowError($"CVR/VAT/ORG nummer er ugyldig.");
StateHasChanged();
return;
}
CompanyDto.LastVisit = $"{LastVisit:yyyy-MM-dd}";
CompanyDto.NextVisit = $"{NextVisit:yyyy-MM-dd}";
_companyView.LastVisit = $"{_lastVisit:yyyy-MM-dd}";
_companyView.NextVisit = $"{_nextVisit:yyyy-MM-dd}";
_companyView.IsHidden = 0;
if (_companyView.VatNumber != _orgVat)
_companyView.UpdateErpVat = 1;
Console.WriteLine(JsonSerializer.Serialize(CompanyDto));
await CompanyRepo.UpdateCompany(CompanyDto);
ToastService.ShowSuccess($"'{CompanyDto!.Name}' er opdateret.");
Navigation.NavigateTo($"/company/{CompanyDto.CompanyId}");
}
private async Task GetInfoFromVat(string vatNumber)
{
if (string.IsNullOrWhiteSpace(vatNumber) || !_vatUtils.ValidateFormat(CompanyDto.CountryCode, CompanyDto.VatNumber))
var x = JsonSerializer.Serialize(_companyView, _options);
Logger.LogInformation(x);
var success = await CompanyRepo.UpdateCompany(CompanyId, _companyView );
if (success)
{
ToastService.ShowError($"CVR er ugyldigt eller mangler");
return;
}
VInfos = await VatInfoLookupService
.QueryVirkRegistry(
new VirkParams
{
VatNumber = vatNumber.Trim()
});
if (string.IsNullOrWhiteSpace(VInfos[0].VatNumber))
{
ToastService.ShowError($"CVR nummer '{vatNumber}' findes ikke.");
return;
ToastService.ShowSuccess("Check");
Navigation.NavigateTo($"/company/{CompanyId}");
}
}
private async Task GetInfoFromAddress(VatAddress address)
{
VInfos = await VatInfoLookupService.QueryVirkRegistry(
_vInfos = await VatInfoLookupService.QueryVirkRegistry(
new VirkParams
{
StreetName = address.StreetName,
HouseNumber = address.HouseNumber,
ZipCode = address.ZipCode
});
if (!VInfos.Any())
if (!_vInfos.Any())
{
ToastService.ShowError($"Ingen virksomheder fundet.");
}
}
private void SelectCompany(string vatNumber)
private void SelectCompany(string vatNumber, bool syncAll)
{
_virkRegInfo = (from x in VInfos where x.VatNumber == vatNumber select x).First();
CompanyDto.Name = _virkRegInfo.Name;
CompanyDto.Address1 = _virkRegInfo.CoName;
CompanyDto.Address2 = _virkRegInfo.Address;
CompanyDto.ZipCode = _virkRegInfo.ZipCode;
CompanyDto.City = _virkRegInfo.City;
CompanyDto.VatNumber = _virkRegInfo.VatNumber;
_virkRegInfo = (from x in _vInfos where x.VatNumber == vatNumber select x).First();
if (syncAll)
{
_companyView.VatNumber = _virkRegInfo.VatNumber;
_companyView.Name = _virkRegInfo.Name;
_companyView.Address1 = _virkRegInfo.CoName;
_companyView.Address2 = _virkRegInfo.Address;
_companyView.ZipCode = _virkRegInfo.ZipCode;
_companyView.City = _virkRegInfo.City;
}
else
{
_companyView.VatNumber = _virkRegInfo.VatNumber;
}
_vInfos = new List<VirkRegInfo>();
StateHasChanged();
}
public void Dispose()
{
Interceptor.DisposeEvent();
_updateCompany.OnFieldChanged -= HandleFieldChanged;
_updateCompany.OnValidationStateChanged -= ValidationChanged;
}
private static VatAddress PrepareVatAddress(CompanyDto model)
{
var digits = "123456789".ToCharArray();
var pos1 = model.Address1.IndexOfAny(digits);
if (pos1 > 0)
{
return new VatAddress
{
ZipCode = model.ZipCode,
StreetName = model.Address1[..pos1],
HouseNumber = Regex.Replace(model.Address1[pos1..], "[^0-9]", "")
};
}
var pos2 = model.Address2.IndexOfAny(digits);
if (pos2 > 0)
{
return new VatAddress
{
ZipCode = model.ZipCode,
StreetName = model.Address2[..pos2],
HouseNumber = Regex.Replace(model.Address2[pos2..], "[^0-9]", "")
};
}
return new VatAddress();
}
}

View file

@ -21,56 +21,67 @@
@using Wonky.Client.Helpers
@attribute [Authorize(Roles = "Adviser")]
@if (CompanyDto != null)
@if (_companyDto != null)
{
<div class="card">
<div class="card-header">
<div class="h2"><img src="gravestone.png" class="img-fluid" style="float:left;width:48px;height:48px;display:@(_hasFolded ? "block" : "none")" alt="tombstone"/> @CompanyDto.Name</div>
<div class="h2">
<img src="gravestone.png" class="img-fluid"
style="float:left;width:48px;height:48px;display:@(_companyDto.HasFolded == 1 ? "block" : "none")" alt="tombstone"/>
@_companyDto.Name
</div>
</div>
<div class="card-body">
<table class="table table-sm table-striped table-bordered">
<tbody>
<tr>
<th scope="row">Konto</th>
<td colspan="2">@CompanyDto.Account</td>
<td colspan="2">@_companyDto.Account</td>
</tr>
<tr>
<th scope="row">Conavn</th>
<td colspan="2">@CompanyDto.Address1</td>
<td colspan="2">@_companyDto.Address1</td>
</tr>
<tr>
<th scope="row">Adresse</th>
<td colspan="2">@CompanyDto.Address2</td>
<td colspan="2">@_companyDto.Address2</td>
</tr>
<tr>
<th scope="row">Postnummer</th>
<td colspan="2">@CompanyDto.ZipCode</td>
<td colspan="2">@_companyDto.ZipCode</td>
</tr>
<tr>
<th scope="row">Bynavn</th>
<td colspan="2">@CompanyDto.City</td>
<td colspan="2">@_companyDto.City</td>
</tr>
<tr>
<th scope="row">CVR</th>
<td class="state"><DisplayStateComponent StateClass="@_vatState"></DisplayStateComponent></td>
<td>@CompanyDto.VatNumber</td>
<td class="state">
<DisplayStateComponent StateClass="@_vatState"></DisplayStateComponent>
</td>
<td>@_companyDto.VatNumber</td>
</tr>
<tr>
<th scope="row">Telefon</th>
<td colspan="2">@CompanyDto.Phone</td>
<td colspan="2">@_companyDto.Phone</td>
</tr>
<tr>
<th scope="row">Email</th>
<td colspan="2">@CompanyDto.Email</td>
<td colspan="2">@_companyDto.Email</td>
</tr>
<tr>
<th scope="row">Sidste besøg</th>
<td colspan="2">@CompanyDto.LastVisit</td>
<td colspan="2">@_companyDto.LastVisit</td>
</tr>
<tr>
<th scope="row">Næste besøg</th>
<td class="state"><DisplayStateComponent StateClass="@(_hasFolded ? "the-dead" : Utils.GetVisitState(CompanyDto.NextVisit))"></DisplayStateComponent></td>
<td>@CompanyDto.NextVisit</td>
<td class="state">
<DisplayStateComponent
StateClass="@(_companyDto.HasFolded == 1
? "the-dead" : Utils.GetVisitState(_companyDto.NextVisit))">
</DisplayStateComponent>
</td>
<td>@_companyDto.NextVisit</td>
</tr>
</tbody>
</table>
@ -78,14 +89,16 @@
<div class="card-footer">
<div class="d-flex align-items-end">
<a class="btn btn-primary mx-2" href="/companies">Tilbage</a>
<a class="btn btn-primary mx-2" href="/company/@(CompanyDto.CompanyId)/update">Rediger</a>
@if (_vatInvalid || string.IsNullOrWhiteSpace(CompanyDto.Address1))
<a class="btn btn-primary mx-2" href="/company/@(_companyDto.CompanyId)/update">Rediger</a>
@if (_vatInvalid || string.IsNullOrWhiteSpace(_companyDto.Address1))
{
<a type="button" class="btn btn-primary mx-2 disabled" aria-disabled="true">Aktivitet</a>
<a type="button" class="btn btn-primary mx-2 disabled" aria-disabled="true">Kanvas/Nulsalg</a>
<a type="button" class="btn btn-primary mx-2 disabled" aria-disabled="true">Besøg</a>
}
else
{
<a type="button" class="btn btn-primary mx-2" href="/company/@CompanyDto.CompanyId/activity">Aktivitet</a>
<a type="button" class="btn btn-primary mx-2 disabled" aria-disabled="true">Kanvas/Nulsalg</a>
<a type="button" class="btn btn-primary mx-2" href="/company/@_companyDto.CompanyId/activity">Aktivittet</a>
}
</div>
</div>

View file

@ -15,6 +15,7 @@
using System;
using System.Threading.Tasks;
using Blazored.LocalStorage;
using Blazored.Toast.Services;
using Wonky.Client.HttpInterceptors;
using Wonky.Client.HttpRepository;
@ -23,6 +24,7 @@ using Wonky.Client.Helpers;
using Wonky.Client.Services;
using Wonky.Entity.DTO;
using Wonky.Entity.Models;
using Wonky.Entity.Views;
namespace Wonky.Client.Pages;
@ -34,28 +36,23 @@ public partial class CompanyView : IDisposable
[Inject] public ICompanyHttpRepository CompanyRepo { get; set; }
[Inject] public HttpInterceptorService Interceptor { get; set; }
[Inject] public VatInfoLookupService VatInfoLookup { get; set; }
[Inject] public ILocalStorageService Storage { get; set; }
[Parameter] public string CompanyId { get; set; } = "";
private CompanyDto CompanyDto { get; set; } = new ();
private string _vatState { get; set; } = "the-dead";
private bool _hasFolded;
private bool _vatInvalid;
private VatUtils _vatUtils { get; set; }
private CompanyDto _companyDto = new ();
private string _vatState = "the-dead";
private bool _vatInvalid = true;
protected override async Task OnInitializedAsync()
{
_vatUtils = new VatUtils();
Interceptor.RegisterEvent();
Interceptor.RegisterBeforeSendEvent();
CompanyDto = await CompanyRepo.GetCompanyById(CompanyId);
if(CompanyDto.HasFolded == 1)
var xu = await Storage.GetItemAsync<UserInfoView>("_xu");
_companyDto = await CompanyRepo.GetCompanyById(CompanyId);
_companyDto.CountryCode = xu.CountryCode;
if (_companyDto.HasFolded == 0)
{
_hasFolded = true;
}
else
{
_vatInvalid = !_vatUtils.ValidateFormat(CompanyDto.CountryCode, CompanyDto.VatNumber);
_vatState = _vatUtils.ValidateFormat(CompanyDto.CountryCode, CompanyDto.VatNumber) ? "the-good" : "the-draw";
_vatInvalid = !VatUtils.ValidateFormat(_companyDto.CountryCode, _companyDto.VatNumber);
_vatState = _vatInvalid ? "the-draw" : "the-good";
}
}

View file

@ -55,7 +55,7 @@
<label for="activityType" class="col-md-2 col-form-label">Kontakt</label>
<div class="col-md-4">
<select id="activityType" class="form-select" @bind-Value="@_poDraft.ActivityTypeEnum" @bind-Value:event="oninput" @onchange="CheckActivity">
<option value="" selected>"IKKE VALGT"</option>
<option value="" selected disabled>"IKKE VALGT"</option>
<option value="onSite">Besøg</option>
<option value="phone">Telefon</option>
</select>
@ -64,7 +64,7 @@
<label for="statusType" class="col-md-2 col-form-label">Status</label>
<div class="col-md-4">
<select id="statusType" class="form-select" @bind-Value="@_poDraft.ActivityStatusEnum" @bind-Value:event="oninput" @onchange="CheckStatus">
<option value="" selected>"IKKE VALGT"</option>
<option value="" selected disabled>"IKKE VALGT"</option>
<option value="noSale">Ingen salg</option>
<option value="order">Bestilling</option>
<option value="quote">Tilbud</option>
@ -247,7 +247,7 @@
<li class="list-group-item d-flex justify-content-between align-items-end">
<div class="text-sm-start px-2">@rate.Quantity</div>
<div class="text-sm-end">@rate.Rate</div>
<button class="btn btn-primary btn-sm" @onclick="@(() => SelectItem(item.ItemId, rate.Quantity, rate.Rate))">Vælg</button>
<button type="button" class="btn btn-primary btn-sm" @onclick="@(() => SelectItem(item.ItemId, rate.Quantity, rate.Rate))">Vælg</button>
</li>
}
</ul>
@ -309,7 +309,7 @@
<input type="checkbox" class="form-check" @bind-value="@Sas"/>
</div>
<div class="col">
<button class="btn btn-info" @onclick="@(() => AddItem(_selectedItem))">Læg til</button>
<button type="button" class="btn btn-info" @onclick="@(() => AddItem(_selectedItem))">Læg til</button>
</div>
</div>
</div>
@ -347,7 +347,7 @@
<input type="checkbox" checked="@cItem.Sas" disabled/>
</td>
<td>
<button class="btn btn-warning" @onclick="@(() => RemoveItem(@cItem))">Slet</button>
<button type="button" class="btn btn-warning" @onclick="@(() => RemoveItem(@cItem))">Slet</button>
</td>
</tr>
}
@ -366,7 +366,7 @@
<div class="card-footer">
<div class="row">
<div class="col">
<button class="btn btn-danger" @onclick="@DeleteDraft" disabled="@(DraftStateProvider.Draft.Items.Count == 0)">Slet kladde</button>
<button type="button" class="btn btn-danger" @onclick="@DeleteDraft" disabled="@(DraftStateProvider.Draft.Items.Count == 0)">Slet kladde</button>
</div>
</div>
</div>
@ -375,10 +375,11 @@
</div>
</div>
<div class="row mt-2 mb-2">
<div class="col">
<div class="col-md-6"></div>
<div class="col-md-2">
<a class="btn btn-primary" href="/company/@NgCompany.CompanyId">Tilbage</a>
</div>
<div class="col">
<div class="col-md-2">
@* <button type="submit" class="btn btn-success" disabled="@InvalidActivity">Gem</button> *@
<button type="submit" class="btn btn-success">OK</button>
</div>

View file

@ -63,7 +63,6 @@ public partial class CrmActivityCreate : IDisposable
private bool InvalidCanvas { get; set; } = true;
private bool InvalidDate { get; set; } = true;
private UserInfoView Ux { get; set; } = new();
private VatUtils _vatUtils { get; set; }
private DateTime _draftDate { get; set; }
protected override void OnParametersSet()
@ -76,7 +75,6 @@ public partial class CrmActivityCreate : IDisposable
Interceptor.RegisterEvent();
Interceptor.RegisterBeforeSendEvent();
_vatUtils = new VatUtils();
_prefs = await UserPrefs.GetPreferences();
_paging.SearchColumn = _prefs.ItemSearch;
_paging.PageSize = Convert.ToInt32(_prefs.PageSize);
@ -84,10 +82,6 @@ public partial class CrmActivityCreate : IDisposable
Ux = await StorageService.GetItemAsync<UserInfoView>("_xu");
NgCompany = await CompanyRepo.GetCompanyById(CompanyId);
DraftContext = new EditContext(_poDraft);
DraftContext.OnFieldChanged += HandleFieldChanged;
DraftContext.OnValidationStateChanged += ValidationChanged;
// set up identification
_poDraft.SalesHeadId = "";
_poDraft.CompanyId = CompanyId;
@ -114,6 +108,11 @@ public partial class CrmActivityCreate : IDisposable
_poDraft.DlvAddress2 = NgCompany.Address2;
_poDraft.DlvZipCode = NgCompany.ZipCode;
_poDraft.DlvCity = NgCompany.City;
DraftContext = new EditContext(_poDraft);
DraftContext.OnFieldChanged += HandleFieldChanged;
DraftContext.OnValidationStateChanged += ValidationChanged;
}
private async Task CreateActivity()
@ -152,7 +151,7 @@ public partial class CrmActivityCreate : IDisposable
await StorageService.SetItemAsync(CompanyId, _poDraft);
var result = await ActivityRepo.CreateActivity(_poDraft).ConfigureAwait(true);
var result = await ActivityRepo.CreateActivity(_poDraft);
Console.WriteLine(JsonSerializer.Serialize(result));
ToastService.ShowSuccess($"{result.Message}.");
NavigationManager.NavigateTo($"/companies");
@ -270,20 +269,26 @@ public partial class CrmActivityCreate : IDisposable
private void HandleFieldChanged(object sender, FieldChangedEventArgs e)
{
_poFormInvalid = !DraftContext.Validate();
InvalidCanvas = InvalidActivityType || InvalidDate;
InvalidActivity = InvalidActivityType
|| _poFormInvalid
|| DraftStateProvider.Draft.Items.Count == 0
|| InvalidDate
|| (_poDraft.ActivityStatusEnum == "offer" && string.IsNullOrWhiteSpace(_poDraft.EMail));
if (InvalidCanvas || InvalidActivity)
{
_poFormInvalid = true;
return;
}
_poFormInvalid = !DraftContext.Validate();
StateHasChanged();
}
private void ValidationChanged(object sender, ValidationStateChangedEventArgs e)
{
if (!string.IsNullOrEmpty(_poDraft.VatNumber))
{
if(!_vatUtils.ValidateFormat(NgCompany.CountryCode, _poDraft.VatNumber))
if(!VatUtils.ValidateFormat(NgCompany.CountryCode, _poDraft.VatNumber))
ToastService.ShowWarning("CVR / ORG nummer er ikke et gyldigt registreringsnummer");
}
if (string.IsNullOrEmpty(_poDraft.ActivityTypeEnum))

View file

@ -13,13 +13,17 @@
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see [https://www.gnu.org/licenses/agpl-3.0.en.html]
//
*@@using Wonky.Client.Components
*@
@using Wonky.Client.Components
@using Microsoft.Extensions.Options
@using Wonky.Entity.Configuration
@page "/info"
<AuthorizeView>
<Authorized>
<div class="row mb-2">
<div class="col text-center">
<img class="grumpy-coder" src="grumpy-coder.png" alt="Wonky Logo"/>
<img class="grumpy-coder" src="@_app?.Image" alt="Wonky Logo"/>
</div>
</div>
<div class="row">
@ -37,4 +41,15 @@
</div>
</Authorized>
</AuthorizeView>
@code{
[Inject] IOptions<AppInfo>? AppInfo { get; set; }
private AppInfo? _app;
protected override void OnInitialized()
{
_app = AppInfo?.Value;
}
}

View file

@ -31,8 +31,8 @@
<div class="card mb-3">
<div class="row g-0 d-flex align-items-center">
<div class="col-lg-4 d-none d-lg-flex">
<img src="grumpy-coder.png" alt="Innotec Danmark"
class="w-100 rounded-t-5 rounded-tr-lg-0 rounded-bl-lg-5"/>
<img src="grumpy-coder.png" alt="Innotec Danmark" style="max-height: 300px; max-width: 300px"
class="img-fluid w-100 rounded-t-5 rounded-tr-lg-0 rounded-bl-lg-5"/>
</div>
<div class="col-lg-8">
<div class="card-body py-5 px-md-5">

View file

@ -28,7 +28,7 @@ public partial class Logout
protected override async Task OnInitializedAsync()
{
await AuthenticationService.Logout().ConfigureAwait(true);
await AuthenticationService.Logout();
NavigationManager.NavigateTo("/");
}

View file

@ -28,9 +28,9 @@
<div class="card-body">
<table class="table">
<thead>
<th scope="col">Dagtype</th>
<th scope="col">Begyndt</th>
<th scope="col">Afsluttet</th>
<th scope="col">Dag Type</th>
<th scope="col">Dag Begyndt</th>
<th scope="col">Dag Afsluttet</th>
</thead>
<tbody>
<tr>

View file

@ -48,7 +48,7 @@ public partial class SalesReport
? DateTime.Now
: DateTime.Parse(_prefs.WorkDate);
_reportDto.CheckOut = _reportDto.CheckIn;
await UserPrefs.SetWorkDate(_reportDto.CheckIn).ConfigureAwait(true);
await UserPrefs.SetWorkDate(_reportDto.CheckIn);
}
private void HandleFieldChanged(object sender, FieldChangedEventArgs e)
@ -73,10 +73,10 @@ public partial class SalesReport
private async Task GetActivities()
{
Activities = null;
await UserPrefs.SetWorkDate(_reportDto.CheckIn).ConfigureAwait(true);
await UserPrefs.SetWorkDate(_reportDto.CheckIn);
Activities = await ActivityRepo
.GetActivities($"{_reportDto.CheckIn:yyyy-MM-dd}")
.ConfigureAwait(true);
;
}
public void Dispose()

View file

@ -46,6 +46,7 @@ builder.Services.AddBlazoredToast();
builder.Services.AddHttpClientInterceptor();
builder.Services.Configure<ApiConfig>(builder.Configuration.GetSection("ApiConfig"));
builder.Services.Configure<AppInfo>(builder.Configuration.GetSection("AppInfo"));
builder.Services.AddScoped<ICompanyHttpRepository, CompanyHttpRepository>();
builder.Services.AddScoped<ISalesItemHttpRepository, SalesItemHttpRepository>();

View file

@ -54,11 +54,9 @@ namespace Wonky.Client.Services
};
var response = await _client
.PostAsync(_apiConfig.Value.TokenPath, new FormUrlEncodedContent(credentials))
.ConfigureAwait(true);
.PostAsync(_apiConfig.Value.TokenPath, new FormUrlEncodedContent(credentials));
var resContent = await response.Content.ReadAsStringAsync()
.ConfigureAwait(true);
var resContent = await response.Content.ReadAsStringAsync();
// if not success - return error status
if (!response.IsSuccessStatusCode)
@ -69,93 +67,56 @@ namespace Wonky.Client.Services
// process response content
var data = JsonSerializer.Deserialize<AuthResponseView>(resContent, _options);
await _localStorage.SetItemAsync("_xa", data.AccessToken)
.ConfigureAwait(true);
await _localStorage.SetItemAsync("_xr", data.RefreshToken)
.ConfigureAwait(true);
await _localStorage.SetItemAsync("_xe",
(int)DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1)).TotalSeconds + data.ExpiresIn - 60)
.ConfigureAwait(true);
await _localStorage.SetItemAsync("_xa", data.AccessToken);
await _localStorage.SetItemAsync("_xr", data.RefreshToken);
await _localStorage.SetItemAsync("_xe", (int)DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1)).TotalSeconds + data.ExpiresIn - 60);
// set default request headers using access_token
_client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("bearer", data.AccessToken);
var userInfo = await UserInfo();
await _localStorage.SetItemAsync("_xu", userInfo)
.ConfigureAwait(true);
await _localStorage.SetItemAsync("_xu", userInfo);
// notify system on state change
((AuthStateProvider)_authStateProvider).NotifyUserAuthenticationAsync(data.AccessToken);
data.IsSuccess = true;
return data;
}
public async Task<string> RefreshToken()
{
var refreshToken = await _localStorage.GetItemAsync<string>("_xr")
.ConfigureAwait(true);
var refreshToken = await _localStorage.GetItemAsync<string>("_xr");
var credentials = new Dictionary<string, string>
{
["grant_type"] = "refresh_token",
["refresh_token"] = refreshToken
};
var response = await _client.PostAsync(_apiConfig.Value.TokenPath, new FormUrlEncodedContent(credentials))
.ConfigureAwait(true);
var response = await _client.PostAsync(_apiConfig.Value.TokenPath, new FormUrlEncodedContent(credentials));
if (!response.IsSuccessStatusCode) return string.Empty;
var resContent = await response.Content.ReadAsStringAsync()
.ConfigureAwait(true);
var resContent = await response.Content.ReadAsStringAsync();
var data = JsonSerializer.Deserialize<AuthResponseView>(resContent, _options);
// set default request headers using access_token
_client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("bearer", data.AccessToken);
await _localStorage.SetItemAsync("_xa", data.AccessToken)
.ConfigureAwait(true);
await _localStorage.SetItemAsync("_xr", data.RefreshToken)
.ConfigureAwait(true);
await _localStorage.SetItemAsync("_xe",
(int)DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1)).TotalSeconds + data.ExpiresIn - 60)
.ConfigureAwait(true);
await _localStorage.SetItemAsync("_xa", data.AccessToken);
await _localStorage.SetItemAsync("_xr", data.RefreshToken);
await _localStorage.SetItemAsync("_xe", (int)DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1)).TotalSeconds + data.ExpiresIn - 60);
return data.AccessToken;
}
public async Task Logout()
{
await _localStorage.ClearAsync()
.ConfigureAwait(true);
await _localStorage.ClearAsync();
((AuthStateProvider)_authStateProvider).NotifyUserLogout();
_client.DefaultRequestHeaders.Authorization = null;
}
public async Task<UserInfoView> UserInfo(bool write = false)
{
var infoResponse = await _client.GetAsync(_apiConfig.Value.UserInfo)
.ConfigureAwait(true);
var infoContent = await infoResponse.Content.ReadAsStringAsync()
.ConfigureAwait(true);
var infoResponse = await _client.GetAsync(_apiConfig.Value.UserInfo);
var infoContent = await infoResponse.Content.ReadAsStringAsync();
var userInfo = JsonSerializer.Deserialize<UserInfoView>(infoContent, _options);
if(write)
await _localStorage.SetItemAsync("_xu", userInfo).ConfigureAwait(true);
await _localStorage.SetItemAsync("_xu", userInfo);
return userInfo ?? new UserInfoView();
}
}
}

View file

@ -36,8 +36,7 @@ public class RefreshTokenService
public async Task<string> TryRefreshToken()
{
var authState = await _authStateProvider.GetAuthenticationStateAsync()
.ConfigureAwait(true);
var authState = await _authStateProvider.GetAuthenticationStateAsync();
var user = authState.User;
@ -47,7 +46,7 @@ public class RefreshTokenService
var diff = expTime - DateTime.UtcNow;
return diff.TotalMinutes <= 2
? await _authService.RefreshToken()
? await _authService.RefreshToken().ConfigureAwait(true)
: string.Empty;
}

View file

@ -29,8 +29,7 @@ public class UserPreferenceService
WorkDate = $"{workDate:yyyy-MM-dd}"
};
await _localStorageService.SetItemAsync("preferences", newPreferences)
.ConfigureAwait(true);
await _localStorageService.SetItemAsync("preferences", newPreferences);
OnChange?.Invoke(newPreferences);
}
@ -40,63 +39,54 @@ public class UserPreferenceService
var newPreferences = preferences
with { CompanySearch = companySearch };
await _localStorageService.SetItemAsync("preferences", newPreferences)
.ConfigureAwait(true);
await _localStorageService.SetItemAsync("preferences", newPreferences);
OnChange?.Invoke(newPreferences);
}
public async Task SetCompanySort(string companySort)
{
var preferences = await GetPreferences()
.ConfigureAwait(true);
var preferences = await GetPreferences();
var newPreferences = preferences
with { CompanySort = companySort };
await _localStorageService.SetItemAsync("preferences", newPreferences)
.ConfigureAwait(true);
await _localStorageService.SetItemAsync("preferences", newPreferences);
OnChange?.Invoke(newPreferences);
}
public async Task SetItemSearch(string itemSearch)
{
var preferences = await GetPreferences()
.ConfigureAwait(true);
var preferences = await GetPreferences();
var newPreferences = preferences
with { ItemSearch = itemSearch };
await _localStorageService.SetItemAsync("preferences", newPreferences)
.ConfigureAwait(true);
await _localStorageService.SetItemAsync("preferences", newPreferences);
OnChange?.Invoke(newPreferences);
}
public async Task SetItemSort(string itemSort)
{
var preferences = await GetPreferences()
.ConfigureAwait(true);
var preferences = await GetPreferences();
var newPreferences = preferences
with { ItemSort = itemSort };
await _localStorageService.SetItemAsync("preferences", newPreferences)
.ConfigureAwait(true);
await _localStorageService.SetItemAsync("preferences", newPreferences);
OnChange?.Invoke(newPreferences);
}
public async Task SetPageSize(string pageSize)
{
var preferences = await GetPreferences()
.ConfigureAwait(true);
var preferences = await GetPreferences();
var newPreferences = preferences
with { PageSize = pageSize };
await _localStorageService.SetItemAsync("preferences", newPreferences)
.ConfigureAwait(true);
await _localStorageService.SetItemAsync("preferences", newPreferences);
OnChange?.Invoke(newPreferences);
}

View file

@ -29,11 +29,13 @@ public class VatInfoLookupService
private readonly HttpClient _client;
private readonly IOptions<ApiConfig> _apiConfig;
private readonly List<VirkRegInfo> _noData = new() { new VirkRegInfo { Name = "INGEN DATA" } };
private readonly ApiConfig _config;
public VatInfoLookupService(HttpClient client, IOptions<ApiConfig> apiConfig)
{
_client = client;
_apiConfig = apiConfig;
_config = _apiConfig.Value;
}
public async Task<List<VirkRegInfo>> QueryVirkRegistry(VirkParams query)
@ -49,18 +51,53 @@ public class VatInfoLookupService
["houseNumber"] = $"{query.HouseNumber}",
["zipCode"] = $"{query.ZipCode}"
};
var endpoint = QueryHelpers.AddQueryString(_apiConfig.Value.CvrLookup, queryString);
var response = await _client.GetAsync(endpoint)
.ConfigureAwait(true);
var endpoint = QueryHelpers.AddQueryString(_config.BrRegEndpoint, queryString);
var response = await _client.GetAsync(endpoint);
var content = await response.Content.ReadAsStringAsync();
var jsonResult = JsonSerializer.Deserialize<List<VirkRegInfo>>(content, _options);
if (string.IsNullOrWhiteSpace(content) || !jsonResult.Any()) return _noData;
if (!string.IsNullOrWhiteSpace(query.VatNumber)) return jsonResult;
var result = jsonResult
//.Where(x => x.States[^1].State == "NORMAL")
.OrderBy(x => x.Name).ToList();
var content = await response.Content.ReadAsStringAsync()
.ConfigureAwait(true);
return result.Count == 0 ? _noData : result;
}
public async Task<List<VirkRegInfo>> QueryNoVatRegistry(string vatNumber)
{
var queryString = new Dictionary<string, string>
{
["vatNumber"] = $"{vatNumber}"
};
var endpoint = QueryHelpers.AddQueryString(_config.BrRegEndpoint, queryString);
var response = await _client.GetAsync(endpoint);
var content = await response.Content.ReadAsStringAsync();
var jsonResult = JsonSerializer.Deserialize<List<VirkRegInfo>>(content, _options);
if (string.IsNullOrWhiteSpace(content) || !jsonResult.Any()) return _noData;
if (!string.IsNullOrWhiteSpace(query.VatNumber)) return jsonResult;
var result = jsonResult
//.Where(x => x.States[^1].State == "NORMAL")
.OrderBy(x => x.Name).ToList();
return result.Count == 0 ? _noData : result;
}
public async Task<List<VirkRegInfo>> QueryViesVatRegistry(string vatNumber)
{
var queryString = new Dictionary<string, string>
{
["vatNumber"] = $"{vatNumber}"
};
var endpoint = QueryHelpers.AddQueryString(_config.ViesEndpoint, queryString);
var response = await _client.GetAsync(endpoint);
var content = await response.Content.ReadAsStringAsync();
var jsonResult = JsonSerializer.Deserialize<List<VirkRegInfo>>(content, _options);
if (string.IsNullOrWhiteSpace(content) || !jsonResult.Any()) return _noData;
var result = jsonResult
//.Where(x => x.States[^1].State == "NORMAL")
@ -74,7 +111,8 @@ public class VatInfoLookupService
if (!string.IsNullOrWhiteSpace(query.VatNumber))
return true;
return !string.IsNullOrWhiteSpace(query.HouseNumber) && !string.IsNullOrWhiteSpace(query.StreetName) &&
!string.IsNullOrWhiteSpace(query.ZipCode);
return !string.IsNullOrWhiteSpace(query.HouseNumber)
&& !string.IsNullOrWhiteSpace(query.StreetName)
&& !string.IsNullOrWhiteSpace(query.ZipCode);
}
}

View file

@ -17,7 +17,6 @@ using System.Net.Http.Headers;
using System.Security.Claims;
using Blazored.LocalStorage;
using Microsoft.AspNetCore.Components.Authorization;
using Wonky.Entity.DTO;
using Wonky.Entity.Views;
namespace Wonky.Client.Shared
@ -38,31 +37,24 @@ namespace Wonky.Client.Shared
public override async Task<AuthenticationState> GetAuthenticationStateAsync()
{
// fetch token from localStorage
var token = await _storage.GetItemAsync<string>("_xa")
.ConfigureAwait(true);
var token = await _storage.GetItemAsync<string>("_xa");
if (string.IsNullOrEmpty(token))
// return anonymous if empty
return _anonymous;
// create an authorized user
var userInfo = await _storage.GetItemAsync<UserInfoView>("_xu")
.ConfigureAwait(true);
var userInfo = await _storage.GetItemAsync<UserInfoView>("_xu");
if (userInfo == null)
return _anonymous;
// set client authorization header
_client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("bearer", token);
var exp = await _storage.GetItemAsync<string>("_xe")
.ConfigureAwait(true);
var exp = await _storage.GetItemAsync<string>("_xe");
var roles = ExtractRoles(userInfo);
var claims = new List<Claim>
{
new(ClaimTypes.Name, userInfo.FullName),
new(ClaimTypes.Email, userInfo.Email),
new(ClaimTypes.Country, userInfo.CountryCode),
new(ClaimTypes.MobilePhone, userInfo.PhoneNumber),
new(ClaimTypes.Expiration, exp)
};
@ -85,10 +77,10 @@ namespace Wonky.Client.Shared
// create an authorized user
var userInfo = await _storage.GetItemAsync<UserInfoView>("_xu")
.ConfigureAwait(true);
;
var exp = await _storage.GetItemAsync<string>("_xe")
.ConfigureAwait(true);
;
var roles = ExtractRoles(userInfo);
var claims = new List<Claim>

View file

@ -31,7 +31,7 @@ public partial class DraftStateProvider
protected override async Task OnParametersSetAsync()
{
Draft = await LocalStorageService.GetItemAsync<Draft>(Account)
.ConfigureAwait(true);
;
if (Draft == null || Draft.Items.Count == 0)
{
@ -51,13 +51,13 @@ public partial class DraftStateProvider
public async Task SaveChangesAsync()
{
await LocalStorageService.SetItemAsync(Account, Draft)
.ConfigureAwait(true);
;
}
public async Task DeleteDraftAsync()
{
Draft.Items.Clear();
await LocalStorageService.SetItemAsync(Account, Draft)
.ConfigureAwait(true);
;
}
}

View file

@ -6,19 +6,21 @@
"Microsoft": "Information"
}
},
"app": {
"name": "Willy Wonka",
"version": "0.3",
"appInfo": {
"name": "Wonky Client",
"version": "0.3.y",
"isBeta": true,
"image": "grumpy-coder.png"
},
"apiConfig": {
"baseAddress": "https://staging.innotec.dk",
"baseAddress": "https://dev.innotec.dk",
"tokenPath": "token",
"userInfo": "api/auth/userinfo",
"customerEndpoint": "api/v2/crm/companies",
"catalogEndpoint": "api/v2/crm/catalog",
"cvrLookup": "api/v2/services/virk",
"dkEndpointVat": "api/v2/services/virk",
"noEndpointVat": "api/v2/services/brReg",
"seEndpointVat": "api/v2/services/vies",
"glsTrackUrl": "https://www.gls-group.eu/276-I-PORTAL-WEB/content/GLS/DK01/DA/5004.htm?txtAction=71000&txtRefNo=",
"glsId": "",
"activityEndpoint": "api/v2/crm/salesReps/sales"

View file

@ -9,10 +9,10 @@
/* visit / vat state classes */
.state {
width: 8px;
height: 8px;
min-width: 8px;
min-height: 8px;
width: 24px;
height: 24px;
min-width: 24px;
min-height: 24px;
}
.text-inno {
color: #ffaa00;
@ -27,16 +27,21 @@
background-color: orange;
}
.the-ugly {
background-color: red;
background-color: #ff0000;
}
.the-draw {
background-color: purple;
background-color: #9b02fc;
}
.the-dead {
background-color: black;
background-color: #262626;
}
/* end visit state classes*/
.btn.btn-edit {
color: #030303;
background-color: #a2a2ec;
}
html, body {
font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif;
}

View file

@ -24,7 +24,9 @@ public class ApiConfig
public string ImageUpload { get; set; } = "";
public string UserRegistration { get; set; } = "";
public string UserInfo { get; set; } = "";
public string CvrLookup { get; set; } = "";
public string BrRegEndpoint { get; set; } = "";
public string NoEndpointVat { get; set; } = "";
public string ViesEndpoint { get; set; } = "";
public string TokenPath { get; set; } = "";
public string GlsTrackUrl { get; set; } = "";
public string GlsId { get; set; } = "";

View file

@ -0,0 +1,9 @@
namespace Wonky.Entity.Configuration;
public class AppInfo
{
public string Version { get; set; }
public string Name { get; set; }
public string Image { get; set; }
public bool IsBeta { get; set; }
}

View file

@ -21,10 +21,15 @@ namespace Wonky.Entity.DTO;
public class CompanyDto
{
[Required(ErrorMessage = "Navn skal udfyldes")] [MaxLength(100, ErrorMessage = "Du kan højst bruge 100 tegn")] public string Name { get; set; }
[Required(ErrorMessage = "Postnummer skal udfyldes")] [MaxLength(20, ErrorMessage = "Du kan højst bruge 20 tegn")] public string ZipCode { get; set; }
[Required(ErrorMessage = "Bynavn skal udfyldes")] [MaxLength(30, ErrorMessage = "Du kan højst bruge 30 tegn")] public string City { get; set; }
[Required(ErrorMessage = "ORG/VAT/CVR er ikke et gyldigt nummer")] public string VatNumber { get; set; }
[Required(ErrorMessage = "Navn skal udfyldes")] [MaxLength(100, ErrorMessage = "Du kan højst bruge 100 tegn")]
public string Name { get; set; }
[Required(ErrorMessage = "Postnummer skal udfyldes")] [MaxLength(20, ErrorMessage = "Du kan højst bruge 20 tegn")]
public string ZipCode { get; set; }
[Required(ErrorMessage = "Bynavn skal udfyldes")] [MaxLength(30, ErrorMessage = "Du kan højst bruge 30 tegn")]
public string City { get; set; }
[Required(ErrorMessage = "ORG/VAT/CVR er ikke et gyldigt nummer")]
public string VatNumber { get; set; } = "";
public string CompanyId { get; set; } = "";
public string SalesRepId { get; set; } = "";
public string BcId { get; set; } = "";
@ -35,11 +40,20 @@ public class CompanyDto
[MaxLength(20, ErrorMessage = "Du kan højst bruge 20 tegn")] public string Mobile { get; set; } = "";
[MaxLength(80, ErrorMessage = "Du kan højst bruge 80 tegn")] public string Email { get; set; } = "";
[MaxLength(100, ErrorMessage = "Du kan højst bruge 100 tegn")] public string Attention { get; set; } = "";
public string CountryCode { get; set; } = "";
public string LastVisit { get; set; } = "";
public string NextVisit { get; set; } = "";
[Range(1, 52, ErrorMessage = "Angiv interval mellem 1 og 52 uger")]
public int Interval { get; set; } = 8;
public int IsHidden { get; set; }
public int HasFolded { get; set; }
public int IsHidden { get; set; }
public int ValidVat { get; set; }
public string CountryCode { get; set; } = "";
public int UpdateErpVat { get; set; }
public bool ValidDateSpan()
{
var notAllowed = new List<string> {"1970-01-01", "0001-01-01"};
if (notAllowed.Contains(LastVisit) || notAllowed.Contains(NextVisit))
return false;
return DateTime.Parse(LastVisit) < DateTime.Parse(NextVisit);
}
}

View file

@ -1,33 +0,0 @@
// Copyright (C) 2022 FCS Frede's Computer Services.
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see [https://www.gnu.org/licenses/agpl-3.0.en.html]
//
using System.ComponentModel.DataAnnotations;
namespace Wonky.Entity.DTO;
public class CompanyUpdateDto
{
public string CompanyId { get; set; } = "";
[MaxLength(100)] public string Name { get; set; } = "";
[MaxLength(30)] public string City { get; set; } = "";
[MaxLength(20)] public string ZipCode { get; set; } = "";
[MaxLength(20)] public string VatNumber { get; set; } = "";
[MaxLength(30)] public string Phone { get; set; } = "";
[MaxLength(100)] public string Address1 { get; set; } = "";
[MaxLength(50)] public string Address2 { get; set; } = "";
[MaxLength(100)] public string Attention { get; set; } = "";
[MaxLength(80)] public string Email { get; set; } = "";
[MaxLength(30)] public string Mobile { get; set; } = "";
}