mirror of
https://github.com/MindWorkAI/AI-Studio.git
synced 2026-02-12 11:01:38 +00:00
Some checks failed
Build and Release / Read metadata (push) Has been cancelled
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-apple-darwin, osx-arm64, macos-latest, aarch64-apple-darwin, dmg updater) (push) Has been cancelled
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-pc-windows-msvc.exe, win-arm64, windows-latest, aarch64-pc-windows-msvc, nsis updater) (push) Has been cancelled
Build and Release / Publish release (push) Has been cancelled
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-unknown-linux-gnu, linux-arm64, ubuntu-22.04-arm, aarch64-unknown-linux-gnu, appimage deb updater) (push) Has been cancelled
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-apple-darwin, osx-x64, macos-latest, x86_64-apple-darwin, dmg updater) (push) Has been cancelled
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-pc-windows-msvc.exe, win-x64, windows-latest, x86_64-pc-windows-msvc, nsis updater) (push) Has been cancelled
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-unknown-linux-gnu, linux-x64, ubuntu-22.04, x86_64-unknown-linux-gnu, appimage deb updater) (push) Has been cancelled
Build and Release / Prepare & create release (push) Has been cancelled
295 lines
11 KiB
C#
295 lines
11 KiB
C#
using AIStudio.Chat;
|
|
using AIStudio.Dialogs;
|
|
using AIStudio.Tools.PluginSystem;
|
|
using AIStudio.Tools.Rust;
|
|
using AIStudio.Tools.Services;
|
|
using AIStudio.Tools.Validation;
|
|
|
|
using Microsoft.AspNetCore.Components;
|
|
|
|
namespace AIStudio.Components;
|
|
|
|
using DialogOptions = Dialogs.DialogOptions;
|
|
|
|
public partial class AttachDocuments : MSGComponentBase
|
|
{
|
|
private static string TB(string fallbackEN) => I18N.I.T(fallbackEN, typeof(AttachDocuments).Namespace, nameof(AttachDocuments));
|
|
|
|
[Parameter]
|
|
public string Name { get; set; } = string.Empty;
|
|
|
|
/// <summary>
|
|
/// On which layer to register the drop area. Higher layers have priority over lower layers.
|
|
/// </summary>
|
|
[Parameter]
|
|
public int Layer { get; set; }
|
|
|
|
/// <summary>
|
|
/// When true, pause catching dropped files. Default is false.
|
|
/// </summary>
|
|
[Parameter]
|
|
public bool PauseCatchingDrops { get; set; }
|
|
|
|
[Parameter]
|
|
public HashSet<FileAttachment> DocumentPaths { get; set; } = [];
|
|
|
|
[Parameter]
|
|
public EventCallback<HashSet<FileAttachment>> DocumentPathsChanged { get; set; }
|
|
|
|
[Parameter]
|
|
public Func<HashSet<FileAttachment>, Task> OnChange { get; set; } = _ => Task.CompletedTask;
|
|
|
|
/// <summary>
|
|
/// Catch all documents that are hovered over the AI Studio window and not only over the drop zone.
|
|
/// </summary>
|
|
[Parameter]
|
|
public bool CatchAllDocuments { get; set; }
|
|
|
|
[Parameter]
|
|
public bool UseSmallForm { get; set; }
|
|
|
|
/// <summary>
|
|
/// When true, validate media file types before attaching. Default is true. That means that
|
|
/// the user cannot attach unsupported media file types when the provider or model does not
|
|
/// support them. Set it to false in order to disable this validation. This is useful for places
|
|
/// where the user might want to prepare a template.
|
|
/// </summary>
|
|
[Parameter]
|
|
public bool ValidateMediaFileTypes { get; set; } = true;
|
|
|
|
[Parameter]
|
|
public AIStudio.Settings.Provider? Provider { get; set; }
|
|
|
|
[Inject]
|
|
private ILogger<AttachDocuments> Logger { get; set; } = null!;
|
|
|
|
[Inject]
|
|
private RustService RustService { get; init; } = null!;
|
|
|
|
[Inject]
|
|
private IDialogService DialogService { get; init; } = null!;
|
|
|
|
[Inject]
|
|
private PandocAvailabilityService PandocAvailabilityService { get; init; } = null!;
|
|
|
|
private const Placement TOOLBAR_TOOLTIP_PLACEMENT = Placement.Top;
|
|
private static readonly string DROP_FILES_HERE_TEXT = TB("Drop files here to attach them.");
|
|
|
|
private uint numDropAreasAboveThis;
|
|
private bool isComponentHovered;
|
|
private bool isDraggingOver;
|
|
|
|
#region Overrides of MSGComponentBase
|
|
|
|
protected override async Task OnInitializedAsync()
|
|
{
|
|
this.ApplyFilters([], [ Event.TAURI_EVENT_RECEIVED, Event.REGISTER_FILE_DROP_AREA, Event.UNREGISTER_FILE_DROP_AREA ]);
|
|
|
|
// Register this drop area:
|
|
await this.MessageBus.SendMessage(this, Event.REGISTER_FILE_DROP_AREA, this.Layer);
|
|
await base.OnInitializedAsync();
|
|
}
|
|
|
|
protected override async Task ProcessIncomingMessage<T>(ComponentBase? sendingComponent, Event triggeredEvent, T? data) where T : default
|
|
{
|
|
switch (triggeredEvent)
|
|
{
|
|
case Event.REGISTER_FILE_DROP_AREA when sendingComponent != this:
|
|
{
|
|
if(data is int layer && layer > this.Layer)
|
|
{
|
|
this.numDropAreasAboveThis++;
|
|
this.PauseCatchingDrops = true;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case Event.UNREGISTER_FILE_DROP_AREA when sendingComponent != this:
|
|
{
|
|
if(data is int layer && layer > this.Layer)
|
|
{
|
|
if(this.numDropAreasAboveThis > 0)
|
|
this.numDropAreasAboveThis--;
|
|
|
|
if(this.numDropAreasAboveThis is 0)
|
|
this.PauseCatchingDrops = false;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case Event.TAURI_EVENT_RECEIVED when data is TauriEvent { EventType: TauriEventType.FILE_DROP_HOVERED }:
|
|
if(this.PauseCatchingDrops)
|
|
return;
|
|
|
|
if(!this.isComponentHovered && !this.CatchAllDocuments)
|
|
{
|
|
this.Logger.LogDebug("Attach documents component '{Name}' is not hovered, ignoring file drop hovered event.", this.Name);
|
|
return;
|
|
}
|
|
|
|
this.isDraggingOver = true;
|
|
this.SetDragClass();
|
|
this.StateHasChanged();
|
|
break;
|
|
|
|
case Event.TAURI_EVENT_RECEIVED when data is TauriEvent { EventType: TauriEventType.FILE_DROP_CANCELED }:
|
|
if(this.PauseCatchingDrops)
|
|
return;
|
|
|
|
this.isDraggingOver = false;
|
|
this.StateHasChanged();
|
|
break;
|
|
|
|
case Event.TAURI_EVENT_RECEIVED when data is TauriEvent { EventType: TauriEventType.WINDOW_NOT_FOCUSED }:
|
|
if(this.PauseCatchingDrops)
|
|
return;
|
|
|
|
this.isDraggingOver = false;
|
|
this.isComponentHovered = false;
|
|
this.ClearDragClass();
|
|
this.StateHasChanged();
|
|
break;
|
|
|
|
case Event.TAURI_EVENT_RECEIVED when data is TauriEvent { EventType: TauriEventType.FILE_DROP_DROPPED, Payload: var paths }:
|
|
if(this.PauseCatchingDrops)
|
|
return;
|
|
|
|
if(!this.isComponentHovered && !this.CatchAllDocuments)
|
|
{
|
|
this.Logger.LogDebug("Attach documents component '{Name}' is not hovered, ignoring file drop dropped event.", this.Name);
|
|
return;
|
|
}
|
|
|
|
// Ensure that Pandoc is installed and ready:
|
|
var pandocState = await this.PandocAvailabilityService.EnsureAvailabilityAsync(
|
|
showSuccessMessage: false,
|
|
showDialog: true);
|
|
|
|
// If Pandoc is not available (user cancelled installation), abort file drop:
|
|
if (!pandocState.IsAvailable)
|
|
{
|
|
this.Logger.LogWarning("The user cancelled the Pandoc installation or Pandoc is not available. Aborting file drop.");
|
|
this.isDraggingOver = false;
|
|
this.ClearDragClass();
|
|
this.StateHasChanged();
|
|
return;
|
|
}
|
|
|
|
foreach (var path in paths)
|
|
{
|
|
if(!await FileExtensionValidation.IsExtensionValidWithNotifyAsync(FileExtensionValidation.UseCase.ATTACHING_CONTENT, path, this.ValidateMediaFileTypes, this.Provider))
|
|
continue;
|
|
|
|
this.DocumentPaths.Add(FileAttachment.FromPath(path));
|
|
}
|
|
|
|
await this.DocumentPathsChanged.InvokeAsync(this.DocumentPaths);
|
|
await this.OnChange(this.DocumentPaths);
|
|
this.isDraggingOver = false;
|
|
this.ClearDragClass();
|
|
this.StateHasChanged();
|
|
break;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
private const string DEFAULT_DRAG_CLASS = "relative rounded-lg border-2 border-dashed pa-4 mt-4 mud-width-full mud-height-full";
|
|
|
|
private string dragClass = DEFAULT_DRAG_CLASS;
|
|
|
|
private async Task AddFilesManually()
|
|
{
|
|
// Ensure that Pandoc is installed and ready:
|
|
var pandocState = await this.PandocAvailabilityService.EnsureAvailabilityAsync(
|
|
showSuccessMessage: false,
|
|
showDialog: true);
|
|
|
|
// If Pandoc is not available (user cancelled installation), abort file selection:
|
|
if (!pandocState.IsAvailable)
|
|
{
|
|
this.Logger.LogWarning("The user cancelled the Pandoc installation or Pandoc is not available. Aborting file selection.");
|
|
return;
|
|
}
|
|
|
|
var selectFiles = await this.RustService.SelectFiles(T("Select files to attach"));
|
|
if (selectFiles.UserCancelled)
|
|
return;
|
|
|
|
foreach (var selectedFilePath in selectFiles.SelectedFilePaths)
|
|
{
|
|
if (!File.Exists(selectedFilePath))
|
|
continue;
|
|
|
|
if (!await FileExtensionValidation.IsExtensionValidWithNotifyAsync(FileExtensionValidation.UseCase.ATTACHING_CONTENT, selectedFilePath, this.ValidateMediaFileTypes, this.Provider))
|
|
continue;
|
|
|
|
this.DocumentPaths.Add(FileAttachment.FromPath(selectedFilePath));
|
|
}
|
|
|
|
await this.DocumentPathsChanged.InvokeAsync(this.DocumentPaths);
|
|
await this.OnChange(this.DocumentPaths);
|
|
}
|
|
|
|
private async Task OpenAttachmentsDialog()
|
|
{
|
|
this.DocumentPaths = await ReviewAttachmentsDialog.OpenDialogAsync(this.DialogService, this.DocumentPaths);
|
|
}
|
|
|
|
private async Task ClearAllFiles()
|
|
{
|
|
this.DocumentPaths.Clear();
|
|
await this.DocumentPathsChanged.InvokeAsync(this.DocumentPaths);
|
|
await this.OnChange(this.DocumentPaths);
|
|
}
|
|
|
|
private void SetDragClass() => this.dragClass = $"{DEFAULT_DRAG_CLASS} mud-border-primary border-4";
|
|
|
|
private void ClearDragClass() => this.dragClass = DEFAULT_DRAG_CLASS;
|
|
|
|
private void OnMouseEnter(EventArgs _)
|
|
{
|
|
if(this.PauseCatchingDrops)
|
|
return;
|
|
|
|
this.Logger.LogDebug("Attach documents component '{Name}' is hovered.", this.Name);
|
|
this.isComponentHovered = true;
|
|
this.SetDragClass();
|
|
this.StateHasChanged();
|
|
}
|
|
|
|
private void OnMouseLeave(EventArgs _)
|
|
{
|
|
if(this.PauseCatchingDrops)
|
|
return;
|
|
|
|
this.Logger.LogDebug("Attach documents component '{Name}' is no longer hovered.", this.Name);
|
|
this.isComponentHovered = false;
|
|
this.ClearDragClass();
|
|
this.StateHasChanged();
|
|
}
|
|
|
|
private async Task RemoveDocument(FileAttachment fileAttachment)
|
|
{
|
|
this.DocumentPaths.Remove(fileAttachment);
|
|
|
|
await this.DocumentPathsChanged.InvokeAsync(this.DocumentPaths);
|
|
await this.OnChange(this.DocumentPaths);
|
|
}
|
|
|
|
/// <summary>
|
|
/// The user might want to check what we actually extract from his file and therefore give the LLM as an input.
|
|
/// </summary>
|
|
/// <param name="fileAttachment">The file to check.</param>
|
|
private async Task InvestigateFile(FileAttachment fileAttachment)
|
|
{
|
|
var dialogParameters = new DialogParameters<DocumentCheckDialog>
|
|
{
|
|
{ x => x.Document, fileAttachment },
|
|
};
|
|
|
|
await this.DialogService.ShowAsync<DocumentCheckDialog>(T("Document Preview"), dialogParameters, DialogOptions.FULLSCREEN);
|
|
}
|
|
} |