Gestiti allegati nel form
This commit is contained in:
@@ -1,10 +1,14 @@
|
||||
using SteUp.Shared.Core.Dto;
|
||||
using SteUp.Shared.Core.Helpers;
|
||||
using SteUp.Shared.Core.Interface.System;
|
||||
|
||||
namespace SteUp.Maui.Core.Services;
|
||||
|
||||
public class AttachedService : IAttachedService
|
||||
{
|
||||
private static string AttachedRoot =>
|
||||
Path.Combine(FileSystem.CacheDirectory, "attached");
|
||||
|
||||
public async Task<AttachedDto?> SelectImageFromCamera()
|
||||
{
|
||||
var cameraPerm = await Permissions.RequestAsync<Permissions.Camera>();
|
||||
@@ -18,6 +22,7 @@ public class AttachedService : IAttachedService
|
||||
try
|
||||
{
|
||||
result = await MediaPicker.Default.CapturePhotoAsync();
|
||||
result?.FileName = $"img_{DateTime.Now:ddMMyyy_hhmmss}{result.FileName[result.FileName.IndexOf('.')..]}";
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
@@ -29,17 +34,16 @@ public class AttachedService : IAttachedService
|
||||
return result is null ? null : await ConvertToDto(result, AttachedDto.TypeAttached.Image);
|
||||
}
|
||||
|
||||
public async Task<AttachedDto?> SelectImageFromGallery()
|
||||
public async Task<List<AttachedDto>?> SelectImageFromGallery()
|
||||
{
|
||||
List<FileResult>? resultList;
|
||||
var storagePerm = await Permissions.RequestAsync<Permissions.StorageRead>();
|
||||
if (storagePerm != PermissionStatus.Granted)
|
||||
return null;
|
||||
|
||||
FileResult? result;
|
||||
|
||||
try
|
||||
{
|
||||
result = await MediaPicker.Default.PickPhotoAsync();
|
||||
resultList = await MediaPicker.Default.PickPhotosAsync();
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
@@ -48,7 +52,15 @@ public class AttachedService : IAttachedService
|
||||
return null;
|
||||
}
|
||||
|
||||
return result is null ? null : await ConvertToDto(result, AttachedDto.TypeAttached.Image);
|
||||
if (resultList.IsNullOrEmpty()) return null;
|
||||
|
||||
List<AttachedDto> returnList = [];
|
||||
foreach (var fileResult in resultList)
|
||||
{
|
||||
returnList.Add(await ConvertToDto(fileResult, AttachedDto.TypeAttached.Image));
|
||||
}
|
||||
|
||||
return returnList;
|
||||
}
|
||||
|
||||
private static async Task<AttachedDto> ConvertToDto(FileResult file, AttachedDto.TypeAttached type)
|
||||
@@ -86,6 +98,15 @@ public class AttachedService : IAttachedService
|
||||
return filePath;
|
||||
}
|
||||
|
||||
public Task CleanTempStorageAsync(CancellationToken ct = default)
|
||||
{
|
||||
return Task.Run(() =>
|
||||
{
|
||||
if (Directory.Exists(AttachedRoot))
|
||||
Directory.Delete(AttachedRoot, true);
|
||||
}, ct);
|
||||
}
|
||||
|
||||
public Task OpenFile(string fileName, string filePath)
|
||||
{
|
||||
#if IOS
|
||||
@@ -98,4 +119,31 @@ public class AttachedService : IAttachedService
|
||||
});
|
||||
#endif
|
||||
}
|
||||
|
||||
public async Task<(string originalUrl, string thumbUrl)> SaveAndCreateThumbAsync(
|
||||
byte[] bytes, string fileName, CancellationToken ct = default)
|
||||
{
|
||||
Directory.CreateDirectory(AttachedRoot);
|
||||
|
||||
var id = Guid.NewGuid().ToString("N");
|
||||
var safeName = SanitizeFileName(fileName);
|
||||
|
||||
var originalFile = $"{id}_{safeName}";
|
||||
var thumbFile = $"{id}_thumb.jpg";
|
||||
|
||||
var originalPath = Path.Combine(AttachedRoot, originalFile);
|
||||
await File.WriteAllBytesAsync(originalPath, bytes, ct);
|
||||
|
||||
var thumbPath = Path.Combine(AttachedRoot, thumbFile);
|
||||
await ImageThumb.CreateThumbnailAsync(originalPath, thumbPath, maxSide: 320, quality: 70, ct);
|
||||
|
||||
return ($"https://localfiles/attached/{originalFile}",
|
||||
$"https://localfiles/attached/{thumbFile}");
|
||||
}
|
||||
|
||||
private static string SanitizeFileName(string fileName)
|
||||
{
|
||||
var name = Path.GetFileName(fileName);
|
||||
return Path.GetInvalidFileNameChars().Aggregate(name, (current, c) => current.Replace(c, '_'));
|
||||
}
|
||||
}
|
||||
52
SteUp.Maui/Core/Services/ImageThumb.cs
Normal file
52
SteUp.Maui/Core/Services/ImageThumb.cs
Normal file
@@ -0,0 +1,52 @@
|
||||
using SkiaSharp;
|
||||
|
||||
namespace SteUp.Maui.Core.Services;
|
||||
|
||||
public static class ImageThumb
|
||||
{
|
||||
public static async Task CreateThumbnailAsync(
|
||||
string inputPath,
|
||||
string outputPath,
|
||||
int maxSide = 320,
|
||||
int quality = 70,
|
||||
CancellationToken ct = default)
|
||||
{
|
||||
// Leggi bytes (meglio in async)
|
||||
var data = await File.ReadAllBytesAsync(inputPath, ct);
|
||||
|
||||
using var codec = SKCodec.Create(new SKMemoryStream(data));
|
||||
if (codec is null)
|
||||
throw new InvalidOperationException("Formato immagine non supportato o file corrotto.");
|
||||
|
||||
// Decodifica
|
||||
var info = codec.Info;
|
||||
using var bitmap = SKBitmap.Decode(codec);
|
||||
if (bitmap is null)
|
||||
throw new InvalidOperationException("Impossibile decodificare l'immagine.");
|
||||
|
||||
// Calcola resize mantenendo aspect ratio
|
||||
var w = bitmap.Width;
|
||||
var h = bitmap.Height;
|
||||
|
||||
if (w <= 0 || h <= 0) throw new InvalidOperationException("Dimensioni immagine non valide.");
|
||||
|
||||
var scale = (float)maxSide / Math.Max(w, h);
|
||||
if (scale > 1f) scale = 1f; // non ingrandire
|
||||
|
||||
var newW = Math.Max(1, (int)Math.Round(w * scale));
|
||||
var newH = Math.Max(1, (int)Math.Round(h * scale));
|
||||
|
||||
using var resized = bitmap.Resize(new SKImageInfo(newW, newH), SKFilterQuality.Medium);
|
||||
if (resized is null)
|
||||
throw new InvalidOperationException("Resize fallito.");
|
||||
|
||||
using var image = SKImage.FromBitmap(resized);
|
||||
using var encoded = image.Encode(SKEncodedImageFormat.Jpeg, quality);
|
||||
|
||||
Directory.CreateDirectory(Path.GetDirectoryName(outputPath)!);
|
||||
|
||||
await using var fs = File.Open(outputPath, FileMode.Create, FileAccess.Write, FileShare.None);
|
||||
encoded.SaveTo(fs);
|
||||
await fs.FlushAsync(ct);
|
||||
}
|
||||
}
|
||||
@@ -6,7 +6,7 @@
|
||||
SafeAreaEdges="All"
|
||||
BackgroundColor="{DynamicResource PageBackgroundColor}">
|
||||
|
||||
<BlazorWebView HostPage="wwwroot/index.html">
|
||||
<BlazorWebView x:Name="BlazorWebView" HostPage="wwwroot/index.html">
|
||||
<BlazorWebView.RootComponents>
|
||||
<RootComponent Selector="#app" ComponentType="{x:Type shared:Components.Routes}" />
|
||||
</BlazorWebView.RootComponents>
|
||||
|
||||
@@ -1,10 +1,59 @@
|
||||
namespace SteUp.Maui
|
||||
{
|
||||
namespace SteUp.Maui;
|
||||
|
||||
public partial class MainPage : ContentPage
|
||||
{
|
||||
private static readonly string AttachedDir =
|
||||
Path.Combine(FileSystem.CacheDirectory, "attached");
|
||||
|
||||
private const string Prefix = "https://localfiles/attached/";
|
||||
|
||||
public MainPage()
|
||||
{
|
||||
InitializeComponent();
|
||||
}
|
||||
|
||||
Directory.CreateDirectory(AttachedDir);
|
||||
|
||||
BlazorWebView.WebResourceRequested += BlazorWebView_WebResourceRequested;
|
||||
}
|
||||
|
||||
private static void BlazorWebView_WebResourceRequested(object? sender, WebViewWebResourceRequestedEventArgs e)
|
||||
{
|
||||
var uri = e.Uri.ToString();
|
||||
if (string.IsNullOrWhiteSpace(uri) ||
|
||||
!uri.StartsWith(Prefix, StringComparison.OrdinalIgnoreCase))
|
||||
return;
|
||||
|
||||
var fileName = uri[Prefix.Length..];
|
||||
|
||||
fileName = fileName.Replace("\\", "/");
|
||||
if (fileName.Contains("..") || fileName.Contains('/'))
|
||||
{
|
||||
e.Handled = true;
|
||||
e.SetResponse(400, "Bad Request");
|
||||
return;
|
||||
}
|
||||
|
||||
var fullPath = Path.Combine(AttachedDir, fileName);
|
||||
|
||||
if (!File.Exists(fullPath))
|
||||
{
|
||||
e.Handled = true;
|
||||
e.SetResponse(404, "Not Found");
|
||||
return;
|
||||
}
|
||||
|
||||
e.Handled = true;
|
||||
e.SetResponse(200, "OK", GetContentType(fullPath), File.OpenRead(fullPath));
|
||||
}
|
||||
|
||||
private static string GetContentType(string path)
|
||||
{
|
||||
return Path.GetExtension(path).ToLowerInvariant() switch
|
||||
{
|
||||
".png" => "image/png",
|
||||
".jpg" or ".jpeg" => "image/jpeg",
|
||||
".webp" => "image/webp",
|
||||
_ => "application/octet-stream"
|
||||
};
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="it.integry.SteUp">
|
||||
<application
|
||||
android:allowBackup="true"
|
||||
android:icon="@mipmap/appicon"
|
||||
@@ -10,4 +10,19 @@
|
||||
<!-- Rete -->
|
||||
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
|
||||
<uses-permission android:name="android.permission.INTERNET" />
|
||||
|
||||
<!-- Fotocamera -->
|
||||
<uses-permission android:name="android.permission.CAMERA" />
|
||||
|
||||
<!-- Storage / Media -->
|
||||
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
|
||||
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
|
||||
|
||||
<!-- Android 10+ -->
|
||||
<uses-permission android:name="android.permission.ACCESS_MEDIA_LOCATION" />
|
||||
|
||||
<!-- Android 13+ -->
|
||||
<uses-permission android:name="android.permission.READ_MEDIA_IMAGES" />
|
||||
<uses-permission android:name="android.permission.READ_MEDIA_VIDEO" />
|
||||
<uses-permission android:name="android.permission.READ_MEDIA_AUDIO" />
|
||||
</manifest>
|
||||
|
||||
@@ -121,6 +121,7 @@
|
||||
<PackageReference Include="Microsoft.AspNetCore.Components.WebView.Maui" Version="10.0.40" />
|
||||
<PackageReference Include="Microsoft.Extensions.Logging.Debug" Version="10.0.3" />
|
||||
<PackageReference Include="Sentry.Maui" Version="6.1.0" />
|
||||
<PackageReference Include="SkiaSharp" Version="3.119.2" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
|
||||
6
SteUp.Shared/Components/Pages/NotFound.razor
Normal file
6
SteUp.Shared/Components/Pages/NotFound.razor
Normal file
@@ -0,0 +1,6 @@
|
||||
@page "/not-found"
|
||||
@using SteUp.Shared.Components.Layout
|
||||
@layout MainLayout
|
||||
|
||||
<h3>Not Found</h3>
|
||||
<p>Sorry, the content you are looking for does not exist.</p>
|
||||
@@ -4,7 +4,7 @@
|
||||
<ErrorBoundary @ref="ErrorBoundary">
|
||||
<ChildContent>
|
||||
<CascadingAuthenticationState>
|
||||
<Router AppAssembly="@typeof(Routes).Assembly">
|
||||
<Router AppAssembly="@typeof(Routes).Assembly" NotFoundPage="typeof(Pages.NotFound)">
|
||||
<Found Context="routeData">
|
||||
<AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(Layout.MainLayout)">
|
||||
<Authorizing>
|
||||
@@ -23,12 +23,6 @@
|
||||
</AuthorizeRouteView>
|
||||
<FocusOnNavigate RouteData="@routeData" Selector="h1"/>
|
||||
</Found>
|
||||
<NotFound>
|
||||
<PageTitle>Not found</PageTitle>
|
||||
<LayoutView>
|
||||
<p role="alert">Sorry, there's nothing at this address.</p>
|
||||
</LayoutView>
|
||||
</NotFound>
|
||||
</Router>
|
||||
</CascadingAuthenticationState>
|
||||
</ChildContent>
|
||||
@@ -44,7 +38,7 @@
|
||||
@code {
|
||||
|
||||
private ErrorBoundary? ErrorBoundary { get; set; }
|
||||
private ExceptionModal ExceptionModal { get; set; }
|
||||
private ExceptionModal ExceptionModal { get; set; } = null!;
|
||||
|
||||
protected override async Task OnInitializedAsync()
|
||||
{
|
||||
|
||||
@@ -92,7 +92,7 @@
|
||||
var puntoVendita = SteupDataService.PuntiVenditaList.Find(x =>
|
||||
x.CodMdep != null && x.CodMdep.EqualsIgnoreCase(Ispezione.CodMdep)
|
||||
);
|
||||
await Task.Delay(500);
|
||||
await Task.Delay(250);
|
||||
|
||||
PuntoVendita = puntoVendita ?? throw new Exception("Punto vendita non trovato");
|
||||
OnLoading = false;
|
||||
|
||||
@@ -0,0 +1,21 @@
|
||||
<MudMessageBox @ref="_confirmSave" Title="Attenzione!" CancelText="Non salvare">
|
||||
<MessageContent>
|
||||
Sono state apportate delle modifiche. Vuoi salvarle prima di continuare?
|
||||
</MessageContent>
|
||||
<YesButton>
|
||||
<MudButton Size="Size.Small" Variant="Variant.Filled" Color="Color.Primary">
|
||||
Salva
|
||||
</MudButton>
|
||||
</YesButton>
|
||||
</MudMessageBox>
|
||||
|
||||
@code
|
||||
{
|
||||
private MudMessageBox? _confirmSave;
|
||||
|
||||
public async Task<bool?> ShowAsync()
|
||||
{
|
||||
if (_confirmSave == null) return null;
|
||||
return await _confirmSave.ShowAsync();
|
||||
}
|
||||
}
|
||||
@@ -5,14 +5,8 @@
|
||||
|
||||
<MudDialog Class="disable-safe-area">
|
||||
<DialogContent>
|
||||
<HeaderLayout SmallHeader="true" Cancel="true" OnCancel="@(() => MudDialog.Cancel())" Title="@TitleModal"/>
|
||||
<HeaderLayout SmallHeader="true" Cancel="true" OnCancel="@(() => MudDialog.Cancel())" Title="Aggiungi allegati"/>
|
||||
|
||||
@if (RequireNewName)
|
||||
{
|
||||
<MudTextField @bind-Value="NewName" Class="px-3" Variant="Variant.Outlined"/>
|
||||
}
|
||||
else
|
||||
{
|
||||
<div style="margin-bottom: 1rem;" class="content attached">
|
||||
<MudFab Size="Size.Small" Color="Color.Primary"
|
||||
StartIcon="@Icons.Material.Rounded.CameraAlt"
|
||||
@@ -22,67 +16,29 @@
|
||||
StartIcon="@Icons.Material.Rounded.Image"
|
||||
Label="Galleria" OnClick="@OnGallery"/>
|
||||
</div>
|
||||
}
|
||||
</DialogContent>
|
||||
<DialogActions>
|
||||
@if (RequireNewName)
|
||||
{
|
||||
<MudButton Disabled="NewName.IsNullOrEmpty()" Class="my-3" Size="Size.Small" Variant="Variant.Filled"
|
||||
Color="Color.Primary"
|
||||
StartIcon="@Icons.Material.Rounded.Check" OnClick="@OnNewName">
|
||||
Salva
|
||||
</MudButton>
|
||||
}
|
||||
</DialogActions>
|
||||
</MudDialog>
|
||||
|
||||
@code {
|
||||
[CascadingParameter] private IMudDialogInstance MudDialog { get; set; } = null!;
|
||||
[Parameter] public bool CanAddPosition { get; set; }
|
||||
|
||||
private AttachedDto? Attached { get; set; }
|
||||
|
||||
private bool _requireNewName;
|
||||
|
||||
private bool RequireNewName
|
||||
{
|
||||
get => _requireNewName;
|
||||
set
|
||||
{
|
||||
_requireNewName = value;
|
||||
TitleModal = _requireNewName ? "Nome allegato" : "Aggiungi allegati";
|
||||
StateHasChanged();
|
||||
}
|
||||
}
|
||||
|
||||
private string TitleModal { get; set; } = "Aggiungi allegati";
|
||||
|
||||
private string? _newName;
|
||||
|
||||
private string? NewName
|
||||
{
|
||||
get => _newName;
|
||||
set
|
||||
{
|
||||
_newName = value;
|
||||
StateHasChanged();
|
||||
}
|
||||
}
|
||||
private List<AttachedDto>? Attached { get; set; }
|
||||
|
||||
protected override void OnInitialized()
|
||||
{
|
||||
RequireNewName = false;
|
||||
Snackbar.Configuration.PositionClass = Defaults.Classes.Position.TopCenter;
|
||||
}
|
||||
|
||||
private async Task OnCamera()
|
||||
{
|
||||
Attached = await AttachedService.SelectImageFromCamera();
|
||||
var selectImageFromCamera = await AttachedService.SelectImageFromCamera();
|
||||
|
||||
if (Attached != null)
|
||||
if (selectImageFromCamera != null)
|
||||
{
|
||||
RequireNewName = true;
|
||||
StateHasChanged();
|
||||
Attached ??= [];
|
||||
Attached.Add(selectImageFromCamera);
|
||||
MudDialog.Close(Attached);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -90,32 +46,6 @@
|
||||
{
|
||||
Attached = await AttachedService.SelectImageFromGallery();
|
||||
|
||||
if (Attached != null)
|
||||
{
|
||||
RequireNewName = true;
|
||||
StateHasChanged();
|
||||
if (Attached != null) MudDialog.Close(Attached);
|
||||
}
|
||||
}
|
||||
|
||||
private void OnNewName()
|
||||
{
|
||||
if (Attached != null)
|
||||
{
|
||||
switch (Attached.Type)
|
||||
{
|
||||
case AttachedDto.TypeAttached.Image:
|
||||
{
|
||||
var extension = Path.GetExtension(Attached.Name);
|
||||
Attached.Name = NewName! + extension;
|
||||
|
||||
break;
|
||||
}
|
||||
default:
|
||||
throw new ArgumentOutOfRangeException();
|
||||
}
|
||||
}
|
||||
|
||||
MudDialog.Close(Attached);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -1,18 +1,22 @@
|
||||
@using SteUp.Shared.Components.Layout
|
||||
@using SteUp.Shared.Components.Layout.Overlay
|
||||
@using SteUp.Shared.Components.Layout.Spinner
|
||||
@using SteUp.Shared.Components.SingleElements.Card.ModalForm
|
||||
@using SteUp.Shared.Components.SingleElements.MessageBox
|
||||
@using SteUp.Shared.Core.Dto
|
||||
@using SteUp.Shared.Core.Entities
|
||||
@using SteUp.Shared.Core.Interface.IntegryApi
|
||||
@using SteUp.Shared.Core.Interface.System
|
||||
@using SteUp.Shared.Core.Interface.System.Network
|
||||
@inject INetworkService NetworkService
|
||||
@inject IDialogService Dialog
|
||||
@inject IIntegryApiService IntegryApiService
|
||||
@inject IAttachedService AttachedService
|
||||
|
||||
<MudDialog Class="customDialog-form">
|
||||
<DialogContent>
|
||||
|
||||
<HeaderLayout Cancel="true" OnCancel="@(() => MudDialog.Cancel())" LabelSave="@LabelSave"
|
||||
<MudForm @ref="_form">
|
||||
<HeaderLayout Cancel="true" OnCancel="@Cancel" LabelSave="@LabelSave"
|
||||
OnSave="Save" Title="Scheda"/>
|
||||
|
||||
<div class="content">
|
||||
@@ -20,7 +24,8 @@
|
||||
<CardFormModal Title="Reparto" Loading="SteupDataService.Reparti.IsNullOrEmpty()">
|
||||
<MudSelectExtended ReadOnly="IsView" T="JtbFasiDto?" Variant="Variant.Text"
|
||||
@bind-Value="Scheda.Reparto" ToStringFunc="@(x => x?.Descrizione)"
|
||||
@bind-Value:after="OnAfterChangeValue">
|
||||
@bind-Value:after="OnAfterChangeValue" Required="true"
|
||||
RequiredError="Reparto obbligatorio">
|
||||
@foreach (var fasi in SteupDataService.Reparti)
|
||||
{
|
||||
<MudSelectItemExtended Class="custom-item-select" Value="@fasi">
|
||||
@@ -31,9 +36,11 @@
|
||||
</CardFormModal>
|
||||
|
||||
<CardFormModal Title="Motivo" Loading="SteupDataService.TipiAttività.IsNullOrEmpty()">
|
||||
<MudSelectExtended ReadOnly="IsView" T="string?" Variant="Variant.Text"
|
||||
@bind-Value="Scheda.ActivityTypeId" @bind-Value:after="OnAfterChangeValue">
|
||||
@foreach (var type in SteupDataService.TipiAttività)
|
||||
<MudSelectExtended ReadOnly="@(IsView || Scheda.CodJfas.IsNullOrEmpty())" T="string?"
|
||||
Variant="Variant.Text"
|
||||
@bind-Value="Scheda.ActivityTypeId" @bind-Value:after="OnAfterChangeValue"
|
||||
Required="true" RequiredError="Motivo obbligatorio">
|
||||
@foreach (var type in SteupDataService.TipiAttività.Where(x => x.CodJfas.EqualsIgnoreCase(Scheda.CodJfas!)))
|
||||
{
|
||||
<MudSelectItemExtended Class="custom-item-select"
|
||||
Value="@type.ActivityTypeId">@type.ActivityTypeId</MudSelectItemExtended>
|
||||
@@ -41,47 +48,28 @@
|
||||
</MudSelectExtended>
|
||||
</CardFormModal>
|
||||
|
||||
@* <div class="container-chip-attached"> *@
|
||||
@* @if (!AttachedList.IsNullOrEmpty()) *@
|
||||
@* { *@
|
||||
@* foreach (var item in AttachedList!.Select((p, index) => new { p, index })) *@
|
||||
@* { *@
|
||||
@* if (item.p.Type == AttachedDTO.TypeAttached.Position) *@
|
||||
@* { *@
|
||||
@* <MudChip T="string" Icon="@Icons.Material.Rounded.LocationOn" Color="Color.Success" *@
|
||||
@* OnClick="@(() => OpenPosition(item.p))" *@
|
||||
@* OnClose="@(() => OnRemoveAttached(item.index))"> *@
|
||||
@* @item.p.Description *@
|
||||
@* </MudChip> *@
|
||||
@* } *@
|
||||
@* else *@
|
||||
@* { *@
|
||||
@* <MudChip T="string" Color="Color.Default" OnClick="@(() => OpenAttached(item.p))" *@
|
||||
@* OnClose="@(() => OnRemoveAttached(item.index))"> *@
|
||||
@* @item.p.Name *@
|
||||
@* </MudChip> *@
|
||||
@* } *@
|
||||
@* } *@
|
||||
@* } *@
|
||||
@* *@
|
||||
@* @if (!IsLoading) *@
|
||||
@* { *@
|
||||
@* if (ActivityFileList != null) *@
|
||||
@* { *@
|
||||
@* foreach (var file in ActivityFileList) *@
|
||||
@* { *@
|
||||
@* <MudChip T="string" OnClick="@(() => OpenAttached(file.FileName))" *@
|
||||
@* OnClose="@(() => DeleteAttach(file))" Color="Color.Default"> *@
|
||||
@* @file.FileName *@
|
||||
@* </MudChip> *@
|
||||
@* } *@
|
||||
@* } *@
|
||||
@* } *@
|
||||
@* else *@
|
||||
@* { *@
|
||||
@* <MudProgressLinear Color="Color.Primary" Indeterminate="true" Class="my-7"/> *@
|
||||
@* } *@
|
||||
@* </div> *@
|
||||
@if (!AttachedList.IsNullOrEmpty())
|
||||
{
|
||||
<div class="container-attached">
|
||||
<div class="scroll-attached">
|
||||
@foreach (var item in AttachedList!.Select((p, index) => new { p, index }))
|
||||
{
|
||||
<MudCard>
|
||||
@if (!item.p.ThumbPath.IsNullOrEmpty())
|
||||
{
|
||||
<MudCardMedia Image="@item.p.ThumbPath" Height="100"/>
|
||||
}
|
||||
<MudCardContent Class="image_card">
|
||||
<MudText Typo="Typo.subtitle1"><b>@item.p.Name</b></MudText>
|
||||
<MudIconButton Variant="Variant.Outlined" Icon="@Icons.Material.Rounded.Close"
|
||||
Size="Size.Small" Color="Color.Error"
|
||||
OnClick="@(() => OnRemoveAttached(item.index))"/>
|
||||
</MudCardContent>
|
||||
</MudCard>
|
||||
}
|
||||
</div>
|
||||
</div>
|
||||
}
|
||||
|
||||
@if (!IsView)
|
||||
{
|
||||
@@ -130,6 +118,9 @@
|
||||
</MudButton>
|
||||
</div>
|
||||
</div>
|
||||
</MudForm>
|
||||
|
||||
<ConfirmUpdateActivity @ref="_confirmUpdateMessage"/>
|
||||
</DialogContent>
|
||||
</MudDialog>
|
||||
|
||||
@@ -148,10 +139,17 @@
|
||||
private bool VisibleOverlay { get; set; }
|
||||
private bool SuccessAnimation { get; set; }
|
||||
|
||||
private string? LabelSave { get; set; }
|
||||
private ConfirmUpdateActivity _confirmUpdateMessage = null!;
|
||||
private MudForm _form = null!;
|
||||
|
||||
protected override async Task OnInitializedAsync()
|
||||
private string? LabelSave { get; set; }
|
||||
private bool IsDirty { get; set; }
|
||||
private Scheda _originalScheda = null!;
|
||||
private List<AttachedDto>? AttachedList { get; set; }
|
||||
|
||||
protected override void OnInitialized()
|
||||
{
|
||||
_originalScheda = Scheda.Clone();
|
||||
Snackbar.Configuration.PositionClass = Defaults.Classes.Position.TopCenter;
|
||||
}
|
||||
|
||||
@@ -159,30 +157,129 @@
|
||||
{
|
||||
}
|
||||
|
||||
private async Task OpenAddAttached()
|
||||
private async Task Cancel()
|
||||
{
|
||||
var result = await ModalHelper.OpenAddAttached(Dialog);
|
||||
|
||||
// if (result is { Canceled: false, Data: not null } && result.Data.GetType() == typeof(AttachedDTO))
|
||||
// {
|
||||
// var attached = (AttachedDTO)result.Data;
|
||||
//
|
||||
// if (attached.Type == AttachedDTO.TypeAttached.Position)
|
||||
// CanAddPosition = false;
|
||||
//
|
||||
// AttachedList ??= [];
|
||||
// AttachedList.Add(attached);
|
||||
// }
|
||||
if (await CheckSavePreAction())
|
||||
{
|
||||
await AttachedService.CleanTempStorageAsync();
|
||||
MudDialog.Cancel();
|
||||
}
|
||||
}
|
||||
|
||||
#region Form
|
||||
|
||||
private void OnAfterChangeValue()
|
||||
{
|
||||
if (!IsNew)
|
||||
LabelSave = "Aggiorna";
|
||||
|
||||
RecalcDirty();
|
||||
StateHasChanged();
|
||||
}
|
||||
|
||||
private void RecalcDirty()
|
||||
{
|
||||
IsDirty = !ValueComparer.AreEqual(Scheda, _originalScheda);
|
||||
|
||||
if (IsDirty) LabelSave = !IsNew ? "Aggiorna" : "Salva";
|
||||
else LabelSave = null;
|
||||
}
|
||||
|
||||
private static class ValueComparer
|
||||
{
|
||||
public static bool AreEqual(Scheda? a, Scheda? b)
|
||||
{
|
||||
if (a is null || b is null) return a == b;
|
||||
|
||||
return
|
||||
a.CodJfas == b.CodJfas &&
|
||||
a.DescrizioneReparto == b.DescrizioneReparto &&
|
||||
a.ActivityTypeId == b.ActivityTypeId &&
|
||||
a.Note == b.Note &&
|
||||
a.Responsabile == b.Responsabile &&
|
||||
a.Scadenza == b.Scadenza;
|
||||
}
|
||||
}
|
||||
|
||||
private async Task<bool> CheckSavePreAction()
|
||||
{
|
||||
if (!IsDirty) return true;
|
||||
|
||||
var resul = await _confirmUpdateMessage.ShowAsync();
|
||||
if (resul is not true) return true;
|
||||
|
||||
VisibleOverlay = true;
|
||||
StateHasChanged();
|
||||
|
||||
await Submit();
|
||||
|
||||
VisibleOverlay = false;
|
||||
StateHasChanged();
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
private async Task Submit()
|
||||
{
|
||||
await _form.Validate();
|
||||
if (_form.IsValid) await Save();
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region File
|
||||
|
||||
private async Task OpenAddAttached()
|
||||
{
|
||||
var result = await ModalHelper.OpenAddAttached(Dialog);
|
||||
if (result is not { Canceled: false, Data: List<AttachedDto> attachedList }) return;
|
||||
|
||||
VisibleOverlay = true;
|
||||
await InvokeAsync(StateHasChanged);
|
||||
await Task.Yield();
|
||||
|
||||
// prepara placeholder in UI subito (così vedi le card con spinner)
|
||||
AttachedList ??= [];
|
||||
foreach (var a in attachedList)
|
||||
AttachedList.Add(new AttachedDto { Name = a.Name, MimeType = a.MimeType, FileBytes = a.FileBytes });
|
||||
|
||||
await InvokeAsync(StateHasChanged);
|
||||
|
||||
// Processa in background e aggiorna UI man mano (o a blocchi)
|
||||
_ = Task.Run(async () =>
|
||||
{
|
||||
for (var i = 0; i < attachedList.Count; i++)
|
||||
{
|
||||
var a = attachedList[i];
|
||||
if (a.FileBytes is null || a.Name is null) continue;
|
||||
|
||||
var (origUrl, thumbUrl) = await AttachedService.SaveAndCreateThumbAsync(a.FileBytes, a.Name);
|
||||
|
||||
await InvokeAsync(() =>
|
||||
{
|
||||
var target = AttachedList![AttachedList.Count - attachedList.Count + i];
|
||||
target.TempPath = origUrl;
|
||||
target.ThumbPath = thumbUrl;
|
||||
StateHasChanged();
|
||||
});
|
||||
}
|
||||
|
||||
await InvokeAsync(() =>
|
||||
{
|
||||
VisibleOverlay = false;
|
||||
StateHasChanged();
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
private void OnRemoveAttached(int index)
|
||||
{
|
||||
if (AttachedList is null || index < 0 || index >= AttachedList.Count)
|
||||
return;
|
||||
|
||||
AttachedList.RemoveAt(index);
|
||||
StateHasChanged();
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
private void SuggestActivityDescription()
|
||||
{
|
||||
if (Scheda.ActivityTypeId == null)
|
||||
|
||||
@@ -1,6 +1,21 @@
|
||||
.container-chip-attached {
|
||||
.container-attached {
|
||||
width: 100%;
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
.scroll-attached {
|
||||
max-height: 185px;
|
||||
overflow: auto;
|
||||
display: flex;
|
||||
gap: 1rem;
|
||||
flex-direction: column;
|
||||
padding: .5rem;
|
||||
}
|
||||
|
||||
.container-attached ::deep .image_card {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
padding: 14px;
|
||||
}
|
||||
|
||||
.container-button {
|
||||
|
||||
@@ -11,5 +11,5 @@ public interface ISteupDataService
|
||||
List<PuntoVenditaDto> PuntiVenditaList { get; }
|
||||
InspectionPageState InspectionPageState { get; set; }
|
||||
List<JtbFasiDto> Reparti { get; }
|
||||
List<StbActivityTypeDto> TipiAttività { get; }
|
||||
List<ActivityTypeDto> TipiAttività { get; }
|
||||
}
|
||||
@@ -42,5 +42,5 @@ public class SteupDataService(
|
||||
public InspectionPageState InspectionPageState { get; set; } = new();
|
||||
public List<PuntoVenditaDto> PuntiVenditaList { get; private set; } = [];
|
||||
public List<JtbFasiDto> Reparti { get; private set; } = [];
|
||||
public List<StbActivityTypeDto> TipiAttività { get; private set; } = [];
|
||||
public List<ActivityTypeDto> TipiAttività { get; private set; } = [];
|
||||
}
|
||||
12
SteUp.Shared/Core/Dto/ActivityTypeDto.cs
Normal file
12
SteUp.Shared/Core/Dto/ActivityTypeDto.cs
Normal file
@@ -0,0 +1,12 @@
|
||||
using System.Text.Json.Serialization;
|
||||
|
||||
namespace SteUp.Shared.Core.Dto;
|
||||
|
||||
public class ActivityTypeDto
|
||||
{
|
||||
[JsonPropertyName("activityTypeId")]
|
||||
public string ActivityTypeId { get; set; } = string.Empty;
|
||||
|
||||
[JsonPropertyName("codJfas")]
|
||||
public string CodJfas { get; set; } = string.Empty;
|
||||
}
|
||||
@@ -11,6 +11,9 @@ public class AttachedDto
|
||||
|
||||
public byte[]? FileBytes { get; set; }
|
||||
|
||||
public string? TempPath { get; set; }
|
||||
public string? ThumbPath { get; set; }
|
||||
|
||||
public Stream? FileContent =>
|
||||
FileBytes is null ? null : new MemoryStream(FileBytes);
|
||||
|
||||
|
||||
@@ -1,9 +0,0 @@
|
||||
using System.Text.Json.Serialization;
|
||||
|
||||
namespace SteUp.Shared.Core.Dto;
|
||||
|
||||
public class StbActivityTypeDto
|
||||
{
|
||||
[JsonPropertyName("activityTypeId")]
|
||||
public string? ActivityTypeId { get; set; }
|
||||
}
|
||||
9
SteUp.Shared/Core/Entities/EntityBase.cs
Normal file
9
SteUp.Shared/Core/Entities/EntityBase.cs
Normal file
@@ -0,0 +1,9 @@
|
||||
namespace SteUp.Shared.Core.Entities;
|
||||
|
||||
public class EntityBase<T>
|
||||
{
|
||||
public T Clone()
|
||||
{
|
||||
return (T)MemberwiseClone();
|
||||
}
|
||||
}
|
||||
@@ -3,7 +3,7 @@ using SteUp.Shared.Core.Enum;
|
||||
|
||||
namespace SteUp.Shared.Core.Entities;
|
||||
|
||||
public class Ispezione
|
||||
public class Ispezione : EntityBase<Ispezione>
|
||||
{
|
||||
[Required]
|
||||
public string CodMdep { get; set; } = string.Empty;
|
||||
|
||||
@@ -4,7 +4,7 @@ using SteUp.Shared.Core.Dto;
|
||||
|
||||
namespace SteUp.Shared.Core.Entities;
|
||||
|
||||
public class Scheda
|
||||
public class Scheda : EntityBase<Scheda>
|
||||
{
|
||||
[Key]
|
||||
public int Id { get; set; }
|
||||
@@ -28,24 +28,24 @@ public class Scheda
|
||||
{
|
||||
get
|
||||
{
|
||||
if (_reparto == null && CodJfas != null)
|
||||
if (field == null && CodJfas != null)
|
||||
{
|
||||
_reparto = new JtbFasiDto
|
||||
field = new JtbFasiDto
|
||||
{
|
||||
CodJfas = CodJfas,
|
||||
Descrizione = DescrizioneReparto
|
||||
};
|
||||
}
|
||||
return _reparto;
|
||||
|
||||
return field;
|
||||
}
|
||||
set
|
||||
{
|
||||
_reparto = value;
|
||||
field = value;
|
||||
if (value == null) return;
|
||||
|
||||
CodJfas = value.CodJfas;
|
||||
DescrizioneReparto = value.Descrizione;
|
||||
}
|
||||
}
|
||||
private JtbFasiDto? _reparto;
|
||||
}
|
||||
@@ -4,7 +4,7 @@ using SteUp.Shared.Core.Dto;
|
||||
|
||||
namespace SteUp.Shared.Core.Helpers;
|
||||
|
||||
public class ModalHelper
|
||||
public abstract class ModalHelper
|
||||
{
|
||||
public static async Task<DialogResult?> OpenSelectShop(IDialogService dialog)
|
||||
{
|
||||
|
||||
@@ -7,5 +7,5 @@ public interface IIntegrySteupService
|
||||
//Retrieve
|
||||
Task<List<PuntoVenditaDto>> RetrievePuntiVendita();
|
||||
Task<List<JtbFasiDto>> RetrieveReparti();
|
||||
Task<List<StbActivityTypeDto>> RetrieveActivityType();
|
||||
Task<List<ActivityTypeDto>> RetrieveActivityType();
|
||||
}
|
||||
@@ -5,8 +5,11 @@ namespace SteUp.Shared.Core.Interface.System;
|
||||
public interface IAttachedService
|
||||
{
|
||||
Task<AttachedDto?> SelectImageFromCamera();
|
||||
Task<AttachedDto?> SelectImageFromGallery();
|
||||
Task<List<AttachedDto>?> SelectImageFromGallery();
|
||||
|
||||
Task<string> SaveToTempStorage(Stream file, string fileName, CancellationToken ct = default);
|
||||
Task CleanTempStorageAsync(CancellationToken ct = default);
|
||||
Task OpenFile(string fileName, string filePath);
|
||||
|
||||
Task<(string originalUrl, string thumbUrl)> SaveAndCreateThumbAsync(byte[] bytes, string fileName, CancellationToken ct = default);
|
||||
}
|
||||
@@ -16,8 +16,8 @@ public class IntegrySteupService(IIntegryApiRestClient integryApiRestClient) : I
|
||||
public Task<List<JtbFasiDto>> RetrieveReparti() =>
|
||||
integryApiRestClient.AuthorizedGet<List<JtbFasiDto>>($"{BaseRequest}/retrieveReparti")!;
|
||||
|
||||
public Task<List<StbActivityTypeDto>> RetrieveActivityType() =>
|
||||
integryApiRestClient.AuthorizedGet<List<StbActivityTypeDto>>($"{BaseRequest}/retrieveActivityType")!;
|
||||
public Task<List<ActivityTypeDto>> RetrieveActivityType() =>
|
||||
integryApiRestClient.AuthorizedGet<List<ActivityTypeDto>>($"{BaseRequest}/retrieveActivityType")!;
|
||||
|
||||
#endregion
|
||||
}
|
||||
@@ -21,7 +21,10 @@
|
||||
height: calc(100vh - (.6rem + 40px));
|
||||
overflow: auto;
|
||||
gap: 1.5rem;
|
||||
padding: 0 .75rem 2rem 75rem !important;
|
||||
padding-top: unset !important;
|
||||
padding-bottom: 2rem !important;
|
||||
padding-left: .75rem !important;
|
||||
padding-right: .75rem !important;
|
||||
}
|
||||
|
||||
@supports (-webkit-touch-callout: none) {
|
||||
|
||||
Reference in New Issue
Block a user