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);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user