Compare commits

...

7 commits

Author SHA1 Message Date
398cfa310e Refactor path handling in UI components
Some checks failed
Build Windows Avalonia / build (push) Failing after 1m46s
Build Windows Avalonia / release (push) Has been skipped
- Removed redundant folder and file opening methods from AiTabView and FaceAiTabView.
- Introduced PathPickerField control to streamline path selection and opening functionality across multiple views.
- Updated FaceAiTabView and GeneralTabView to utilize PathPickerField for source and destination path selection.
- Created PathShellService to encapsulate logic for opening paths in the file explorer.
- Simplified XAML structure by replacing manual grid definitions with PathPickerField components.
- Removed unused namespaces and cleaned up code for better readability and maintainability.
2026-05-24 19:07:17 +02:00
f3ac1ea920 feat: Implement AI CSV overwrite confirmation and update CSV output path based on destination 2026-05-24 18:45:51 +02:00
6e05869b04 feat: Add GPU support options and enhance UI for Face Encoder and Matcher functionalities 2026-05-24 18:33:54 +02:00
af74c90ce7 feat: Update package references and enhance AI extraction service with CSV output functionality 2026-05-24 17:29:05 +02:00
55e8f0face Local build fix 2026-05-24 11:21:30 +02:00
5511817896 feat: Update .gitignore and project file to include TestArtifacts and local CUDA libraries for publish 2026-05-24 10:49:41 +02:00
c261557a29 feat: Add Face Matcher functionality and related settings
- Implemented FilePathToBitmapConverter for image loading.
- Enhanced DataModel with commands and properties for Face Matcher.
- Created FaceMatcherResultItem model to store results.
- Updated SettingsDto to include Face Matcher paths and tolerance.
- Introduced PickerPreferenceService for managing folder paths.
- Expanded AiSettingsViewModel to manage Face Matcher settings and results.
2026-05-09 20:27:44 +02:00
33 changed files with 2843 additions and 553 deletions

1
.gitignore vendored
View file

