mirror of
https://github.com/aliasvault/aliasvault.git
synced 2026-02-02 02:13:48 -05:00
271 lines
9.0 KiB
Plaintext
271 lines
9.0 KiB
Plaintext
@using AliasVault.Client.Main.Utilities
|
|
@using AliasClientDb.Models
|
|
@using Microsoft.Extensions.Localization
|
|
@inject ItemService ItemService
|
|
|
|
@* FieldBlock component - renders a single field based on its type *@
|
|
@switch (Field.FieldType)
|
|
{
|
|
case FieldType.Password:
|
|
case FieldType.Hidden:
|
|
<div class="col-span-6">
|
|
@RenderLabelWithHistory()
|
|
<CopyPastePasswordFormRow
|
|
Id="@GetFieldId()"
|
|
Label=""
|
|
Value="@(Field.Value ?? string.Empty)" />
|
|
</div>
|
|
break;
|
|
case FieldType.TextArea:
|
|
<div class="col-span-6">
|
|
@if (!HideLabel)
|
|
{
|
|
@RenderLabelWithHistory()
|
|
}
|
|
<div class="p-3 bg-gray-50 dark:bg-gray-700 rounded-lg text-gray-900 dark:text-white whitespace-pre-wrap text-sm">
|
|
@RenderTextWithLinks(Field.Value ?? string.Empty)
|
|
</div>
|
|
</div>
|
|
break;
|
|
case FieldType.URL:
|
|
<div class="col-span-6">
|
|
@RenderLabelWithHistory()
|
|
@if (!string.IsNullOrEmpty(Field.Value))
|
|
{
|
|
@if (Field.Value.StartsWith("http://", StringComparison.OrdinalIgnoreCase) || Field.Value.StartsWith("https://", StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
<a href="@Field.Value" target="_blank" rel="noopener noreferrer" class="text-blue-600 dark:text-blue-400 hover:underline break-all">
|
|
@Field.Value
|
|
</a>
|
|
}
|
|
else
|
|
{
|
|
<span class="text-gray-700 dark:text-gray-300 break-all">@Field.Value</span>
|
|
}
|
|
}
|
|
</div>
|
|
break;
|
|
case FieldType.Date:
|
|
<div class="@(FullWidth ? "col-span-6" : "col-span-6 sm:col-span-3")">
|
|
@RenderLabelWithHistory()
|
|
<CopyPasteFormRow
|
|
Id="@GetFieldId()"
|
|
Label=""
|
|
Value="@(Field.Value ?? string.Empty)" />
|
|
</div>
|
|
break;
|
|
default:
|
|
@* Text, Email, Phone, Number - use standard copy/paste row *@
|
|
<div class="@(FullWidth ? "col-span-6" : "col-span-6 sm:col-span-3")">
|
|
@RenderLabelWithHistory()
|
|
<CopyPasteFormRow
|
|
Id="@GetFieldId()"
|
|
Label=""
|
|
Value="@(Field.Value ?? string.Empty)" />
|
|
</div>
|
|
break;
|
|
}
|
|
|
|
@* Field History Modal *@
|
|
@if (ShowHistoryModal)
|
|
{
|
|
<FieldHistoryModal
|
|
ItemId="@ItemId"
|
|
FieldKey="@Field.FieldKey"
|
|
FieldLabel="@GetLabel()"
|
|
IsHidden="@(Field.FieldType == FieldType.Password || Field.FieldType == FieldType.Hidden)"
|
|
OnClose="CloseHistoryModal" />
|
|
}
|
|
|
|
@code {
|
|
[Inject]
|
|
private IStringLocalizerFactory LocalizerFactory { get; set; } = default!;
|
|
|
|
private IStringLocalizer Localizer => LocalizerFactory.Create("Components.Fields.FieldBlock", "AliasVault.Client");
|
|
|
|
/// <summary>
|
|
/// Gets or sets the display field to render.
|
|
/// </summary>
|
|
[Parameter]
|
|
public required DisplayField Field { get; set; }
|
|
|
|
/// <summary>
|
|
/// Gets or sets the item ID (required for history lookup).
|
|
/// </summary>
|
|
[Parameter]
|
|
public Guid ItemId { get; set; }
|
|
|
|
/// <summary>
|
|
/// Gets or sets whether the field should span full width.
|
|
/// </summary>
|
|
[Parameter]
|
|
public bool FullWidth { get; set; }
|
|
|
|
/// <summary>
|
|
/// Gets or sets whether to hide the label (useful when section header already shows the label).
|
|
/// </summary>
|
|
[Parameter]
|
|
public bool HideLabel { get; set; }
|
|
|
|
private bool ShowHistoryModal { get; set; }
|
|
private int HistoryCount { get; set; }
|
|
|
|
/// <inheritdoc />
|
|
protected override async Task OnParametersSetAsync()
|
|
{
|
|
await base.OnParametersSetAsync();
|
|
|
|
// Check history count if field has history enabled and item ID is provided
|
|
// Pass current value for smart comparison (only show icon if history differs from current)
|
|
if (Field.EnableHistory && ItemId != Guid.Empty && !string.IsNullOrEmpty(Field.FieldKey))
|
|
{
|
|
HistoryCount = await ItemService.GetFieldHistoryCountAsync(ItemId, Field.FieldKey, Field.Value);
|
|
}
|
|
else
|
|
{
|
|
HistoryCount = 0;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Renders the label with optional history icon.
|
|
/// </summary>
|
|
private RenderFragment RenderLabelWithHistory() => builder =>
|
|
{
|
|
builder.OpenElement(0, "label");
|
|
builder.AddAttribute(1, "for", GetFieldId());
|
|
builder.AddAttribute(2, "class", "block mb-2 text-sm font-medium text-gray-900 dark:text-white");
|
|
|
|
// Label text
|
|
builder.AddContent(3, GetLabel());
|
|
|
|
// History icon (only if history is enabled and there's history data)
|
|
if (Field.EnableHistory && HistoryCount > 0)
|
|
{
|
|
builder.OpenElement(4, "button");
|
|
builder.AddAttribute(5, "type", "button");
|
|
builder.AddAttribute(6, "class", "ml-2 inline-flex items-center text-gray-500 hover:text-gray-700 dark:text-gray-400 dark:hover:text-gray-200 focus:outline-none");
|
|
builder.AddAttribute(7, "title", Localizer["ViewHistory"]);
|
|
builder.AddAttribute(8, "onclick", EventCallback.Factory.Create(this, OpenHistoryModal));
|
|
|
|
// Clock/history icon SVG
|
|
builder.OpenElement(9, "svg");
|
|
builder.AddAttribute(10, "class", "h-4 w-4");
|
|
builder.AddAttribute(11, "fill", "none");
|
|
builder.AddAttribute(12, "viewBox", "0 0 24 24");
|
|
builder.AddAttribute(13, "stroke-width", "1.5");
|
|
builder.AddAttribute(14, "stroke", "currentColor");
|
|
builder.OpenElement(15, "path");
|
|
builder.AddAttribute(16, "stroke-linecap", "round");
|
|
builder.AddAttribute(17, "stroke-linejoin", "round");
|
|
builder.AddAttribute(18, "d", "M12 6v6h4.5m4.5 0a9 9 0 1 1-18 0 9 9 0 0 1 18 0Z");
|
|
builder.CloseElement(); // path
|
|
builder.CloseElement(); // svg
|
|
|
|
builder.CloseElement(); // button
|
|
}
|
|
|
|
builder.CloseElement(); // label
|
|
};
|
|
|
|
/// <summary>
|
|
/// Opens the history modal.
|
|
/// </summary>
|
|
private void OpenHistoryModal()
|
|
{
|
|
ShowHistoryModal = true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Closes the history modal.
|
|
/// </summary>
|
|
private void CloseHistoryModal()
|
|
{
|
|
ShowHistoryModal = false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the label for the field.
|
|
/// </summary>
|
|
private string GetLabel()
|
|
{
|
|
// For custom fields, use the Label property directly
|
|
if (Field.IsCustomField)
|
|
{
|
|
return !string.IsNullOrEmpty(Field.Label) ? Field.Label : FormatFieldName(Field.FieldKey);
|
|
}
|
|
|
|
// Try to get localized label for system field
|
|
if (!string.IsNullOrEmpty(Field.FieldKey))
|
|
{
|
|
// Convert field key to localization key format
|
|
// e.g., "login.username" -> "FieldLabel_login_username"
|
|
var localizationKey = "FieldLabel_" + Field.FieldKey.Replace(".", "_");
|
|
var localizedLabel = Localizer[localizationKey];
|
|
|
|
// If the localized string is different from the key, return it
|
|
if (localizedLabel.ResourceNotFound == false)
|
|
{
|
|
return localizedLabel.Value;
|
|
}
|
|
|
|
// Fallback: convert field key to readable format
|
|
// e.g., "login.username" -> "Username"
|
|
var parts = Field.FieldKey.Split('.');
|
|
if (parts.Length > 1)
|
|
{
|
|
return FormatFieldName(parts[1]);
|
|
}
|
|
}
|
|
|
|
return FormatFieldName(Field.FieldKey);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Formats a field name to be human-readable.
|
|
/// </summary>
|
|
private static string FormatFieldName(string fieldName)
|
|
{
|
|
if (string.IsNullOrEmpty(fieldName))
|
|
{
|
|
return string.Empty;
|
|
}
|
|
|
|
// Convert snake_case to Title Case
|
|
var words = fieldName.Split('_');
|
|
return string.Join(" ", words.Select(w =>
|
|
string.IsNullOrEmpty(w) ? string.Empty :
|
|
char.ToUpperInvariant(w[0]) + w[1..]));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a unique ID for the field.
|
|
/// </summary>
|
|
private string GetFieldId()
|
|
{
|
|
return !string.IsNullOrEmpty(Field.FieldKey)
|
|
? Field.FieldKey.Replace(".", "-")
|
|
: Field.FieldDefinitionId ?? Guid.NewGuid().ToString();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Renders text with clickable links.
|
|
/// </summary>
|
|
private MarkupString RenderTextWithLinks(string text)
|
|
{
|
|
if (string.IsNullOrEmpty(text))
|
|
{
|
|
return new MarkupString(string.Empty);
|
|
}
|
|
|
|
// Simple URL regex pattern
|
|
var urlPattern = @"(https?://[^\s<>""]+)";
|
|
var result = System.Text.RegularExpressions.Regex.Replace(
|
|
System.Web.HttpUtility.HtmlEncode(text),
|
|
urlPattern,
|
|
"<a href=\"$1\" target=\"_blank\" rel=\"noopener noreferrer\" class=\"text-blue-600 dark:text-blue-400 hover:underline\">$1</a>");
|
|
|
|
return new MarkupString(result);
|
|
}
|
|
}
|