2026-04-20 16:11:27 +02:00
|
|
|
@page "/work-unit"
|
|
|
|
|
@page "/work-unit/{DateStr}"
|
|
|
|
|
@page "/work-unit/{DateStr}/{UnitId}"
|
2026-03-17 22:10:19 +01:00
|
|
|
@attribute [Authorize]
|
|
|
|
|
@rendermode InteractiveServer
|
|
|
|
|
|
|
|
|
|
@inject IWorkDayService WorkDayService
|
|
|
|
|
@inject IAppSettingsService AppSettingsService
|
|
|
|
|
@inject NavigationManager Navigation
|
2026-04-20 16:11:27 +02:00
|
|
|
@inject IJSRuntime JS
|
2026-03-17 22:10:19 +01:00
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
<PageTitle>Work Unit</PageTitle>
|
2026-03-17 22:10:19 +01:00
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
<h1>Work Unit</h1>
|
2026-03-17 22:10:19 +01:00
|
|
|
|
|
|
|
|
@if (!loaded)
|
|
|
|
|
{
|
|
|
|
|
<p><em>Loading...</em></p>
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
<div class="row g-3">
|
|
|
|
|
<div class="col-12 col-md-6 col-lg-4">
|
|
|
|
|
<label class="form-label">Date</label>
|
2026-04-20 16:11:27 +02:00
|
|
|
<input type="date" class="form-control" value="@selectedDate.ToString("yyyy-MM-dd")" @onchange="OnDateChanged" disabled="@isExistingUnit" />
|
2026-03-17 22:10:19 +01:00
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="col-12 col-md-6 col-lg-4">
|
2026-04-20 16:11:27 +02:00
|
|
|
<label class="form-label">Label</label>
|
|
|
|
|
<input class="form-control" @bind="label" maxlength="40" />
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="col-12 col-md-6 col-lg-4">
|
|
|
|
|
<label class="form-label">Location</label>
|
|
|
|
|
<select class="form-select" @bind="location">
|
|
|
|
|
@foreach (var item in Enum.GetValues<WorkUnitLocation>())
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
<option value="@item">@item</option>
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
</select>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="col-12 col-md-6 col-lg-4">
|
|
|
|
|
<label class="form-label">Start Time</label>
|
|
|
|
|
<input type="time" class="form-control" value="@startTimeStr" @onchange="OnStartTimeChanged" />
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="col-12 col-md-6 col-lg-4">
|
2026-04-20 16:11:27 +02:00
|
|
|
<label class="form-label">End Time</label>
|
|
|
|
|
<input type="time" class="form-control" value="@endTimeStr" @onchange="OnEndTimeChanged" />
|
2026-03-17 22:10:19 +01:00
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="col-12 col-md-6 col-lg-4">
|
2026-04-20 16:11:27 +02:00
|
|
|
<label class="form-label">Counted Hours</label>
|
|
|
|
|
<input type="text" class="form-control" value="@manualWorkedHoursStr" @onchange="OnManualWorkedHoursChanged" placeholder="00:00" inputmode="numeric" />
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="col-12 col-md-6 col-lg-4 d-flex align-items-end">
|
|
|
|
|
<div class="form-check mb-2">
|
|
|
|
|
<input id="preview-checkbox" type="checkbox" class="form-check-input" @bind="isPreview" />
|
|
|
|
|
<label class="form-check-label" for="preview-checkbox">Preview work unit</label>
|
|
|
|
|
</div>
|
2026-03-17 22:10:19 +01:00
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="col-12">
|
|
|
|
|
<label class="form-label">Notes</label>
|
|
|
|
|
<textarea class="form-control" rows="2" @bind="notes"></textarea>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<hr />
|
|
|
|
|
|
|
|
|
|
<h2 class="h5">Computed values</h2>
|
|
|
|
|
<div class="row g-3">
|
|
|
|
|
<div class="col-6 col-md-4 col-lg-3">
|
2026-04-20 16:11:27 +02:00
|
|
|
<label class="form-label text-muted">Calculated Hours</label>
|
|
|
|
|
<div class="form-control-plaintext fw-bold">@FormatHours(calculatedWorkedHours)</div>
|
2026-03-17 22:10:19 +01:00
|
|
|
</div>
|
|
|
|
|
<div class="col-6 col-md-4 col-lg-3">
|
2026-04-20 16:11:27 +02:00
|
|
|
<label class="form-label text-muted">Difference</label>
|
|
|
|
|
<div class="form-control-plaintext fw-bold">@FormatSignedHours(workedHoursDelta)</div>
|
2026-03-17 22:10:19 +01:00
|
|
|
</div>
|
|
|
|
|
<div class="col-6 col-md-4 col-lg-3">
|
|
|
|
|
<label class="form-label text-muted">Gross Income</label>
|
|
|
|
|
<div class="form-control-plaintext fw-bold">€@grossIncome.ToString("N2")</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="col-6 col-md-4 col-lg-3">
|
|
|
|
|
<label class="form-label text-muted">Net Income</label>
|
|
|
|
|
<div class="form-control-plaintext fw-bold">€@netIncome.ToString("N2")</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
<div class="mt-4">
|
|
|
|
|
<h2 class="h5">Day Total</h2>
|
|
|
|
|
<div class="row g-3">
|
|
|
|
|
<div class="col-6 col-md-4 col-lg-3">
|
|
|
|
|
<label class="form-label text-muted">Total Hours For Day</label>
|
|
|
|
|
<div class="form-control-plaintext fw-bold">@FormatHours(dayTotalHours)</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="col-6 col-md-4 col-lg-3">
|
|
|
|
|
<label class="form-label text-muted">Work Units Counted</label>
|
|
|
|
|
<div class="form-control-plaintext fw-bold">@dayWorkUnitCount</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
2026-03-17 22:10:19 +01:00
|
|
|
<div class="d-flex align-items-center gap-2 mt-4">
|
|
|
|
|
<button class="btn btn-primary" @onclick="SaveAsync">Save</button>
|
2026-04-20 16:11:27 +02:00
|
|
|
@if (isExistingUnit)
|
|
|
|
|
{
|
|
|
|
|
<button class="btn btn-outline-danger" @onclick="DeleteAsync">Delete</button>
|
|
|
|
|
}
|
|
|
|
|
<button class="btn btn-outline-secondary" @onclick="BackToCalendar">Back to Calendar</button>
|
2026-03-17 22:10:19 +01:00
|
|
|
@if (!string.IsNullOrWhiteSpace(statusMessage))
|
|
|
|
|
{
|
|
|
|
|
<span class="text-success">@statusMessage</span>
|
|
|
|
|
}
|
|
|
|
|
</div>
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@code {
|
|
|
|
|
[Parameter] public string? DateStr { get; set; }
|
2026-04-20 16:11:27 +02:00
|
|
|
[Parameter] public string? UnitId { get; set; }
|
2026-03-17 22:10:19 +01:00
|
|
|
|
|
|
|
|
private bool loaded;
|
|
|
|
|
private DateOnly selectedDate = DateOnly.FromDateTime(DateTime.Today);
|
2026-04-20 16:11:27 +02:00
|
|
|
private string unitId = string.Empty;
|
|
|
|
|
private string label = "Work unit";
|
|
|
|
|
private WorkUnitLocation location = WorkUnitLocation.Office;
|
2026-03-17 22:10:19 +01:00
|
|
|
private string? startTimeStr;
|
2026-04-20 16:11:27 +02:00
|
|
|
private string? endTimeStr;
|
|
|
|
|
private decimal manualWorkedHours;
|
|
|
|
|
private string manualWorkedHoursStr = "00:00";
|
|
|
|
|
private bool isPreview;
|
2026-03-17 22:10:19 +01:00
|
|
|
private string? notes;
|
|
|
|
|
private string? statusMessage;
|
2026-04-20 16:11:27 +02:00
|
|
|
private bool isExistingUnit;
|
|
|
|
|
private WorkDayDocument? selectedDay;
|
2026-03-17 22:10:19 +01:00
|
|
|
|
|
|
|
|
private decimal grossIncome;
|
|
|
|
|
private decimal netIncome;
|
2026-04-20 16:11:27 +02:00
|
|
|
private decimal? calculatedWorkedHours;
|
|
|
|
|
private decimal workedHoursDelta;
|
|
|
|
|
private decimal dayTotalHours;
|
|
|
|
|
private int dayWorkUnitCount;
|
2026-03-17 22:10:19 +01:00
|
|
|
|
|
|
|
|
private AppSettingsDocument settings = new();
|
|
|
|
|
|
|
|
|
|
protected override async Task OnInitializedAsync()
|
|
|
|
|
{
|
|
|
|
|
if (!string.IsNullOrEmpty(DateStr) && DateOnly.TryParseExact(DateStr, "yyyy-MM-dd", out var parsed))
|
|
|
|
|
{
|
|
|
|
|
selectedDate = parsed;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
settings = await AppSettingsService.GetAsync();
|
2026-04-20 16:11:27 +02:00
|
|
|
await LoadUnitAsync();
|
2026-03-17 22:10:19 +01:00
|
|
|
loaded = true;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private async Task LoadUnitAsync()
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
if (string.IsNullOrWhiteSpace(UnitId))
|
|
|
|
|
{
|
|
|
|
|
selectedDay = await WorkDayService.GetAsync(selectedDate);
|
|
|
|
|
SetDefaults();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
selectedDay = await WorkDayService.GetAsync(selectedDate);
|
|
|
|
|
var existing = await WorkDayService.GetWorkUnitAsync(selectedDate, UnitId);
|
2026-03-17 22:10:19 +01:00
|
|
|
if (existing is not null)
|
|
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
unitId = existing.Id;
|
|
|
|
|
label = existing.Label;
|
|
|
|
|
location = existing.Location;
|
2026-03-17 22:10:19 +01:00
|
|
|
startTimeStr = existing.StartTime?.ToString("HH:mm");
|
2026-04-20 16:11:27 +02:00
|
|
|
endTimeStr = existing.EndTime?.ToString("HH:mm");
|
|
|
|
|
manualWorkedHours = existing.ManualWorkedHours;
|
|
|
|
|
manualWorkedHoursStr = FormatDurationHours(existing.ManualWorkedHours);
|
|
|
|
|
isPreview = existing.IsPreview;
|
2026-03-17 22:10:19 +01:00
|
|
|
notes = existing.Notes;
|
2026-04-20 16:11:27 +02:00
|
|
|
isExistingUnit = true;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
SetDefaults();
|
|
|
|
|
statusMessage = "The selected work unit was not found. A new unit will be created for this day.";
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
2026-04-20 16:11:27 +02:00
|
|
|
|
|
|
|
|
RecomputePreview();
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task OnDateChanged(ChangeEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
if (DateOnly.TryParse(e.Value?.ToString(), out var d))
|
|
|
|
|
{
|
|
|
|
|
selectedDate = d;
|
|
|
|
|
statusMessage = null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private Task OnStartTimeChanged(ChangeEventArgs e)
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
|
|
|
|
startTimeStr = e.Value?.ToString();
|
2026-04-20 16:11:27 +02:00
|
|
|
SyncManualHoursToCalculated();
|
2026-03-17 22:10:19 +01:00
|
|
|
statusMessage = null;
|
2026-04-20 16:11:27 +02:00
|
|
|
return Task.CompletedTask;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private Task OnEndTimeChanged(ChangeEventArgs e)
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
endTimeStr = e.Value?.ToString();
|
|
|
|
|
SyncManualHoursToCalculated();
|
2026-03-17 22:10:19 +01:00
|
|
|
statusMessage = null;
|
2026-04-20 16:11:27 +02:00
|
|
|
return Task.CompletedTask;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private Task OnManualWorkedHoursChanged(ChangeEventArgs e)
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
var rawValue = e.Value?.ToString();
|
|
|
|
|
if (TryParseDurationHours(rawValue, out var parsedHours))
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
manualWorkedHours = parsedHours;
|
|
|
|
|
manualWorkedHoursStr = FormatDurationHours(parsedHours);
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
2026-04-20 16:11:27 +02:00
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
manualWorkedHoursStr = FormatDurationHours(manualWorkedHours);
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-17 22:10:19 +01:00
|
|
|
RecomputePreview();
|
|
|
|
|
statusMessage = null;
|
2026-04-20 16:11:27 +02:00
|
|
|
return Task.CompletedTask;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private void SetDefaults()
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
unitId = string.Empty;
|
|
|
|
|
label = "Work unit";
|
|
|
|
|
location = WorkUnitLocation.Office;
|
|
|
|
|
startTimeStr = null;
|
|
|
|
|
endTimeStr = null;
|
|
|
|
|
manualWorkedHours = 0m;
|
|
|
|
|
manualWorkedHoursStr = "00:00";
|
|
|
|
|
isPreview = false;
|
|
|
|
|
notes = null;
|
|
|
|
|
isExistingUnit = false;
|
|
|
|
|
RecomputePreview();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void SyncManualHoursToCalculated()
|
|
|
|
|
{
|
|
|
|
|
var calculated = CalculateDuration(ParseTime(startTimeStr), ParseTime(endTimeStr));
|
|
|
|
|
manualWorkedHours = calculated ?? 0m;
|
|
|
|
|
manualWorkedHoursStr = FormatDurationHours(manualWorkedHours);
|
|
|
|
|
RecomputePreview();
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void RecomputePreview()
|
|
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
calculatedWorkedHours = CalculateDuration(ParseTime(startTimeStr), ParseTime(endTimeStr));
|
|
|
|
|
workedHoursDelta = manualWorkedHours - (calculatedWorkedHours ?? 0m);
|
|
|
|
|
grossIncome = manualWorkedHours * settings.HourlyGrossRate;
|
|
|
|
|
var taxableBase = grossIncome * settings.ProfitabilityCoefficient;
|
|
|
|
|
netIncome = grossIncome - (taxableBase * settings.InpsRate) - (taxableBase * settings.SubstituteTaxRate);
|
|
|
|
|
RecomputeDayTotals();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task SaveAsync()
|
|
|
|
|
{
|
|
|
|
|
RecomputePreview();
|
|
|
|
|
|
|
|
|
|
var workUnit = new WorkUnitDocument
|
|
|
|
|
{
|
|
|
|
|
Id = unitId,
|
|
|
|
|
Label = label,
|
|
|
|
|
Location = location,
|
|
|
|
|
StartTime = ParseTime(startTimeStr),
|
|
|
|
|
EndTime = ParseTime(endTimeStr),
|
|
|
|
|
ManualWorkedHours = Math.Max(0m, manualWorkedHours),
|
|
|
|
|
IsPreview = isPreview,
|
|
|
|
|
Notes = notes
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var saved = await WorkDayService.SaveWorkUnitAsync(selectedDate, workUnit);
|
|
|
|
|
|
|
|
|
|
unitId = saved.Id;
|
|
|
|
|
isExistingUnit = true;
|
|
|
|
|
label = saved.Label;
|
|
|
|
|
location = saved.Location;
|
|
|
|
|
startTimeStr = saved.StartTime?.ToString("HH:mm");
|
|
|
|
|
endTimeStr = saved.EndTime?.ToString("HH:mm");
|
|
|
|
|
manualWorkedHours = saved.ManualWorkedHours;
|
|
|
|
|
manualWorkedHoursStr = FormatDurationHours(saved.ManualWorkedHours);
|
|
|
|
|
isPreview = saved.IsPreview;
|
|
|
|
|
notes = saved.Notes;
|
|
|
|
|
calculatedWorkedHours = saved.CalculatedWorkedHours;
|
|
|
|
|
workedHoursDelta = saved.WorkedHoursDelta;
|
|
|
|
|
grossIncome = saved.GrossIncome;
|
|
|
|
|
netIncome = saved.NetIncome;
|
|
|
|
|
|
|
|
|
|
Navigation.NavigateTo($"/calendar/{selectedDate:yyyy-MM}");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task DeleteAsync()
|
|
|
|
|
{
|
|
|
|
|
if (!isExistingUnit || string.IsNullOrWhiteSpace(unitId))
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
return;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
var confirmed = await JS.InvokeAsync<bool>("confirm", $"Delete work unit '{label}' on {selectedDate:dddd d MMMM}?\nThis cannot be undone.");
|
|
|
|
|
if (!confirmed)
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
return;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
2026-04-20 16:11:27 +02:00
|
|
|
|
|
|
|
|
var deleted = await WorkDayService.DeleteWorkUnitAsync(selectedDate, unitId);
|
|
|
|
|
if (deleted)
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
Navigation.NavigateTo($"/calendar/{selectedDate:yyyy-MM}");
|
|
|
|
|
return;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
statusMessage = "Unable to delete the work unit.";
|
|
|
|
|
}
|
2026-03-17 22:10:19 +01:00
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private void BackToCalendar()
|
|
|
|
|
{
|
|
|
|
|
Navigation.NavigateTo($"/calendar/{selectedDate:yyyy-MM}");
|
|
|
|
|
}
|
2026-03-17 22:10:19 +01:00
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private void RecomputeDayTotals()
|
|
|
|
|
{
|
|
|
|
|
var existingUnits = selectedDay?.WorkUnits ?? [];
|
|
|
|
|
dayTotalHours = existingUnits
|
|
|
|
|
.Where(unit => !string.Equals(unit.Id, unitId, StringComparison.Ordinal))
|
|
|
|
|
.Sum(unit => unit.ManualWorkedHours) + manualWorkedHours;
|
|
|
|
|
|
|
|
|
|
dayWorkUnitCount = existingUnits
|
|
|
|
|
.Where(unit => !string.Equals(unit.Id, unitId, StringComparison.Ordinal))
|
|
|
|
|
.Count() + 1;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private static TimeOnly? ParseTime(string? value)
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
return !string.IsNullOrWhiteSpace(value) && TimeOnly.TryParse(value, out var parsed)
|
|
|
|
|
? parsed
|
|
|
|
|
: null;
|
|
|
|
|
}
|
2026-03-17 22:10:19 +01:00
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private static decimal? CalculateDuration(TimeOnly? startTime, TimeOnly? endTime)
|
|
|
|
|
{
|
|
|
|
|
if (!startTime.HasValue || !endTime.HasValue || endTime <= startTime)
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
return null;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
2026-04-20 16:11:27 +02:00
|
|
|
|
|
|
|
|
return Math.Round((decimal)(endTime.Value - startTime.Value).TotalHours, 2, MidpointRounding.AwayFromZero);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static bool TryParseDurationHours(string? value, out decimal hours)
|
|
|
|
|
{
|
|
|
|
|
hours = 0m;
|
|
|
|
|
if (string.IsNullOrWhiteSpace(value))
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
return true;
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
if (TimeSpan.TryParseExact(value, [@"h\:mm", @"hh\:mm"], null, out var timeSpan))
|
2026-03-17 22:10:19 +01:00
|
|
|
{
|
2026-04-20 16:11:27 +02:00
|
|
|
hours = Math.Round((decimal)timeSpan.TotalMinutes / 60m, 2, MidpointRounding.AwayFromZero);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2026-03-17 22:10:19 +01:00
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
if (decimal.TryParse(value, out var decimalHours))
|
|
|
|
|
{
|
|
|
|
|
hours = Math.Max(0m, decimalHours);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2026-03-17 22:10:19 +01:00
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-20 23:56:23 +02:00
|
|
|
private static string FormatHours(decimal? value) => value.HasValue ? DurationFormatter.FormatHours(value.Value) : "—";
|
2026-04-20 16:11:27 +02:00
|
|
|
|
2026-04-20 23:56:23 +02:00
|
|
|
private static string FormatSignedHours(decimal value) => DurationFormatter.FormatSignedHours(value);
|
2026-03-17 22:10:19 +01:00
|
|
|
|
2026-04-20 16:11:27 +02:00
|
|
|
private static string FormatDurationHours(decimal value)
|
|
|
|
|
{
|
2026-04-20 23:56:23 +02:00
|
|
|
return DurationFormatter.FormatHours(value);
|
2026-04-20 16:11:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnParametersSet()
|
|
|
|
|
{
|
|
|
|
|
RecomputePreview();
|
2026-03-17 22:10:19 +01:00
|
|
|
}
|
|
|
|
|
}
|