@ -257,3 +257,4 @@ paket-files/
*.sln.iml
.vscode/settings.json
tmp/**
TestArtifacts/**

View file

@ -8,10 +8,10 @@
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="10.0.3" />
<PackageReference Include="Microsoft.Extensions.Http" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.3" />
<PackageReference Include="Microsoft.Extensions.Options" Version="10.0.3" />
<PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="10.0.8" />
<PackageReference Include="Microsoft.Extensions.Http" Version="10.0.8" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.8" />
<PackageReference Include="Microsoft.Extensions.Options" Version="10.0.8" />
</ItemGroup>
</Project>

View file

@ -11,6 +11,7 @@
}
],
"settings": {
"commentTranslate.hover.enabled": false
"commentTranslate.hover.enabled": false,
"github.copilot.chat.otel.dbSpanExporter.enabled": true
}
}

View file

@ -10,8 +10,8 @@
<ItemGroup>
<PackageReference Include="BenchmarkDotNet" Version="0.15.8" />
<PackageReference Include="Microsoft.Extensions.Logging" Version="10.0.3" />
<PackageReference Include="Microsoft.Extensions.Logging.Console" Version="10.0.3" />
<PackageReference Include="Microsoft.Extensions.Logging" Version="10.0.8" />
<PackageReference Include="Microsoft.Extensions.Logging.Console" Version="10.0.8" />
</ItemGroup>
<ItemGroup>

View file

@ -9,15 +9,15 @@
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="18.0.1" />
<PackageReference Include="MSTest.TestAdapter" Version="4.1.0" />
<PackageReference Include="MSTest.TestFramework" Version="4.1.0" />
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="18.5.1" />
<PackageReference Include="MSTest.TestAdapter" Version="4.2.3" />
<PackageReference Include="MSTest.TestFramework" Version="4.2.3" />
<PackageReference Include="NSubstitute" Version="5.3.0" />
<PackageReference Include="Shouldly" Version="4.3.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.3" />
<PackageReference Include="SixLabors.ImageSharp" Version="3.1.12" />
<PackageReference Include="SixLabors.ImageSharp.Drawing" Version="2.1.7" />
<PackageReference Include="SixLabors.Fonts" Version="2.1.3" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.8" />
<PackageReference Include="SixLabors.ImageSharp" Version="4.0.0" />
<PackageReference Include="SixLabors.ImageSharp.Drawing" Version="3.0.0" />
<PackageReference Include="SixLabors.Fonts" Version="3.0.0" />
</ItemGroup>
<ItemGroup>

View file

@ -96,6 +96,45 @@ public class DataModelCharacterizationTests
model.DestinationPath.ShouldBe($"C:{System.IO.Path.DirectorySeparatorChar}output{System.IO.Path.DirectorySeparatorChar}");
}
[TestMethod]
public void DestinationPathChange_UpdatesAiCsvFileNameToMatchFinalFolderSegment()
{
var model = CreateModel();
model.CsvOutputPath = @"K:\various\catalogtest\aioutput\test2.csv";
model.DestinationPath = @"K:\various\catalogtest\Dest\03.KM_8_A\";
model.CsvOutputPath.ShouldBe(@"K:\various\catalogtest\aioutput\03.KM_8_A.csv");
}
[TestMethod]
public async Task ConfirmAiCsvOverwriteIfNeededAsync_CanCancelWhenCsvAlreadyExists()
{
using var tempDirectory = new TemporaryDirectory();
var csvPath = Path.Combine(tempDirectory.Path, "existing.csv");
File.WriteAllText(csvPath, "existing");
var model = CreateModel();
model.CsvOutputPath = csvPath;
string? requestedTitle = null;
string? requestedMessage = null;
model.ConfirmAiCsvOverwriteAsync = (title, message) =>
{
requestedTitle = title;
requestedMessage = message;
return Task.FromResult(false);
};
var shouldContinue = await model.ConfirmAiCsvOverwriteIfNeededAsync();
shouldContinue.ShouldBeFalse();
requestedTitle.ShouldBe("File CSV gia esistente");
requestedMessage.ShouldNotBeNull();
requestedMessage.ShouldContain("Vuoi sovrascriverlo?");
requestedMessage.ShouldContain(csvPath);
}
[TestMethod]
public void AiChildChange_RaisesDataModelPropertyChanged()
{

View file

@ -17,7 +17,7 @@
<ItemGroup>
<PackageReference Include="NSubstitute" Version="5.3.0" />
<PackageReference Include="Shouldly" Version="4.3.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.5" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.8" />
<PackageReference Include="System.Drawing.Common" Version="10.0.3" />
</ItemGroup>

View file

@ -0,0 +1,58 @@
using System;
using System.IO;
using ImageCatalog;
using ImageCatalog_2.Services;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using Shouldly;
namespace MaddoShared.Tests;
[TestClass]
public class PickerPreferenceServiceTests
{
[TestMethod]
public void RememberValue_PersistsExactFilePath()
{
using var tempDirectory = new TemporaryDirectory();
var preferencesFile = Path.Combine(tempDirectory.Path, "userprefs.xml");
var service = new PickerPreferenceService(new ParametriSetup(preferencesFile));
var settingsFile = Path.Combine(tempDirectory.Path, "nested", "settings.xml");
service.RememberValue(PickerPreferenceKeys.LastSettingsFile, settingsFile);
service.GetRememberedValue(PickerPreferenceKeys.LastSettingsFile).ShouldBe(settingsFile);
}
[TestMethod]
public void ForgetValue_RemovesStoredPreference()
{
using var tempDirectory = new TemporaryDirectory();
var preferencesFile = Path.Combine(tempDirectory.Path, "userprefs.xml");
var service = new PickerPreferenceService(new ParametriSetup(preferencesFile));
service.RememberValue(PickerPreferenceKeys.LastSettingsFile, Path.Combine(tempDirectory.Path, "settings.xml"));
service.ForgetValue(PickerPreferenceKeys.LastSettingsFile);
service.GetRememberedValue(PickerPreferenceKeys.LastSettingsFile).ShouldBeNull();
}
private sealed class TemporaryDirectory : IDisposable
{
public TemporaryDirectory()
{
Path = System.IO.Path.Combine(System.IO.Path.GetTempPath(), System.IO.Path.GetRandomFileName());
Directory.CreateDirectory(Path);
}
public string Path { get; }
public void Dispose()
{
if (Directory.Exists(Path))
{
Directory.Delete(Path, recursive: true);
}
}
}
}

View file

@ -1,11 +1,48 @@
namespace MaddoShared.Tests
{
using ImageCatalog_2.Services;
using ImageCatalog_2.Models;
using Shouldly;
namespace MaddoShared.Tests;
[TestClass]
public sealed class Test1
public sealed class AiExtractionServiceCsvTests
{
[TestMethod]
public void TestMethod1()
public void WriteCsvOutput_UsesLegacyCompatibleHeaderAndFilenameColumn()
{
using var tempDir = new TempDirectory();
var csvPath = Path.Combine(tempDir.Path, "ocr.csv");
AiExtractionService.WriteCsvOutput(
csvPath,
[
new AiResultItem { Path = @"C:\images\IMG_7146.JPG", Text = "43,84,61" },
new AiResultItem { Path = @"C:\images\IMG_7207.JPG", Text = "a\"b" }
]);
var lines = File.ReadAllLines(csvPath);
lines[0].ShouldBe("filename,text");
lines[1].ShouldBe("\"IMG_7146.JPG\",\"43,84,61\"");
lines[2].ShouldBe("\"IMG_7207.JPG\",\"a\"\"b\"");
}
private sealed class TempDirectory : IDisposable
{
public TempDirectory()
{
Path = System.IO.Path.Combine(System.IO.Path.GetTempPath(), System.IO.Path.GetRandomFileName());
Directory.CreateDirectory(Path);
}
public string Path { get; }
public void Dispose()
{
if (Directory.Exists(Path))
{
Directory.Delete(Path, recursive: true);
}
}
}
}

View file

@ -2,6 +2,8 @@
<PropertyGroup>
<TargetFramework>net10.0</TargetFramework>
<OutputType>Library</OutputType>
<Nullable>enable</Nullable>
<ImplicitUsings>enable</ImplicitUsings>
<GenerateAssemblyInfo>false</GenerateAssemblyInfo>
<PlatformTarget>x64</PlatformTarget>
<!-- WINDOWS preprocessor symbol mirrors the -windows TFM suffix so #if WINDOWS guards work -->

View file

@ -5,6 +5,7 @@
xmlns:views="clr-namespace:ImageCatalog_2.AvaloniaViews"
xmlns:iconPacks="https://github.com/MahApps/IconPacks.Avalonia"
x:Class="ImageCatalog_2.AvaloniaMainWindow"
x:CompileBindings="False"
mc:Ignorable="d"
Title="Image Catalog - Avalonia" Height="540" Width="800">
@ -106,7 +107,7 @@
</Button>
</StackPanel>
<Border Grid.Row="1" BorderThickness="1" Padding="10" MaxWidth="280" MinWidth="0">
<Grid RowDefinitions="Auto,Auto,Auto,Auto,Auto,Auto,Auto,Auto,*" RowSpacing="8" MinWidth="0">
<Grid RowDefinitions="Auto,Auto,Auto,Auto,Auto,Auto,Auto,Auto,*" MinWidth="0">
<StackPanel Grid.Row="0">
<Button HorizontalAlignment="Stretch" Margin="0,0,0,4" Command="{Binding LoadSettingsCommand}">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center">

View file

@ -5,6 +5,8 @@ using Avalonia.Layout;
using Avalonia.Platform.Storage;
using Avalonia.Styling;
using Avalonia.Threading;
using ImageCatalog_2.Services;
using Microsoft.Extensions.DependencyInjection;
using System.ComponentModel;
using System.IO;
@ -13,52 +15,73 @@ namespace ImageCatalog_2;
public partial class AvaloniaMainWindow : Window
{
private readonly DataModel _model;
private readonly PickerPreferenceService _pickerPreferenceService;
private bool _isDarkTheme;
private bool _startupSettingsRestoreAttempted;
public AvaloniaMainWindow()
: this(Program.ServiceProvider.GetRequiredService<DataModel>())
{
}
public AvaloniaMainWindow(DataModel model)
{
InitializeComponent();
_model = model;
_pickerPreferenceService = Program.ServiceProvider.GetRequiredService<PickerPreferenceService>();
DataContext = _model;
Opened += (_, _) => SyncThemeStateFromCurrentTheme();
Opened += async (_, _) =>
{
SyncThemeStateFromCurrentTheme();
await TryLoadLastSettingsOnStartupAsync();
};
Closing += AvaloniaMainWindow_Closing;
// Let DataModel marshal callbacks onto Avalonia UI thread.
_model.UiInvoker = action => Dispatcher.UIThread.Invoke(action);
_model.ConfirmAiCsvOverwriteAsync = ShowConfirmationDialogAsync;
_model.SelectSourceFolderRequested += async (_, _) =>
{
var suggestedStartLocation = await _pickerPreferenceService.TryGetStartFolderAsync(StorageProvider, PickerPreferenceKeys.SourceFolder, _model.SourcePath);
var folders = await StorageProvider.OpenFolderPickerAsync(new FolderPickerOpenOptions
{
Title = "Seleziona cartella sorgente"
Title = "Seleziona cartella sorgente",
SuggestedStartLocation = suggestedStartLocation
});
if (folders.Count > 0)
{
_model.SourcePath = folders[0].Path.LocalPath + Path.DirectorySeparatorChar;
_pickerPreferenceService.RememberPath(PickerPreferenceKeys.SourceFolder, _model.SourcePath);
}
};
_model.SelectDestinationFolderRequested += async (_, _) =>
{
var suggestedStartLocation = await _pickerPreferenceService.TryGetStartFolderAsync(StorageProvider, PickerPreferenceKeys.DestinationFolder, _model.DestinationPath);
var folders = await StorageProvider.OpenFolderPickerAsync(new FolderPickerOpenOptions
{
Title = "Seleziona cartella destinazione"
Title = "Seleziona cartella destinazione",
SuggestedStartLocation = suggestedStartLocation
});
if (folders.Count > 0)
{
_model.DestinationPath = folders[0].Path.LocalPath + Path.DirectorySeparatorChar;
_pickerPreferenceService.RememberPath(PickerPreferenceKeys.DestinationFolder, _model.DestinationPath);
}
};
_model.SelectLogoFileRequested += async (_, _) =>
{
var suggestedStartLocation = await _pickerPreferenceService.TryGetStartFolderAsync(StorageProvider, PickerPreferenceKeys.LogoFile, _model.LogoFile);
var files = await StorageProvider.OpenFilePickerAsync(new FilePickerOpenOptions
{
Title = "Seleziona logo",
SuggestedStartLocation = suggestedStartLocation,
FileTypeFilter =
[
new FilePickerFileType("Immagini") { Patterns = ["*.jpg", "*.jpeg", "*.png", "*.bmp", "*.gif"] }
@ -68,63 +91,78 @@ public partial class AvaloniaMainWindow : Window
if (files.Count > 0)
{
_model.LogoFile = files[0].Path.LocalPath;
_pickerPreferenceService.RememberPath(PickerPreferenceKeys.LogoFile, _model.LogoFile);
}
};
_model.SelectModelsFolderRequested += async (_, _) =>
{
var suggestedStartLocation = await _pickerPreferenceService.TryGetStartFolderAsync(StorageProvider, PickerPreferenceKeys.ModelsFolder, _model.ModelsFolderPath);
var folders = await StorageProvider.OpenFolderPickerAsync(new FolderPickerOpenOptions
{
Title = "Seleziona cartella modelli"
Title = "Seleziona cartella modelli",
SuggestedStartLocation = suggestedStartLocation
});
if (folders.Count > 0)
{
_model.ModelsFolderPath = folders[0].Path.LocalPath + Path.DirectorySeparatorChar;
_pickerPreferenceService.RememberPath(PickerPreferenceKeys.ModelsFolder, _model.ModelsFolderPath);
}
};
_model.SelectCsvOutputRequested += async (_, _) =>
{
var suggestedStartLocation = await _pickerPreferenceService.TryGetStartFolderAsync(StorageProvider, PickerPreferenceKeys.CsvOutput, _model.CsvOutputPath);
var file = await StorageProvider.SaveFilePickerAsync(new FilePickerSaveOptions
{
Title = "Salva CSV",
DefaultExtension = "csv",
FileTypeChoices = [new FilePickerFileType("CSV") { Patterns = ["*.csv"] }]
FileTypeChoices = [new FilePickerFileType("CSV") { Patterns = ["*.csv"] }],
SuggestedStartLocation = suggestedStartLocation
});
if (file is not null)
{
_model.CsvOutputPath = file.Path.LocalPath;
_pickerPreferenceService.RememberPath(PickerPreferenceKeys.CsvOutput, _model.CsvOutputPath);
}
};
_model.SaveSettingsRequested += async (_, _) =>
{
var suggestedStartLocation = await _pickerPreferenceService.TryGetStartFolderAsync(StorageProvider, PickerPreferenceKeys.SettingsFile);
var file = await StorageProvider.SaveFilePickerAsync(new FilePickerSaveOptions
{
Title = "Salva impostazioni",
DefaultExtension = "xml",
FileTypeChoices = [new FilePickerFileType("Setup") { Patterns = ["*.xml"] }]
FileTypeChoices = [new FilePickerFileType("Setup") { Patterns = ["*.xml"] }],
SuggestedStartLocation = suggestedStartLocation
});
if (file is not null)
{
await _model.SaveSettingsToFileAsync(file.Path.LocalPath);
_pickerPreferenceService.RememberPath(PickerPreferenceKeys.SettingsFile, file.Path.LocalPath);
_pickerPreferenceService.RememberValue(PickerPreferenceKeys.LastSettingsFile, file.Path.LocalPath);
}
};
_model.LoadSettingsRequested += async (_, _) =>
{
var suggestedStartLocation = await _pickerPreferenceService.TryGetStartFolderAsync(StorageProvider, PickerPreferenceKeys.SettingsFile);
var files = await StorageProvider.OpenFilePickerAsync(new FilePickerOpenOptions
{
Title = "Carica impostazioni",
FileTypeFilter = [new FilePickerFileType("Setup") { Patterns = ["*.xml"] }]
FileTypeFilter = [new FilePickerFileType("Setup") { Patterns = ["*.xml"] }],
SuggestedStartLocation = suggestedStartLocation
});
if (files.Count > 0)
{
await _model.LoadSettingsFromFileAsync(files[0].Path.LocalPath);
_pickerPreferenceService.RememberPath(PickerPreferenceKeys.SettingsFile, files[0].Path.LocalPath);
_pickerPreferenceService.RememberValue(PickerPreferenceKeys.LastSettingsFile, files[0].Path.LocalPath);
}
};
@ -146,9 +184,41 @@ public partial class AvaloniaMainWindow : Window
private bool _isStoppingFaceEncoderForClose;
private async Task TryLoadLastSettingsOnStartupAsync()
{
if (_startupSettingsRestoreAttempted)
{
return;
}
_startupSettingsRestoreAttempted = true;
var lastSettingsFile = _pickerPreferenceService.GetRememberedValue(PickerPreferenceKeys.LastSettingsFile);
if (string.IsNullOrWhiteSpace(lastSettingsFile))
{
return;
}
if (!File.Exists(lastSettingsFile))
{
_pickerPreferenceService.ForgetValue(PickerPreferenceKeys.LastSettingsFile);
return;
}
try
{
await _model.LoadSettingsFromFileAsync(lastSettingsFile);
_pickerPreferenceService.RememberPath(PickerPreferenceKeys.SettingsFile, lastSettingsFile);
}
catch (Exception ex)
{
await ShowMessageDialogAsync("Impostazioni", $"Impossibile caricare il file impostazioni automatico:\n{ex.GetBaseException().Message}");
}
}
private async void AvaloniaMainWindow_Closing(object? sender, CancelEventArgs e)
{
if (_isStoppingFaceEncoderForClose || !_model.IsFaceEncoderRunning)
if (_isStoppingFaceEncoderForClose || (!_model.IsFaceEncoderRunning && !_model.IsFaceMatcherRunning))
{
return;
}
@ -157,9 +227,17 @@ public partial class AvaloniaMainWindow : Window
_isStoppingFaceEncoderForClose = true;
try
{
if (_model.IsFaceMatcherRunning)
{
await _model.StopFaceMatcherAsync("Arresto face matcher in chiusura...", waitForExit: true);
}
if (_model.IsFaceEncoderRunning)
{
await _model.StopFaceEncoderAsync("Arresto face encoder in chiusura...", waitForExit: true);
}
}
finally
{
_isStoppingFaceEncoderForClose = false;
@ -206,6 +284,25 @@ public partial class AvaloniaMainWindow : Window
await dialog.ShowDialog(this);
}
private async Task<bool> ShowConfirmationDialogAsync(string title, string message)
{
var dialog = new Window
{
Title = title,
Width = 520,
CanResize = false,
WindowStartupLocation = WindowStartupLocation.CenterOwner,
SizeToContent = SizeToContent.Height
};
dialog.Content = BuildConfirmationDialogContent(
message,
() => dialog.Close(true),
() => dialog.Close(false));
return await dialog.ShowDialog<bool>(this);
}
private static Control BuildMessageDialogContent(string message, Action closeDialog)
{
var layout = new StackPanel
@ -233,4 +330,46 @@ public partial class AvaloniaMainWindow : Window
layout.Children.Add(closeButton);
return layout;
}
private static Control BuildConfirmationDialogContent(string message, Action confirmDialog, Action cancelDialog)
{
var layout = new StackPanel
{
Margin = new Thickness(16),
Spacing = 12
};
layout.Children.Add(new TextBlock
{
Text = message,
TextWrapping = Avalonia.Media.TextWrapping.Wrap,
MaxWidth = 460
});
var buttons = new StackPanel
{
Orientation = Orientation.Horizontal,
HorizontalAlignment = HorizontalAlignment.Right,
Spacing = 8
};
var cancelButton = new Button
{
Content = "Annulla",
MinWidth = 96
};
cancelButton.Click += (_, _) => cancelDialog();
var confirmButton = new Button
{
Content = "Sovrascrivi",
MinWidth = 96
};
confirmButton.Click += (_, _) => confirmDialog();
buttons.Children.Add(cancelButton);
buttons.Children.Add(confirmButton);
layout.Children.Add(buttons);
return layout;
}
}

View file

@ -1,25 +1,25 @@
<UserControl xmlns="https://github.com/avaloniaui"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:avaloniaDataGrid="clr-namespace:Avalonia.Controls;assembly=Avalonia.Controls.DataGrid"
xmlns:controls="clr-namespace:ImageCatalog_2.Controls"
xmlns:iconPacks="https://github.com/MahApps/IconPacks.Avalonia"
x:Class="ImageCatalog_2.AvaloniaViews.AiTabView">
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="*" />
</Grid.RowDefinitions>
x:Class="ImageCatalog_2.AvaloniaViews.AiTabView"
x:CompileBindings="False">
<TabControl Margin="4">
<TabItem Header="Esecuzione">
<Grid RowDefinitions="Auto,*">
<ScrollViewer Grid.Row="0" VerticalScrollBarVisibility="Auto">
<StackPanel Margin="4">
<StackPanel Margin="4" Spacing="8">
<TextBlock Text="AI / OCR" FontWeight="Bold" />
<StackPanel Orientation="Horizontal" Spacing="12" Margin="0,6,0,0">
<StackPanel Orientation="Horizontal" Spacing="12">
<CheckBox Content="Usa GPU"
IsChecked="{Binding UseNumberAiGpu, Mode=TwoWay}"
IsEnabled="{Binding NumberAiGpuOptionEnabled}" />
<CheckBox Content="Includi thumbnail" IsChecked="{Binding IncludeNumberAiThumbnails, Mode=TwoWay}" />
</StackPanel>
<Grid Margin="0,8,0,0" ColumnDefinitions="Auto,Auto,*" ColumnSpacing="8">
<Grid ColumnDefinitions="Auto,Auto,*" ColumnSpacing="8">
<TextBlock Grid.Column="0" Text="Carico OCR:" VerticalAlignment="Center" />
<ComboBox Grid.Column="1"
Width="84"
@ -32,38 +32,15 @@
FontWeight="SemiBold" />
</Grid>
<Grid Margin="0,8,0,0" ColumnDefinitions="Auto,*,Auto" ColumnSpacing="6">
<TextBlock Grid.Column="0" Text="Sorgente:" VerticalAlignment="Center" />
<TextBox Grid.Column="1" Text="{Binding DestinationPath, Mode=OneWay}" IsReadOnly="True" VerticalAlignment="Center" />
<Button Grid.Column="2" Width="72" Click="OpenAiDestinationFolder_Click">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
<controls:PathPickerField Label="Sorgente:"
Text="{Binding DestinationPath, Mode=TwoWay}"
IsTextReadOnly="True"
PreferenceKey="Picker.DestinationFolder.LastPath"
PickerTitle="Seleziona cartella sorgente AI"
PickerMode="Folder"
AppendDirectorySeparator="True" />
<TextBlock Text="Modelli" FontWeight="Bold" Margin="0,8,0,0" />
<Grid Margin="0,4,0,0" ColumnDefinitions="Auto,*,Auto,Auto">
<TextBlock Text="Cartella modelli:" VerticalAlignment="Center" Margin="0,0,8,0" Grid.Column="0" />
<TextBox Text="{Binding ModelsFolderPath, Mode=TwoWay}" Grid.Column="1" VerticalAlignment="Center" />
<Button Width="104" Margin="6,0,0,0" Command="{Binding SelectModelsFolderCommand}"
Grid.Column="2">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="FolderOpenOutline" Width="14" Height="14" />
<TextBlock Text="Scegli..." />
</StackPanel>
</Button>
<Button Width="72" Margin="6,0,0,0" Grid.Column="3"
Click="OpenModelsFolder_Click">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
<StackPanel Orientation="Horizontal" HorizontalAlignment="Left" Margin="0,8,0,0" Spacing="8">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Left" Spacing="8">
<Button Command="{Binding StartAiCommand}" Width="132">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="PlayCircle" Width="16" Height="16" Foreground="#2E7D32" />
@ -78,37 +55,48 @@
</Button>
</StackPanel>
<TextBlock Text="Output CSV" FontWeight="Bold" Margin="0,8,0,0" />
<Grid Margin="0,4,0,0" ColumnDefinitions="Auto,*,Auto,Auto">
<TextBlock Text="Percorso CSV:" VerticalAlignment="Center" Margin="0,0,8,0" Grid.Column="0" />
<TextBox Text="{Binding CsvOutputPath, Mode=TwoWay}" Grid.Column="1" VerticalAlignment="Center" />
<Button Width="104" Margin="6,0,0,0" Command="{Binding SelectCsvOutputCommand}"
Grid.Column="2">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="FileOutline" Width="14" Height="14" />
<TextBlock Text="Scegli..." />
</StackPanel>
</Button>
<Button Width="72" Margin="6,0,0,0" Grid.Column="3"
Click="OpenCsvOutputFolder_Click">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
<TextBlock Text="Output CSV" FontWeight="Bold" />
<controls:PathPickerField Label="Percorso CSV:"
Text="{Binding CsvOutputPath, Mode=TwoWay}"
PreferenceKey="Picker.CsvOutput.LastPath"
PickerTitle="Salva CSV"
PickerMode="SaveFile"
FileTypeName="CSV"
FilePatterns="*.csv"
DefaultExtension="csv" />
<TextBlock Text="Anteprima risultati" FontWeight="Bold" Margin="0,8,0,0" />
<ProgressBar Minimum="0" Maximum="100" Value="{Binding AiProgress}" Height="16" Margin="0,4,0,4" />
<TextBlock Text="Anteprima risultati" FontWeight="Bold" />
<ProgressBar Minimum="0" Maximum="100" Value="{Binding AiProgress}" Height="16" Margin="0,0,0,4" />
</StackPanel>
</ScrollViewer>
<avaloniaDataGrid:DataGrid Grid.Row="1" ItemsSource="{Binding PreviewResults}" IsReadOnly="True"
AutoGenerateColumns="False" Margin="4,4,4,4" CanUserResizeColumns="True" VerticalAlignment="Stretch">
<avaloniaDataGrid:DataGrid Grid.Row="1"
ItemsSource="{Binding PreviewResults}"
IsReadOnly="True"
AutoGenerateColumns="False"
Margin="4,4,4,4"
CanUserResizeColumns="True"
VerticalAlignment="Stretch">
<avaloniaDataGrid:DataGrid.Columns>
<avaloniaDataGrid:DataGridTextColumn Header="Path" Binding="{Binding Path}" Width="*" />
<avaloniaDataGrid:DataGridTextColumn Header="Text" Binding="{Binding Text}" Width="2*" />
</avaloniaDataGrid:DataGrid.Columns>
</avaloniaDataGrid:DataGrid>
</Grid>
</TabItem>
<TabItem Header="Impostazioni">
<ScrollViewer VerticalScrollBarVisibility="Auto">
<StackPanel Margin="8" Spacing="8">
<TextBlock Text="Modelli" FontWeight="Bold" />
<controls:PathPickerField Label="Cartella modelli:"
Text="{Binding ModelsFolderPath, Mode=TwoWay}"
PreferenceKey="Picker.ModelsFolder.LastPath"
PickerTitle="Seleziona cartella modelli"
PickerMode="Folder"
AppendDirectorySeparator="True" />
</StackPanel>
</ScrollViewer>
</TabItem>
</TabControl>
</UserControl>

View file

@ -1,8 +1,4 @@
using Avalonia.Controls;
using Avalonia.Interactivity;
using System.Diagnostics;
using System.IO;
namespace ImageCatalog_2.AvaloniaViews;
public partial class AiTabView : Avalonia.Controls.UserControl
@ -11,56 +7,4 @@ public partial class AiTabView : Avalonia.Controls.UserControl
{
InitializeComponent();
}
private void OpenModelsFolder_Click(object? sender, RoutedEventArgs e)
{
if (DataContext is DataModel model)
{
OpenInExplorer(model.ModelsFolderPath);
}
}
private void OpenCsvOutputFolder_Click(object? sender, RoutedEventArgs e)
{
if (DataContext is not DataModel model)
{
return;
}
var directory = Path.GetDirectoryName(model.CsvOutputPath);
OpenInExplorer(string.IsNullOrWhiteSpace(directory) ? model.CsvOutputPath : directory);
}
private void OpenAiDestinationFolder_Click(object? sender, RoutedEventArgs e)
{
if (DataContext is DataModel model)
{
OpenInExplorer(model.DestinationPath);
}
}
private static void OpenInExplorer(string? path)
{
if (string.IsNullOrWhiteSpace(path))
{
return;
}
var normalizedPath = path.Trim().Trim('"');
try
{
if (File.Exists(normalizedPath))
{
Process.Start("explorer.exe", $"/select,\"{normalizedPath}\"");
}
else if (Directory.Exists(normalizedPath))
{
Process.Start(new ProcessStartInfo { FileName = normalizedPath, UseShellExecute = true });
}
}
catch
{
// Ignore failures when opening Explorer.
}
}
}

View file

@ -1,86 +1,66 @@
<UserControl xmlns="https://github.com/avaloniaui"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:avaloniaDataGrid="clr-namespace:Avalonia.Controls;assembly=Avalonia.Controls.DataGrid"
xmlns:controls="clr-namespace:ImageCatalog_2.Controls"
xmlns:iconPacks="https://github.com/MahApps/IconPacks.Avalonia"
x:Class="ImageCatalog_2.AvaloniaViews.FaceAiTabView">
xmlns:converters="clr-namespace:ImageCatalog_2.Converters"
x:Class="ImageCatalog_2.AvaloniaViews.FaceAiTabView"
x:CompileBindings="False">
<UserControl.Resources>
<converters:FilePathToBitmapConverter x:Key="FilePathToBitmapConverter" />
</UserControl.Resources>
<TabControl Margin="4">
<TabItem Header="Encoder">
<TabControl Margin="0,4,0,0">
<TabItem Header="Esecuzione">
<ScrollViewer>
<StackPanel Margin="4" Spacing="6">
<TextBlock Text="Face Recognition Encoder" FontWeight="Bold" />
<TextBlock Text="Esegue il face encoder usando la cartella Destinazione corrente come --images e genera automaticamente file .pkl e log nella cartella di output scelta."
TextWrapping="Wrap" Opacity="0.8" />
<TextBlock Text="Cartella Face Encoder" FontWeight="Bold" Margin="0,4,0,0" />
<Grid ColumnDefinitions="Auto,*,Auto,Auto" ColumnSpacing="6">
<TextBlock Grid.Column="0" Text="Percorso:" VerticalAlignment="Center" />
<TextBox Grid.Column="1" Name="FaceExecutablePathTextBox" Text="{Binding FaceExecutablePath, Mode=TwoWay}" Watermark="C:\tools\Face_Recognition_Windows" />
<Button Grid.Column="2" Name="FaceSelectExecutableButton" Click="SelectFaceExecutable_Click" Width="104" Margin="6,0,0,0">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="FolderOutline" Width="14" Height="14" />
<TextBlock Text="Scegli..." />
</StackPanel>
</Button>
<Button Grid.Column="3" Name="FaceOpenExecutableButton" Click="OpenFaceExecutableFolder_Click" Width="72" Margin="6,0,0,0">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
<StackPanel Orientation="Horizontal" Spacing="12">
<CheckBox Content="Ricorsivo (--recursive)" IsChecked="{Binding FaceRecursive, Mode=TwoWay}" />
<CheckBox Content="Includi thumbnail (--include-tn)" IsChecked="{Binding FaceIncludeThumbnails, Mode=TwoWay}" />
<CheckBox Content="Upsample (--upsample)" IsChecked="{Binding FaceUpsample, Mode=TwoWay}" />
<WrapPanel>
<CheckBox Content="Ricorsivo (--recursive)" IsChecked="{Binding FaceRecursive, Mode=TwoWay}" Margin="0,0,12,6" />
<CheckBox Content="Includi thumbnail (--include-tn)" IsChecked="{Binding FaceIncludeThumbnails, Mode=TwoWay}" Margin="0,0,12,6" />
<CheckBox Content="Upsample (--upsample)" IsChecked="{Binding FaceUpsample, Mode=TwoWay}" Margin="0,0,12,6" />
<CheckBox Content="Usa GPU"
IsChecked="{Binding UseFaceGpu, Mode=TwoWay}"
IsEnabled="{Binding FaceGpuOptionEnabled}" />
</StackPanel>
<TextBlock Text="Seleziona la cartella base di Face Recognition Windows: l'app sceglie automaticamente face_encoder_cpu.exe o face_encoder_gpu.exe in base al checkbox GPU."
TextWrapping="Wrap"
Opacity="0.75" />
IsEnabled="{Binding FaceGpuOptionEnabled}"
Margin="0,0,12,6" />
</WrapPanel>
<Grid ColumnDefinitions="Auto,120,Auto,120,*" ColumnSpacing="6">
<TextBlock Grid.Column="0" Text="Parallelismo:" VerticalAlignment="Center" />
<ComboBox Grid.Column="1" ItemsSource="{Binding FaceParallelismOptions}" SelectedItem="{Binding FaceParallelism, Mode=TwoWay}" />
<TextBlock Grid.Column="2" Text="Min size:" VerticalAlignment="Center" />
<TextBox Grid.Column="3" Text="{Binding FaceMinSize, Mode=TwoWay}" Watermark="35" />
<TextBlock Grid.Column="4" Text="Usa --multicore in CPU e --multiprocess in GPU." VerticalAlignment="Center" Opacity="0.75" />
<TextBlock Grid.Column="4" Text="Usa --multicore in CPU e --multiprocess in GPU." VerticalAlignment="Center" TextWrapping="Wrap" Opacity="0.75" />
</Grid>
<Grid ColumnDefinitions="Auto,*,Auto,Auto" ColumnSpacing="6">
<TextBlock Grid.Column="0" Text="Sorgente:" VerticalAlignment="Center" />
<TextBox Grid.Column="1" Name="FaceDestinationPathTextBox" Text="{Binding DestinationPath, Mode=OneWay}" IsReadOnly="True" />
<Button Grid.Column="3" Name="FaceOpenDestinationButton" Click="OpenFaceDestinationFolder_Click" Width="72" Margin="6,0,0,0">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
<controls:PathPickerField Label="Sorgente:"
Text="{Binding DestinationPath, Mode=TwoWay}"
IsTextReadOnly="True"
PreferenceKey="Picker.DestinationFolder.LastPath"
PickerTitle="Seleziona cartella sorgente Face Encoder"
PickerMode="Folder"
AppendDirectorySeparator="True" />
<TextBlock Text="Output encodings" FontWeight="Bold" Margin="0,4,0,0" />
<Grid ColumnDefinitions="Auto,*,Auto,Auto" ColumnSpacing="6">
<TextBlock Grid.Column="0" Text="Cartella out:" VerticalAlignment="Center" />
<TextBox Grid.Column="1" Name="FaceOutputFolderTextBox" Text="{Binding FaceOutputFolderPath, Mode=TwoWay}" Watermark="C:\output\face_encoder" />
<Button Grid.Column="2" Name="FaceSelectOutputButton" Click="SelectFaceOutputFolder_Click" Width="104" Margin="6,0,0,0">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="FolderOutline" Width="14" Height="14" />
<TextBlock Text="Scegli..." />
</StackPanel>
</Button>
<Button Grid.Column="3" Name="FaceOpenOutputButton" Click="OpenFaceOutputFolder_Click" Width="72" Margin="6,0,0,0">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
<controls:PathPickerField Label="Cartella out:"
Text="{Binding FaceOutputFolderPath, Mode=TwoWay}"
Watermark="C:\output\face_encoder"
PreferenceKey="Picker.FaceOutputFolder.LastPath"
PickerTitle="Seleziona la cartella output per encodings e log"
PickerMode="Folder" />
<TextBlock Text="I file vengono creati come face_encodings_yyyyMMdd_HHmmss_nomecartella.pkl e encoder_log_yyyyMMdd_HHmmss_nomecartella.txt."
TextWrapping="Wrap"
Opacity="0.75" />
<StackPanel Orientation="Horizontal" Spacing="6" Margin="0,6,0,0">
<Button Name="FaceRunButton" Command="{Binding StartFaceEncoderCommand}">
<StackPanel Orientation="Horizontal" Spacing="8" VerticalAlignment="Center">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Left" Spacing="8" Margin="0,6,0,0">
<Button Name="FaceRunButton" Command="{Binding StartFaceEncoderCommand}" Width="176">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6" VerticalAlignment="Center">
<iconPacks:PackIconMaterial Kind="PlayCircle" Width="16" Height="16" Foreground="#2E7D32" />
<ProgressBar Width="18"
Height="18"
IsIndeterminate="True"
@ -90,8 +70,13 @@
<TextBlock Text="Esegui Face Encoder" VerticalAlignment="Center" />
</StackPanel>
</Button>
<Button Content="Stop" Command="{Binding StopFaceEncoderCommand}" />
<TextBlock VerticalAlignment="Center" Text="{Binding FaceStatusMessage}" />
<Button Command="{Binding StopFaceEncoderCommand}" Width="120">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Cancel" Width="16" Height="16" Foreground="#C62828" />
<TextBlock Text="Stop" />
</StackPanel>
</Button>
<TextBlock VerticalAlignment="Center" Text="{Binding FaceStatusMessage}" TextWrapping="Wrap" />
</StackPanel>
<TextBlock Text="Output comando" FontWeight="Bold" Margin="0,6,0,0" />
@ -106,4 +91,201 @@
ScrollViewer.HorizontalScrollBarVisibility="Auto" />
</StackPanel>
</ScrollViewer>
</TabItem>
<TabItem Header="Impostazioni">
<ScrollViewer>
<StackPanel Margin="4" Spacing="8">
<TextBlock Text="Cartella Face Encoder" FontWeight="Bold" />
<TextBlock Text="Seleziona la cartella base di face recognition"
TextWrapping="Wrap"
Opacity="0.8" />
<TextBlock Text="L'app sceglie automaticamente face_encoder_cpu.exe o face_encoder_gpu.exe in base al checkbox GPU."
TextWrapping="Wrap"
Opacity="0.75" />
<controls:PathPickerField Label="Percorso:"
Text="{Binding FaceExecutablePath, Mode=TwoWay}"
Watermark="C:\tools\Face_Recognition_Windows"
PreferenceKey="Picker.FaceExecutableFolder.LastPath"
PickerTitle="Seleziona la cartella Face Recognition Windows"
PickerMode="Folder" />
</StackPanel>
</ScrollViewer>
</TabItem>
</TabControl>
</TabItem>
<TabItem Header="Matcher Test">
<ScrollViewer>
<StackPanel Margin="4" Spacing="6">
<TextBlock Text="Face Matcher Test" FontWeight="Bold" />
<TextBlock Text="Esegue face_matcher.exe su un'immagine scelta a runtime e confronta i risultati con gli encodings .pkl. I match vengono poi risolti per nome file dentro la cartella Destinazione corrente per facilitare il debug visivo."
TextWrapping="Wrap" Opacity="0.8" />
<TextBlock Text="Eseguibile matcher" FontWeight="Bold" Margin="0,4,0,0" />
<controls:PathPickerField Label="Percorso:"
Text="{Binding FaceMatcherExecutablePath, Mode=TwoWay}"
Watermark="C:\tools\Face_Recognition_Windows\face_matcher.exe"
PreferenceKey="Picker.FaceMatcherExecutable.LastPath"
PickerTitle="Seleziona face_matcher.exe"
PickerMode="OpenFile"
FileTypeName="Eseguibile"
FilePatterns="*.exe" />
<TextBlock Text="Immagine da testare" FontWeight="Bold" Margin="0,4,0,0" />
<controls:PathPickerField Label="Immagine:"
Text="{Binding FaceMatcherSelectedImagePath, Mode=TwoWay}"
Watermark="Percorso immagine da usare per il match"
PreferenceKey="Picker.FaceMatcherImage.LastPath"
PickerTitle="Seleziona immagine per il match"
PickerMode="OpenFile"
FileTypeName="Immagini"
FilePatterns="*.jpg;*.jpeg;*.png;*.bmp;*.gif;*.webp" />
<TextBlock Text="Questa immagine resta solo runtime e non viene salvata nel file impostazioni."
TextWrapping="Wrap"
Opacity="0.75" />
<StackPanel Spacing="4">
<TextBlock Text="Preview immagine selezionata" FontWeight="SemiBold" />
<Border BorderThickness="1" BorderBrush="#808080" Padding="4" HorizontalAlignment="Left">
<Image Width="184"
Height="120"
Stretch="UniformToFill"
Source="{Binding FaceMatcherSelectedImagePath, Converter={StaticResource FilePathToBitmapConverter}}" />
</Border>
</StackPanel>
<controls:PathPickerField Label="Destinazione attuale:"
Text="{Binding DestinationPath, Mode=TwoWay}"
IsTextReadOnly="True"
PreferenceKey="Picker.DestinationFolder.LastPath"
PickerTitle="Seleziona cartella destinazione Face Matcher"
PickerMode="Folder"
AppendDirectorySeparator="True" />
<TextBlock Text="Encodings e file di appoggio" FontWeight="Bold" Margin="0,4,0,0" />
<controls:PathPickerField Label="Encodings .pkl:"
Text="{Binding FaceMatcherEncodingsPath, Mode=TwoWay}"
Watermark="Se vuoto usa l'ultimo .pkl in output encodings"
PreferenceKey="Picker.FaceMatcherEncodings.LastPath"
PickerTitle="Seleziona file encodings .pkl"
PickerMode="OpenFile"
FileTypeName="Encodings"
FilePatterns="*.pkl" />
<controls:PathPickerField Label="Risultato CSV:"
Text="{Binding FaceMatcherOutputPath, Mode=TwoWay}"
Watermark="Opzionale: file/cartella output CSV"
PreferenceKey="Picker.FaceMatcherOutput.LastPath"
PickerTitle="Seleziona output CSV del matcher"
PickerMode="SaveFile"
FileTypeName="CSV"
FilePatterns="*.csv"
DefaultExtension="csv" />
<controls:PathPickerField Label="Log matcher:"
Text="{Binding FaceMatcherLogPath, Mode=TwoWay}"
Watermark="Opzionale: file/cartella log txt"
PreferenceKey="Picker.FaceMatcherLog.LastPath"
PickerTitle="Seleziona log TXT del matcher"
PickerMode="SaveFile"
FileTypeName="Log"
FilePatterns="*.txt;*.log"
DefaultExtension="txt" />
<Grid ColumnDefinitions="Auto,100,*" ColumnSpacing="6">
<TextBlock Grid.Column="0" Text="Tolleranza:" VerticalAlignment="Center" />
<TextBox Grid.Column="1" Text="{Binding FaceMatcherTolerance, Mode=TwoWay}" Watermark="0.50" />
<TextBlock Grid.Column="2" Text="Range utile: 0.35 - 0.75. Più bassa = match più restrittivo, più alta = più permissivo." VerticalAlignment="Center" TextWrapping="Wrap" Opacity="0.75" />
</Grid>
<StackPanel Orientation="Horizontal" HorizontalAlignment="Left" Spacing="8" Margin="0,6,0,0">
<Button Command="{Binding StartFaceMatcherCommand}" Width="176">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6" VerticalAlignment="Center">
<iconPacks:PackIconMaterial Kind="PlayCircle" Width="16" Height="16" Foreground="#2E7D32" />
<ProgressBar Width="18"
Height="18"
IsIndeterminate="True"
IsVisible="{Binding IsFaceMatcherRunning}"
ShowProgressText="False"
VerticalAlignment="Center" />
<TextBlock Text="Avvia Face Matcher" VerticalAlignment="Center" />
</StackPanel>
</Button>
<Button Command="{Binding StopFaceMatcherCommand}" Width="120">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Cancel" Width="16" Height="16" Foreground="#C62828" />
<TextBlock Text="Stop" />
</StackPanel>
</Button>
<TextBlock VerticalAlignment="Center" Text="{Binding FaceMatcherStatusMessage}" TextWrapping="Wrap" />
</StackPanel>
<TextBlock Text="Output comando" FontWeight="Bold" Margin="0,6,0,0" />
<TextBox Name="FaceMatcherOutputTextBox"
Text="{Binding FaceMatcherCommandOutput}"
IsReadOnly="True"
AcceptsReturn="True"
TextWrapping="Wrap"
FontFamily="Cascadia Mono, Consolas, monospace"
Height="220"
ScrollViewer.VerticalScrollBarVisibility="Auto"
ScrollViewer.HorizontalScrollBarVisibility="Auto" />
<TextBlock Text="Risultati" FontWeight="Bold" Margin="0,8,0,0" />
<avaloniaDataGrid:DataGrid ItemsSource="{Binding FaceMatcherResults}"
IsReadOnly="True"
AutoGenerateColumns="False"
CanUserResizeColumns="True"
DoubleTapped="FaceMatcherResults_DoubleTapped"
MinHeight="220">
<avaloniaDataGrid:DataGrid.Columns>
<avaloniaDataGrid:DataGridTextColumn Header="File" Binding="{Binding PhotoId}" Width="2*" />
<avaloniaDataGrid:DataGridTextColumn Header="Score" Binding="{Binding ScoreDisplay}" Width="Auto" />
<avaloniaDataGrid:DataGridTextColumn Header="Trovato in destinazione" Binding="{Binding ResolvedImagePath}" Width="3*" />
<avaloniaDataGrid:DataGridTemplateColumn Header="Candidati" Width="Auto">
<avaloniaDataGrid:DataGridTemplateColumn.CellTemplate>
<DataTemplate>
<TextBlock Text="{Binding CandidateCount}" Margin="10,0,0,0" />
</DataTemplate>
</avaloniaDataGrid:DataGridTemplateColumn.CellTemplate>
</avaloniaDataGrid:DataGridTemplateColumn>
<avaloniaDataGrid:DataGridTextColumn Header="Debug" Binding="{Binding DebugSummary}" Width="4*" />
</avaloniaDataGrid:DataGrid.Columns>
</avaloniaDataGrid:DataGrid>
<TextBlock Text="Gallery match risolti" FontWeight="Bold" Margin="0,8,0,0" />
<ItemsControl ItemsSource="{Binding FaceMatcherResults}">
<ItemsControl.ItemsPanel>
<ItemsPanelTemplate>
<WrapPanel />
</ItemsPanelTemplate>
</ItemsControl.ItemsPanel>
<ItemsControl.ItemTemplate>
<DataTemplate>
<Button Click="OpenFaceMatcherPreview_Click"
Tag="{Binding}"
Width="184"
Margin="0,0,8,8"
HorizontalContentAlignment="Stretch"
VerticalContentAlignment="Stretch">
<StackPanel Spacing="4">
<Border BorderThickness="1" BorderBrush="#808080" Padding="4">
<Image Width="168"
Height="112"
Stretch="UniformToFill"
Source="{Binding ResolvedImagePath, Converter={StaticResource FilePathToBitmapConverter}}" />
</Border>
<TextBlock Text="{Binding PhotoId}" FontWeight="SemiBold" TextWrapping="Wrap" />
<TextBlock Text="{Binding ScoreDisplay}" Opacity="0.8" />
<TextBlock Text="{Binding ResolvedImagePath}" TextWrapping="Wrap" MaxHeight="36" Opacity="0.8" />
</StackPanel>
</Button>
</DataTemplate>
</ItemsControl.ItemTemplate>
</ItemsControl>
</StackPanel>
</ScrollViewer>
</TabItem>
</TabControl>
</UserControl>

View file

@ -1,11 +1,17 @@
using Avalonia.Controls;
using Avalonia.Interactivity;
using Avalonia.Platform.Storage;
using Avalonia.Input;
using Avalonia.Layout;
using Avalonia.Media;
using Avalonia.Media.Imaging;
using Avalonia.Threading;
using ImageCatalog_2.Models;
using ImageCatalog_2.Services;
using System;
using System.ComponentModel;
using System.Diagnostics;
using System.IO;
using System.Text;
using System.Threading.Tasks;
namespace ImageCatalog_2.AvaloniaViews;
@ -35,12 +41,21 @@ public partial class FaceAiTabView : Avalonia.Controls.UserControl
private void OnFaceAiPropertyChanged(object? sender, PropertyChangedEventArgs e)
{
if (!string.Equals(e.PropertyName, nameof(DataModel.FaceCommandOutput), StringComparison.Ordinal))
if (string.Equals(e.PropertyName, nameof(DataModel.FaceCommandOutput), StringComparison.Ordinal))
{
ScrollOutputTextBoxToEnd("FaceOutputTextBox");
return;
}
var outputBox = this.FindControl<Avalonia.Controls.TextBox>("FaceOutputTextBox");
if (string.Equals(e.PropertyName, nameof(DataModel.FaceMatcherCommandOutput), StringComparison.Ordinal))
{
ScrollOutputTextBoxToEnd("FaceMatcherOutputTextBox");
}
}
private void ScrollOutputTextBoxToEnd(string controlName)
{
var outputBox = this.FindControl<Avalonia.Controls.TextBox>(controlName);
if (outputBox is null)
{
return;
@ -53,172 +68,258 @@ public partial class FaceAiTabView : Avalonia.Controls.UserControl
});
}
private async void SelectFaceExecutable_Click(object? sender, RoutedEventArgs e)
private async void OpenFaceMatcherPreview_Click(object? sender, RoutedEventArgs e)
{
var executableBox = this.FindControl<Avalonia.Controls.TextBox>("FaceExecutablePathTextBox");
if (executableBox is null)
if (sender is not Button { Tag: FaceMatcherResultItem item })
{
return;
}
var topLevel = TopLevel.GetTopLevel(this);
var storageProvider = topLevel?.StorageProvider;
if (storageProvider is null)
await OpenFaceMatcherPreviewAsync(item);
}
private async Task OpenFaceMatcherPreviewAsync(FaceMatcherResultItem item)
{
var owner = TopLevel.GetTopLevel(this) as Window;
var dialog = BuildFaceMatcherPreviewDialog(item);
if (owner is not null)
{
await dialog.ShowDialog(owner);
return;
}
dialog.Show();
}
private async void FaceMatcherResults_DoubleTapped(object? sender, TappedEventArgs e)
{
if (sender is not Avalonia.Controls.DataGrid { SelectedItem: FaceMatcherResultItem item })
{
return;
}
var folders = await storageProvider.OpenFolderPickerAsync(new FolderPickerOpenOptions
{
Title = "Seleziona la cartella Face Recognition Windows"
});
if (folders.Count > 0)
{
executableBox.Text = folders[0].Path.LocalPath;
if (DataContext is DataModel model)
{
model.FaceExecutablePath = executableBox.Text;
}
}
await OpenFaceMatcherPreviewAsync(item);
}
private async void SelectFaceOutputFolder_Click(object? sender, RoutedEventArgs e)
private Window BuildFaceMatcherPreviewDialog(FaceMatcherResultItem item)
{
var outputBox = this.FindControl<Avalonia.Controls.TextBox>("FaceOutputFolderTextBox");
if (outputBox is null)
var dialog = new Window
{
return;
}
Title = $"Preview match: {item.PhotoId}",
Width = 1180,
Height = 900,
WindowStartupLocation = WindowStartupLocation.CenterOwner
};
var topLevel = TopLevel.GetTopLevel(this);
var storageProvider = topLevel?.StorageProvider;
if (storageProvider is null)
Bitmap? bitmap = null;
var dimensionText = "n/d";
if (!string.IsNullOrWhiteSpace(item.ResolvedImagePath) && File.Exists(item.ResolvedImagePath))
{
return;
}
var folders = await storageProvider.OpenFolderPickerAsync(new FolderPickerOpenOptions
{
Title = "Seleziona la cartella output per encodings e log"
});
if (folders.Count > 0)
{
outputBox.Text = folders[0].Path.LocalPath;
if (DataContext is DataModel model)
{
model.FaceOutputFolderPath = outputBox.Text;
}
}
}
private void OpenFaceExecutableFolder_Click(object? sender, RoutedEventArgs e)
{
var executableBox = this.FindControl<Avalonia.Controls.TextBox>("FaceExecutablePathTextBox");
if (executableBox is null)
{
return;
}
var path = executableBox.Text?.Trim();
if (string.IsNullOrWhiteSpace(path))
{
return;
}
if (Directory.Exists(path))
{
OpenInExplorer(path);
return;
}
if (File.Exists(path))
{
OpenInExplorer(path);
return;
}
var directory = Path.GetDirectoryName(path);
OpenInExplorer(string.IsNullOrWhiteSpace(directory) ? path : directory);
}
private void OpenFaceOutputFolder_Click(object? sender, RoutedEventArgs e)
{
var outputBox = this.FindControl<Avalonia.Controls.TextBox>("FaceOutputFolderTextBox");
if (outputBox is null)
{
return;
}
var outputPath = outputBox.Text?.Trim();
if (string.IsNullOrWhiteSpace(outputPath))
{
return;
}
if (Directory.Exists(outputPath))
{
OpenInExplorer(outputPath);
return;
}
if (File.Exists(outputPath))
{
OpenInExplorer(outputPath);
return;
}
var directory = Path.GetDirectoryName(outputPath);
OpenInExplorer(string.IsNullOrWhiteSpace(directory) ? outputPath : directory);
}
private void OpenFaceDestinationFolder_Click(object? sender, RoutedEventArgs e)
{
var destBox = this.FindControl<Avalonia.Controls.TextBox>("FaceDestinationPathTextBox");
string? path = destBox?.Text?.Trim();
if (string.IsNullOrWhiteSpace(path) && DataContext is DataModel model)
{
path = (model.DestinationPath ?? string.Empty).Trim();
}
if (string.IsNullOrWhiteSpace(path))
{
return;
}
if (Directory.Exists(path))
{
OpenInExplorer(path);
return;
}
var directory = Path.GetDirectoryName(path);
OpenInExplorer(string.IsNullOrWhiteSpace(directory) ? path : directory);
}
private static void OpenInExplorer(string? path)
{
if (string.IsNullOrWhiteSpace(path))
{
return;
}
var normalizedPath = path.Trim().Trim('"');
try
{
if (File.Exists(normalizedPath))
{
Process.Start("explorer.exe", $"/select,\"{normalizedPath}\"");
}
else if (Directory.Exists(normalizedPath))
{
Process.Start(new ProcessStartInfo { FileName = normalizedPath, UseShellExecute = true });
}
bitmap = new Bitmap(item.ResolvedImagePath);
dimensionText = $"{bitmap.PixelSize.Width} x {bitmap.PixelSize.Height}px";
}
catch
{
// Ignore failures when opening Explorer.
bitmap = null;
}
}
var fileInfo = !string.IsNullOrWhiteSpace(item.ResolvedImagePath) && File.Exists(item.ResolvedImagePath)
? new FileInfo(item.ResolvedImagePath)
: null;
var debugBuilder = new StringBuilder();
debugBuilder.AppendLine($"File matcher: {item.PhotoId}");
debugBuilder.AppendLine($"Score: {item.ScoreDisplay}");
debugBuilder.AppendLine($"Path risolto: {item.ResolvedImagePath}");
debugBuilder.AppendLine($"Candidati trovati in destinazione: {item.CandidateCount}");
debugBuilder.AppendLine($"Dimensioni immagine: {dimensionText}");
if (fileInfo is not null)
{
debugBuilder.AppendLine($"Dimensione file: {fileInfo.Length / 1024.0:F1} KB");
debugBuilder.AppendLine($"Ultima modifica: {fileInfo.LastWriteTime:yyyy-MM-dd HH:mm:ss}");
}
debugBuilder.AppendLine($"Immagine ricerca: {item.SearchImagePath}");
debugBuilder.AppendLine($"CSV risultati: {item.CsvPath}");
debugBuilder.AppendLine($"Log matcher: {item.LogPath}");
if (!string.IsNullOrWhiteSpace(item.DebugSummary))
{
debugBuilder.AppendLine($"Dettagli riga: {item.DebugSummary}");
}
if (!string.IsNullOrWhiteSpace(item.RawRow))
{
debugBuilder.AppendLine($"Raw CSV: {item.RawRow}");
}
var layout = new Grid
{
Margin = new Avalonia.Thickness(16),
RowDefinitions = new RowDefinitions("Auto,*,Auto")
};
var header = new StackPanel { Spacing = 6 };
header.Children.Add(new TextBlock
{
Text = item.PhotoId,
FontWeight = FontWeight.Bold,
FontSize = 18
});
header.Children.Add(new TextBlock
{
Text = string.IsNullOrWhiteSpace(item.ScoreDisplay)
? "Score: n/d"
: $"Score: {item.ScoreDisplay}%",
FontWeight = FontWeight.SemiBold,
Opacity = 0.9
});
header.Children.Add(new TextBlock
{
Text = string.IsNullOrWhiteSpace(item.ResolvedImagePath)
? "Nessun file immagine risolto nella cartella Destinazione."
: item.ResolvedImagePath,
TextWrapping = TextWrapping.Wrap,
Opacity = 0.8
});
layout.Children.Add(header);
var contentGrid = new Grid
{
Margin = new Avalonia.Thickness(0, 12, 0, 12),
RowDefinitions = new RowDefinitions("Auto,*,Auto")
};
Grid.SetRow(contentGrid, 1);
var zoomLevel = 1.0;
var zoomText = new TextBlock
{
Text = "100%",
VerticalAlignment = VerticalAlignment.Center,
MinWidth = 52,
TextAlignment = TextAlignment.Center
};
var imageControl = bitmap is null
? null
: new Image
{
Source = bitmap,
Stretch = Stretch.None,
HorizontalAlignment = HorizontalAlignment.Left,
VerticalAlignment = VerticalAlignment.Top,
RenderTransform = new ScaleTransform(1, 1)
};
void UpdateZoom(double delta)
{
if (imageControl is null)
{
return;
}
zoomLevel = Math.Clamp(zoomLevel + delta, 0.1, 8.0);
imageControl.RenderTransform = new ScaleTransform(zoomLevel, zoomLevel);
zoomText.Text = $"{zoomLevel * 100:0}%";
}
var toolbar = new StackPanel
{
Orientation = Orientation.Horizontal,
Spacing = 8,
Margin = new Avalonia.Thickness(0, 0, 0, 12)
};
var zoomOutButton = new Button { Content = "Zoom -", MinWidth = 80, IsEnabled = imageControl is not null };
zoomOutButton.Click += (_, _) => UpdateZoom(-0.1);
toolbar.Children.Add(zoomOutButton);
var zoomInButton = new Button { Content = "Zoom +", MinWidth = 80, IsEnabled = imageControl is not null };
zoomInButton.Click += (_, _) => UpdateZoom(0.1);
toolbar.Children.Add(zoomInButton);
var resetZoomButton = new Button { Content = "100%", MinWidth = 72, IsEnabled = imageControl is not null };
resetZoomButton.Click += (_, _) =>
{
if (imageControl is null)
{
return;
}
zoomLevel = 1.0;
imageControl.RenderTransform = new ScaleTransform(1, 1);
zoomText.Text = "100%";
};
toolbar.Children.Add(resetZoomButton);
toolbar.Children.Add(zoomText);
contentGrid.Children.Add(toolbar);
var previewBorder = new Border
{
BorderBrush = Brushes.Gray,
BorderThickness = new Avalonia.Thickness(1),
Padding = new Avalonia.Thickness(8),
Child = new ScrollViewer
{
HorizontalScrollBarVisibility = Avalonia.Controls.Primitives.ScrollBarVisibility.Auto,
VerticalScrollBarVisibility = Avalonia.Controls.Primitives.ScrollBarVisibility.Auto,
Content = bitmap is null
? new TextBlock
{
Text = "Anteprima non disponibile",
HorizontalAlignment = HorizontalAlignment.Center,
VerticalAlignment = VerticalAlignment.Center
}
: imageControl
}
};
Grid.SetRow(previewBorder, 1);
contentGrid.Children.Add(previewBorder);
var debugBox = new TextBox
{
Text = debugBuilder.ToString(),
IsReadOnly = true,
AcceptsReturn = true,
TextWrapping = TextWrapping.Wrap,
FontFamily = new FontFamily("Cascadia Mono, Consolas, monospace"),
MinHeight = 180,
Margin = new Avalonia.Thickness(0, 12, 0, 0)
};
Grid.SetRow(debugBox, 2);
contentGrid.Children.Add(debugBox);
layout.Children.Add(contentGrid);
var footer = new StackPanel
{
Orientation = Orientation.Horizontal,
HorizontalAlignment = HorizontalAlignment.Right,
Spacing = 8
};
Grid.SetRow(footer, 2);
var openFileButton = new Button { Content = "Apri file" };
openFileButton.Click += (_, _) => PathShellService.OpenInExplorer(item.ResolvedImagePath);
footer.Children.Add(openFileButton);
var openFolderButton = new Button { Content = "Apri cartella" };
openFolderButton.Click += (_, _) =>
{
var directory = Path.GetDirectoryName(item.ResolvedImagePath);
PathShellService.OpenInExplorer(string.IsNullOrWhiteSpace(directory) ? item.ResolvedImagePath : directory);
};
footer.Children.Add(openFolderButton);
var closeButton = new Button { Content = "Chiudi", MinWidth = 88 };
closeButton.Click += (_, _) => dialog.Close();
footer.Children.Add(closeButton);
layout.Children.Add(footer);
dialog.Content = layout;
return dialog;
}
}

View file

@ -1,43 +1,24 @@
<UserControl xmlns="https://github.com/avaloniaui"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:iconPacks="https://github.com/MahApps/IconPacks.Avalonia"
xmlns:controls="clr-namespace:ImageCatalog_2.Controls"
x:Class="ImageCatalog_2.AvaloniaViews.GeneralTabView">
<ScrollViewer>
<StackPanel Margin="4" Spacing="8">
<TextBlock Text="Percorsi" FontWeight="Bold" />
<StackPanel Margin="0,2,0,0" Spacing="6">
<Grid Margin="0,0,0,2" ColumnDefinitions="Auto,*,Auto,Auto" ColumnSpacing="6">
<TextBlock Text="Sorgente:" VerticalAlignment="Center" Margin="0,0,8,0" Grid.Column="0" />
<TextBox Text="{Binding SourcePath, Mode=TwoWay}" Grid.Column="1" VerticalAlignment="Center" />
<Button Width="104" Command="{Binding SelectSourceFolderCommand}" Grid.Column="2">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="FolderOpenOutline" Width="14" Height="14" />
<TextBlock Text="Scegli..." />
</StackPanel>
</Button>
<Button Width="72" Grid.Column="3" Click="OpenSourceFolder_Click">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
<Grid ColumnDefinitions="Auto,*,Auto,Auto" ColumnSpacing="6">
<TextBlock Text="Destinazione:" VerticalAlignment="Center" Margin="0,0,8,0" Grid.Column="0" />
<TextBox Text="{Binding DestinationPath, Mode=TwoWay}" Grid.Column="1" VerticalAlignment="Center" />
<Button Width="104" Command="{Binding SelectDestinationFolderCommand}" Grid.Column="2">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="FolderOpenOutline" Width="14" Height="14" />
<TextBlock Text="Scegli..." />
</StackPanel>
</Button>
<Button Width="72" Grid.Column="3" Click="OpenDestinationFolder_Click">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
<controls:PathPickerField Margin="0,0,0,2"
Label="Sorgente:"
Text="{Binding SourcePath, Mode=TwoWay}"
PreferenceKey="Picker.SourceFolder.LastPath"
PickerTitle="Seleziona cartella sorgente"
PickerMode="Folder"
AppendDirectorySeparator="True" />
<controls:PathPickerField Label="Destinazione:"
Text="{Binding DestinationPath, Mode=TwoWay}"
PreferenceKey="Picker.DestinationFolder.LastPath"
PickerTitle="Seleziona cartella destinazione"
PickerMode="Folder"
AppendDirectorySeparator="True" />
</StackPanel>
<Grid ColumnDefinitions="*,*" ColumnSpacing="24" Margin="0,4,0,0">

View file

@ -1,9 +1,4 @@
using Avalonia.Controls;
using Avalonia.Interactivity;
using System;
using System.Diagnostics;
using System.IO;
namespace ImageCatalog_2.AvaloniaViews;
public partial class GeneralTabView : Avalonia.Controls.UserControl
@ -12,45 +7,4 @@ public partial class GeneralTabView : Avalonia.Controls.UserControl
{
InitializeComponent();
}
private void OpenSourceFolder_Click(object? sender, RoutedEventArgs e)
{
if (DataContext is DataModel model)
{
OpenInExplorer(model.SourcePath);
}
}
private void OpenDestinationFolder_Click(object? sender, RoutedEventArgs e)
{
if (DataContext is DataModel model)
{
OpenInExplorer(model.DestinationPath);
}
}
private static void OpenInExplorer(string? path)
{
if (string.IsNullOrWhiteSpace(path))
{
return;
}
var normalizedPath = path.Trim().Trim('"');
try
{
if (File.Exists(normalizedPath))
{
Process.Start("explorer.exe", $"/select,\"{normalizedPath}\"");
}
else if (Directory.Exists(normalizedPath))
{
Process.Start(new ProcessStartInfo { FileName = normalizedPath, UseShellExecute = true });
}
}
catch
{
// Ignore failures when opening Explorer.
}
}
}

View file

@ -7,7 +7,7 @@
<TextBlock Text="Flusso: login admin, creazione gara, creazione punti foto, upload file processati da cartella destinazione locale, indicizzazione punti foto."
TextWrapping="Wrap" Opacity="0.8" />
<Grid ColumnDefinitions="Auto,*" RowDefinitions="Auto,Auto" ColumnSpacing="6" RowSpacing="6">
<Grid ColumnDefinitions="Auto,*" RowDefinitions="Auto,Auto" ColumnSpacing="6">
<TextBlock Grid.Row="0" Grid.Column="0" Text="Login:" VerticalAlignment="Center" />
<TextBox Grid.Row="0" Grid.Column="1" Name="ApiLoginTextBox" Text="{Binding ApiLogin, Mode=TwoWay}" Watermark="admin user" />
@ -16,7 +16,7 @@
</Grid>
<TextBlock Text="Dati gara" FontWeight="Bold" Margin="0,4,0,0" />
<Grid ColumnDefinitions="Auto,*,Auto,*" RowDefinitions="Auto,Auto,Auto,Auto" ColumnSpacing="6" RowSpacing="6">
<Grid ColumnDefinitions="Auto,*,Auto,*" RowDefinitions="Auto,Auto,Auto,Auto" ColumnSpacing="6">
<TextBlock Grid.Row="0" Grid.Column="0" Text="Descrizione:" VerticalAlignment="Center" />
<TextBox Grid.Row="0" Grid.Column="1" Name="ApiRaceDescriptionTextBox" Text="{Binding ApiRaceDescription, Mode=TwoWay}" Watermark="Nome gara" />
@ -49,7 +49,7 @@
</ComboBox>
</Grid>
<Grid ColumnDefinitions="Auto,*,Auto,*" RowDefinitions="Auto,Auto" ColumnSpacing="6" RowSpacing="6">
<Grid ColumnDefinitions="Auto,*,Auto,*" RowDefinitions="Auto,Auto" ColumnSpacing="6">
<TextBlock Grid.Row="0" Grid.Column="0" Text="Evento Omaggio:" VerticalAlignment="Center" />
<ComboBox Grid.Row="0" Grid.Column="1" Name="ApiFreeEventComboBox" SelectedIndex="{Binding ApiFreeEventIndex, Mode=TwoWay}">
<ComboBoxItem Content="0 - No" />

View file

@ -0,0 +1,34 @@
<UserControl xmlns="https://github.com/avaloniaui"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:iconPacks="https://github.com/MahApps/IconPacks.Avalonia"
x:Class="ImageCatalog_2.Controls.PathPickerField"
x:Name="Root">
<Grid ColumnDefinitions="Auto,*,Auto,Auto" ColumnSpacing="6">
<TextBlock Grid.Column="0"
Text="{Binding Label, ElementName=Root}"
VerticalAlignment="Center"
Margin="0,0,8,0" />
<TextBox Grid.Column="1"
Text="{Binding Text, ElementName=Root, Mode=TwoWay}"
IsReadOnly="{Binding IsTextReadOnly, ElementName=Root}"
Watermark="{Binding Watermark, ElementName=Root}"
VerticalAlignment="Center" />
<Button Grid.Column="2"
Width="104"
IsVisible="{Binding ShowPickerButton, ElementName=Root}"
Click="PickPath_Click">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="FolderOpenOutline" Width="14" Height="14" />
<TextBlock Text="Scegli..." />
</StackPanel>
</Button>
<Button Grid.Column="3"
Width="72"
Click="OpenPath_Click">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Spacing="6">
<iconPacks:PackIconMaterial Kind="Folder" Width="14" Height="14" />
<TextBlock Text="Apri" />
</StackPanel>
</Button>
</Grid>
</UserControl>

View file

@ -0,0 +1,285 @@
using Avalonia;
using Avalonia.Controls;
using Avalonia.Data;
using Avalonia.Platform.Storage;
using ImageCatalog_2.Services;
using Microsoft.Extensions.DependencyInjection;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
namespace ImageCatalog_2.Controls;
public partial class PathPickerField : UserControl
{
private readonly PickerPreferenceService _pickerPreferenceService;
public static readonly StyledProperty<string> LabelProperty =
AvaloniaProperty.Register<PathPickerField, string>(nameof(Label), string.Empty);
public static readonly StyledProperty<string> TextProperty =
AvaloniaProperty.Register<PathPickerField, string>(nameof(Text), string.Empty, defaultBindingMode: BindingMode.TwoWay);
public static readonly StyledProperty<string> WatermarkProperty =
AvaloniaProperty.Register<PathPickerField, string>(nameof(Watermark), string.Empty);
public static readonly StyledProperty<string> PreferenceKeyProperty =
AvaloniaProperty.Register<PathPickerField, string>(nameof(PreferenceKey), string.Empty);
public static readonly StyledProperty<string> PickerTitleProperty =
AvaloniaProperty.Register<PathPickerField, string>(nameof(PickerTitle), string.Empty);
public static readonly StyledProperty<string> FileTypeNameProperty =
AvaloniaProperty.Register<PathPickerField, string>(nameof(FileTypeName), string.Empty);
public static readonly StyledProperty<string> FilePatternsProperty =
AvaloniaProperty.Register<PathPickerField, string>(nameof(FilePatterns), string.Empty);
public static readonly StyledProperty<string> DefaultExtensionProperty =
AvaloniaProperty.Register<PathPickerField, string>(nameof(DefaultExtension), string.Empty);
public static readonly StyledProperty<PathPickerSelectionMode> PickerModeProperty =
AvaloniaProperty.Register<PathPickerField, PathPickerSelectionMode>(nameof(PickerMode), PathPickerSelectionMode.Folder);
public static readonly StyledProperty<bool> IsTextReadOnlyProperty =
AvaloniaProperty.Register<PathPickerField, bool>(nameof(IsTextReadOnly), false);
public static readonly StyledProperty<bool> ShowPickerButtonProperty =
AvaloniaProperty.Register<PathPickerField, bool>(nameof(ShowPickerButton), true);
public static readonly StyledProperty<bool> AppendDirectorySeparatorProperty =
AvaloniaProperty.Register<PathPickerField, bool>(nameof(AppendDirectorySeparator), false);
public PathPickerField()
{
_pickerPreferenceService = Program.ServiceProvider.GetRequiredService<PickerPreferenceService>();
InitializeComponent();
}
public string Label
{
get => GetValue(LabelProperty);
set => SetValue(LabelProperty, value);
}
public string Text
{
get => GetValue(TextProperty);
set => SetValue(TextProperty, value);
}
public string Watermark
{
get => GetValue(WatermarkProperty);
set => SetValue(WatermarkProperty, value);
}
public string PreferenceKey
{
get => GetValue(PreferenceKeyProperty);
set => SetValue(PreferenceKeyProperty, value);
}
public string PickerTitle
{
get => GetValue(PickerTitleProperty);
set => SetValue(PickerTitleProperty, value);
}
public string FileTypeName
{
get => GetValue(FileTypeNameProperty);
set => SetValue(FileTypeNameProperty, value);
}
public string FilePatterns
{
get => GetValue(FilePatternsProperty);
set => SetValue(FilePatternsProperty, value);
}
public string DefaultExtension
{
get => GetValue(DefaultExtensionProperty);
set => SetValue(DefaultExtensionProperty, value);
}
public PathPickerSelectionMode PickerMode
{
get => GetValue(PickerModeProperty);
set => SetValue(PickerModeProperty, value);
}
public bool IsTextReadOnly
{
get => GetValue(IsTextReadOnlyProperty);
set => SetValue(IsTextReadOnlyProperty, value);
}
public bool ShowPickerButton
{
get => GetValue(ShowPickerButtonProperty);
set => SetValue(ShowPickerButtonProperty, value);
}
public bool AppendDirectorySeparator
{
get => GetValue(AppendDirectorySeparatorProperty);
set => SetValue(AppendDirectorySeparatorProperty, value);
}
private async void PickPath_Click(object? sender, Avalonia.Interactivity.RoutedEventArgs e)
{
var topLevel = TopLevel.GetTopLevel(this);
var storageProvider = topLevel?.StorageProvider;
if (storageProvider is null)
{
return;
}
var selectedPath = await PickPathAsync(storageProvider);
if (string.IsNullOrWhiteSpace(selectedPath))
{
return;
}
Text = selectedPath;
if (!string.IsNullOrWhiteSpace(PreferenceKey))
{
_pickerPreferenceService.RememberPath(PreferenceKey, selectedPath);
}
}
private void OpenPath_Click(object? sender, Avalonia.Interactivity.RoutedEventArgs e)
{
PathShellService.OpenInExplorer(Text);
}
private async Task<string?> PickPathAsync(IStorageProvider storageProvider)
{
var suggestedStartLocation = await TryGetSuggestedStartLocationAsync(storageProvider);
var pickerTitle = ResolvePickerTitle();
switch (PickerMode)
{
case PathPickerSelectionMode.Folder:
{
var folders = await storageProvider.OpenFolderPickerAsync(new FolderPickerOpenOptions
{
Title = pickerTitle,
SuggestedStartLocation = suggestedStartLocation
});
return folders.Count == 0
? null
: NormalizeSelectedPath(folders[0].Path.LocalPath);
}
case PathPickerSelectionMode.OpenFile:
{
var files = await storageProvider.OpenFilePickerAsync(new FilePickerOpenOptions
{
Title = pickerTitle,
SuggestedStartLocation = suggestedStartLocation,
FileTypeFilter = BuildFileTypes()
});
return files.Count == 0
? null
: files[0].Path.LocalPath;
}
case PathPickerSelectionMode.SaveFile:
{
var file = await storageProvider.SaveFilePickerAsync(new FilePickerSaveOptions
{
Title = pickerTitle,
SuggestedStartLocation = suggestedStartLocation,
DefaultExtension = NormalizeDefaultExtension(DefaultExtension),
FileTypeChoices = BuildFileTypes()
});
return file?.Path.LocalPath;
}
default:
return null;
}
}
private async Task<IStorageFolder?> TryGetSuggestedStartLocationAsync(IStorageProvider storageProvider)
{
if (string.IsNullOrWhiteSpace(PreferenceKey))
{
return null;
}
return await _pickerPreferenceService.TryGetStartFolderAsync(storageProvider, PreferenceKey, Text);
}
private IReadOnlyList<FilePickerFileType> BuildFileTypes()
{
var patterns = (FilePatterns ?? string.Empty)
.Split(';', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);
if (patterns.Length == 0)
{
return Array.Empty<FilePickerFileType>();
}
var fileTypeName = string.IsNullOrWhiteSpace(FileTypeName)
? "File"
: FileTypeName.Trim();
return
[
new FilePickerFileType(fileTypeName)
{
Patterns = patterns.ToList()
}
];
}
private string ResolvePickerTitle()
{
if (!string.IsNullOrWhiteSpace(PickerTitle))
{
return PickerTitle;
}
var cleanedLabel = string.IsNullOrWhiteSpace(Label)
? "percorso"
: Label.Trim().TrimEnd(':');
return PickerMode == PathPickerSelectionMode.SaveFile
? $"Salva {cleanedLabel.ToLowerInvariant()}"
: $"Seleziona {cleanedLabel.ToLowerInvariant()}";
}
private string NormalizeSelectedPath(string selectedPath)
{
if (PickerMode != PathPickerSelectionMode.Folder || !AppendDirectorySeparator)
{
return selectedPath;
}
if (string.IsNullOrWhiteSpace(selectedPath))
{
return string.Empty;
}
return selectedPath.EndsWith(Path.DirectorySeparatorChar)
|| selectedPath.EndsWith(Path.AltDirectorySeparatorChar)
? selectedPath
: selectedPath + Path.DirectorySeparatorChar;
}
private static string NormalizeDefaultExtension(string extension)
{
if (string.IsNullOrWhiteSpace(extension))
{
return string.Empty;
}
return extension.Trim().TrimStart('.');
}
}

View file

@ -0,0 +1,8 @@
namespace ImageCatalog_2.Controls;
public enum PathPickerSelectionMode
{
Folder,
OpenFile,
SaveFile
}

View file

@ -0,0 +1,31 @@
using System;
using System.Globalization;
using Avalonia.Data.Converters;
using Avalonia.Media.Imaging;
namespace ImageCatalog_2.Converters;
public sealed class FilePathToBitmapConverter : IValueConverter
{
public object? Convert(object? value, Type targetType, object? parameter, CultureInfo culture)
{
if (value is not string path || string.IsNullOrWhiteSpace(path))
{
return null;
}
try
{
return new Bitmap(path);
}
catch
{
return null;
}
}
public object? ConvertBack(object? value, Type targetType, object? parameter, CultureInfo culture)
{
throw new NotSupportedException();
}
}

File diff suppressed because it is too large Load diff

View file

@ -3,6 +3,7 @@
<Nullable>enable</Nullable>
<ImplicitUsings>enable</ImplicitUsings>
<ProduceReferenceAssembly>False</ProduceReferenceAssembly>
<AvaloniaUseCompiledBindingsByDefault>false</AvaloniaUseCompiledBindingsByDefault>
<!-- Default assembly name for regular builds -->
<AssemblyName>ImageCatalog</AssemblyName>
<LangVersion>default</LangVersion>
@ -46,12 +47,16 @@
<DebugType>embedded</DebugType>
</PropertyGroup>
<ItemGroup>
<AvaloniaXaml Remove="AvaloniaApp.axaml" />
<AvaloniaXaml Remove="Sorgenti\**" />
<Compile Remove="Sorgenti\**" />
<EmbeddedResource Remove="Sorgenti\**" />
<None Remove="Sorgenti\**" />
<Page Remove="Sorgenti\**" />
</ItemGroup>
<ItemGroup>
<AvaloniaResource Include="AvaloniaApp.axaml" />
</ItemGroup>
<ItemGroup>
<None Update="Properties\Settings.settings">
<Generator>SettingsSingleFileGenerator</Generator>
@ -65,16 +70,16 @@
</ItemGroup>
<ItemGroup>
<PackageReference Include="AIFotoONLUS.Core" Version="0.1.2" Condition="'$(UseLocalAIFotoONLUS)' != 'true'" />
<PackageReference Include="AutoMapper" Version="16.1.0" />
<PackageReference Include="IconPacks.Avalonia" Version="1.3.1" />
<PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="10.0.5" />
<PackageReference Include="Microsoft.Extensions.Logging" Version="10.0.5" />
<PackageReference Include="Microsoft.Extensions.Logging.Console" Version="10.0.5" />
<PackageReference Include="AutoMapper" Version="16.1.1" />
<PackageReference Include="IconPacks.Avalonia" Version="2.0.0" />
<PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="10.0.8" />
<PackageReference Include="Microsoft.Extensions.Logging" Version="10.0.8" />
<PackageReference Include="Microsoft.Extensions.Logging.Console" Version="10.0.8" />
<PackageReference Include="MinVer" Version="7.0.0" PrivateAssets="all" />
<PackageReference Include="Avalonia" Version="11.3.12" />
<PackageReference Include="Avalonia.Desktop" Version="11.3.12" />
<PackageReference Include="Avalonia.Themes.Fluent" Version="11.3.12" />
<PackageReference Include="Avalonia.Controls.DataGrid" Version="11.3.12" />
<PackageReference Include="Avalonia" Version="11.3.13" />
<PackageReference Include="Avalonia.Desktop" Version="11.3.13" />
<PackageReference Include="Avalonia.Themes.Fluent" Version="11.3.13" />
<PackageReference Include="Avalonia.Controls.DataGrid" Version="11.3.13" />
<PackageReference Include="Microsoft.DotNet.UpgradeAssistant.Extensions.Default.Analyzers" Version="0.4.421302">
<PrivateAssets>all</PrivateAssets>
</PackageReference>
@ -159,4 +164,13 @@
<Copy SourceFiles="@(LocalCudaInferenceLibrary)" DestinationFolder="$(TargetDir)" SkipUnchangedFiles="true" Condition="'@(LocalCudaInferenceLibrary)' != ''" />
</Target>
<Target Name="CopyLocalCudaInferenceLibrariesToPublish" AfterTargets="Publish" Condition="$([MSBuild]::IsOsPlatform('Windows')) and '$(UseLocalAIFotoONLUS)' == 'true' and '$(PublishDir)' != ''">
<ItemGroup>
<LocalCudaInferenceLibraryForPublish Include="$(LocalAIFotoOutputDir)\cublasLt64_11.dll" Condition="Exists('$(LocalAIFotoOutputDir)\cublasLt64_11.dll')" />
<LocalCudaInferenceLibraryForPublish Include="$(LocalAIFotoOutputDir)\cudnn_cnn_infer64_8.dll" Condition="Exists('$(LocalAIFotoOutputDir)\cudnn_cnn_infer64_8.dll')" />
</ItemGroup>
<Copy SourceFiles="@(LocalCudaInferenceLibraryForPublish)" DestinationFolder="$(PublishDir)" SkipUnchangedFiles="true" Condition="'@(LocalCudaInferenceLibraryForPublish)' != ''" />
</Target>
</Project>

View file

@ -0,0 +1,24 @@
namespace ImageCatalog_2.Models;
public sealed class FaceMatcherResultItem
{
public string PhotoId { get; init; } = string.Empty;
public double? Score { get; init; }
public string ScoreDisplay => Score.HasValue ? Score.Value.ToString("0.###") : string.Empty;
public string ResolvedImagePath { get; init; } = string.Empty;
public int CandidateCount { get; init; }
public string RawRow { get; init; } = string.Empty;
public string DebugSummary { get; init; } = string.Empty;
public string SearchImagePath { get; init; } = string.Empty;
public string CsvPath { get; init; } = string.Empty;
public string LogPath { get; init; } = string.Empty;
}

View file

@ -314,6 +314,18 @@ namespace ImageCatalog_2.Models
[XmlElement("AI_FaceUpsample")]
public bool FaceUpsample { get; set; } = true;
[JsonPropertyName("UseFaceGpu")]
[XmlElement("AI_UsaGpuFace")]
public bool UseFaceGpu { get; set; }
[JsonPropertyName("FaceMatcherExecutablePath")]
[XmlElement("AI_FaceMatcherExecutablePath")]
public string FaceMatcherExecutablePath { get; set; } = string.Empty;
[JsonPropertyName("FaceMatcherTolerance")]
[XmlElement("AI_FaceMatcherTolerance")]
public double FaceMatcherTolerance { get; set; } = 0.5;
// Race upload settings
[JsonPropertyName("ApiLogin")]
[XmlElement("RaceUpload_Login")]

View file

@ -154,6 +154,7 @@ static class Program
var userPrefsPath = System.IO.Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData),
"ImageCatalog", "userprefs.xml");
services.AddSingleton(new ParametriSetup(userPrefsPath));
services.AddSingleton<PickerPreferenceService>();
services.AddSingleton<PicSettings>();
services.AddCatalogCommunication(options =>

View file

@ -198,20 +198,7 @@ public class AiExtractionService : IAiExtractionService
{
try
{
var dir = Path.GetDirectoryName(request.CsvOutputPath) ?? string.Empty;
if (!string.IsNullOrWhiteSpace(dir) && !Directory.Exists(dir))
{
Directory.CreateDirectory(dir);
}
using var sw = new StreamWriter(request.CsvOutputPath, false, Encoding.UTF8);
sw.WriteLine("Path,Text");
foreach (var r in extractedResults)
{
var csvFileName = Path.GetFileName(r.Path ?? string.Empty);
var safeText = (r.Text ?? string.Empty).Replace("\"", "\"\"");
sw.WriteLine($"\"{csvFileName}\",\"{safeText}\"");
}
WriteCsvOutput(request.CsvOutputPath, extractedResults);
}
catch (Exception ex)
{
@ -222,6 +209,24 @@ public class AiExtractionService : IAiExtractionService
return summary;
}
internal static void WriteCsvOutput(string csvOutputPath, IEnumerable<AiResultItem> extractedResults)
{
var dir = Path.GetDirectoryName(csvOutputPath) ?? string.Empty;
if (!string.IsNullOrWhiteSpace(dir) && !Directory.Exists(dir))
{
Directory.CreateDirectory(dir);
}
using var sw = new StreamWriter(csvOutputPath, false, Encoding.UTF8);
sw.WriteLine("filename,text");
foreach (var result in extractedResults)
{
var csvFileName = Path.GetFileName(result.Path ?? string.Empty);
var safeText = (result.Text ?? string.Empty).Replace("\"", "\"\"");
sw.WriteLine($"\"{csvFileName}\",\"{safeText}\"");
}
}
private static double CalculateAverageImagesPerSecond(int processed, TimeSpan elapsed)
{
return elapsed.TotalSeconds > 0 ? processed / elapsed.TotalSeconds : 0;

View file

@ -0,0 +1,50 @@
using System.Diagnostics;
using System.IO;
namespace ImageCatalog_2.Services;
public static class PathShellService
{
public static void OpenInExplorer(string? path)
{
if (string.IsNullOrWhiteSpace(path))
{
return;
}
var normalizedPath = path.Trim().Trim('"');
try
{
if (File.Exists(normalizedPath))
{
Process.Start("explorer.exe", $"/select,\"{normalizedPath}\"");
return;
}
if (Directory.Exists(normalizedPath))
{
Process.Start(new ProcessStartInfo
{
FileName = normalizedPath,
UseShellExecute = true
});
return;
}
var containingDirectory = Path.GetDirectoryName(normalizedPath);
if (!string.IsNullOrWhiteSpace(containingDirectory) && Directory.Exists(containingDirectory))
{
Process.Start(new ProcessStartInfo
{
FileName = containingDirectory,
UseShellExecute = true
});
}
}
catch
{
// Ignore failures when opening Explorer.
}
}
}

View file

@ -0,0 +1,122 @@
using Avalonia.Platform.Storage;
using ImageCatalog;
using System;
using System.IO;
using System.Threading.Tasks;
namespace ImageCatalog_2.Services;
public static class PickerPreferenceKeys
{
public const string SourceFolder = "Picker.SourceFolder.LastPath";
public const string DestinationFolder = "Picker.DestinationFolder.LastPath";
public const string LogoFile = "Picker.LogoFile.LastPath";
public const string ModelsFolder = "Picker.ModelsFolder.LastPath";
public const string CsvOutput = "Picker.CsvOutput.LastPath";
public const string SettingsFile = "Picker.SettingsFile.LastPath";
public const string LastSettingsFile = "Settings.LastFilePath";
public const string FaceExecutableFolder = "Picker.FaceExecutableFolder.LastPath";
public const string FaceOutputFolder = "Picker.FaceOutputFolder.LastPath";
public const string FaceMatcherExecutable = "Picker.FaceMatcherExecutable.LastPath";
public const string FaceMatcherImage = "Picker.FaceMatcherImage.LastPath";
public const string FaceMatcherEncodings = "Picker.FaceMatcherEncodings.LastPath";
public const string FaceMatcherOutput = "Picker.FaceMatcherOutput.LastPath";
public const string FaceMatcherLog = "Picker.FaceMatcherLog.LastPath";
}
public sealed class PickerPreferenceService
{
private readonly ParametriSetup _userPreferences;
public PickerPreferenceService(ParametriSetup userPreferences)
{
_userPreferences = userPreferences;
}
public async Task<IStorageFolder?> TryGetStartFolderAsync(IStorageProvider storageProvider, string preferenceKey, string? currentPath = null)
{
var startPath = GetPreferredStartDirectory(preferenceKey, currentPath);
if (string.IsNullOrWhiteSpace(startPath))
{
return null;
}
try
{
return await storageProvider.TryGetFolderFromPathAsync(new Uri(startPath)).ConfigureAwait(true);
}
catch
{
return null;
}
}
public void RememberPath(string preferenceKey, string? selectedPath)
{
var directory = TryGetExistingDirectory(selectedPath);
if (string.IsNullOrWhiteSpace(directory))
{
return;
}
_userPreferences.AggiornaParametro(preferenceKey, directory);
_userPreferences.SalvaParametriSetup();
}
public string? GetRememberedValue(string preferenceKey)
{
var value = _userPreferences.LeggiParametroString(preferenceKey);
return string.IsNullOrWhiteSpace(value)
? null
: value.Trim().Trim('"');
}
public void RememberValue(string preferenceKey, string? value)
{
if (string.IsNullOrWhiteSpace(value))
{
return;
}
_userPreferences.AggiornaParametro(preferenceKey, value.Trim().Trim('"'));
_userPreferences.SalvaParametriSetup();
}
public void ForgetValue(string preferenceKey)
{
if (_userPreferences.RimuoviParametro(preferenceKey))
{
_userPreferences.SalvaParametriSetup();
}
}
private string? GetPreferredStartDirectory(string preferenceKey, string? currentPath)
{
var storedPath = _userPreferences.LeggiParametroString(preferenceKey);
return TryGetExistingDirectory(storedPath) ?? TryGetExistingDirectory(currentPath);
}
private static string? TryGetExistingDirectory(string? path)
{
if (string.IsNullOrWhiteSpace(path))
{
return null;
}
var normalizedPath = path.Trim().Trim('"');
if (Directory.Exists(normalizedPath))
{
return normalizedPath;
}
if (File.Exists(normalizedPath))
{
return Path.GetDirectoryName(normalizedPath);
}
var containingDirectory = Path.GetDirectoryName(normalizedPath);
return !string.IsNullOrWhiteSpace(containingDirectory) && Directory.Exists(containingDirectory)
? containingDirectory
: null;
}
}

View file

@ -225,6 +225,105 @@ public class AiSettingsViewModel : ViewModelBase
}
}
private string _faceMatcherExecutablePath = string.Empty;
public string FaceMatcherExecutablePath
{
get => _faceMatcherExecutablePath;
set
{
_faceMatcherExecutablePath = value;
NotifyPropertyChanged();
}
}
private string _faceMatcherEncodingsPath = string.Empty;
public string FaceMatcherEncodingsPath
{
get => _faceMatcherEncodingsPath;
set
{
_faceMatcherEncodingsPath = value;
NotifyPropertyChanged();
}
}
private string _faceMatcherOutputPath = string.Empty;
public string FaceMatcherOutputPath
{
get => _faceMatcherOutputPath;
set
{
_faceMatcherOutputPath = value;
NotifyPropertyChanged();
}
}
private string _faceMatcherLogPath = string.Empty;
public string FaceMatcherLogPath
{
get => _faceMatcherLogPath;
set
{
_faceMatcherLogPath = value;
NotifyPropertyChanged();
}
}
private double _faceMatcherTolerance = 0.5;
public double FaceMatcherTolerance
{
get => _faceMatcherTolerance;
set
{
_faceMatcherTolerance = value;
NotifyPropertyChanged();
}
}
private string _faceMatcherSelectedImagePath = string.Empty;
public string FaceMatcherSelectedImagePath
{
get => _faceMatcherSelectedImagePath;
set
{
_faceMatcherSelectedImagePath = value;
NotifyPropertyChanged();
}
}
private bool _isFaceMatcherRunning;
public bool IsFaceMatcherRunning
{
get => _isFaceMatcherRunning;
set
{
_isFaceMatcherRunning = value;
NotifyPropertyChanged();
}
}
private string _faceMatcherStatusMessage = string.Empty;
public string FaceMatcherStatusMessage
{
get => _faceMatcherStatusMessage;
set
{
_faceMatcherStatusMessage = value;
NotifyPropertyChanged();
}
}
private string _faceMatcherCommandOutput = string.Empty;
public string FaceMatcherCommandOutput
{
get => _faceMatcherCommandOutput;
set
{
_faceMatcherCommandOutput = value;
NotifyPropertyChanged();
}
}
private double _aiProgress;
public double AiProgress
{
@ -237,4 +336,6 @@ public class AiSettingsViewModel : ViewModelBase
}
public ObservableCollection<AiResultItem> PreviewResults { get; } = new();
public ObservableCollection<FaceMatcherResultItem> FaceMatcherResults { get; } = new();
}