1c7e704ad3
Implements a flexible validation framework for student rankings, team assignments, and team composition with administrator-configurable rules and thresholds. Validation System: - Reflection-based rule discovery eliminates manual registration - Base classes (RequiredEventTypeRuleBase, EventCountThresholdRuleBase) reduce code duplication by ~250 lines - 12 validation rules covering event requirements, counts, and team constraints - Configurable severity levels (Warning/Error) per rule type - ValidationService with caching for optimal performance - Rules apply contextually (StudentRanking, StudentAssignment, TeamComposition) Configuration & Admin UI: - ValidationSettings admin page for editing thresholds and severity levels - ChapterSettings admin page for editing chapter information - Settings stored in Data/appsettings.json for runtime configuration - JSON config works in both development and production environments - Auto-creates Data directory and config template on first run User Experience: - ValidationWarnings component displays inline warnings/errors - Integrated in Event Ranking, Registration, and Team Assignment pages - Color-coded severity indicators (warning yellow, error red) - Includes "Too Many Regional Events" rule (max 3 recommended) Technical Improvements: - Template Method pattern for rule base classes - Singleton rule instances with lazy initialization - Configuration loaded via IConfiguration with fallback to defaults - Safe JSON updates preserve other appsettings sections
187 lines
6.7 KiB
Plaintext
187 lines
6.7 KiB
Plaintext
@page "/settings/chapter"
|
|
@attribute [Authorize(Roles = AuthRoles.Administrator)]
|
|
@using WebApp.Authentication
|
|
@using WebApp.Models
|
|
@using System.Text.Json
|
|
@inject IWebHostEnvironment Environment
|
|
@inject IConfiguration Configuration
|
|
|
|
@rendermode InteractiveServer
|
|
|
|
<PageTitle>Chapter Settings</PageTitle>
|
|
|
|
<MudContainer MaxWidth="MaxWidth.Large" Class="mt-4">
|
|
<MudText Typo="Typo.h3" Class="mb-4">Chapter Settings</MudText>
|
|
<MudText Typo="Typo.body1" Class="mb-6">Configure chapter information. Changes take effect on next application restart.</MudText>
|
|
|
|
@if (_settings != null)
|
|
{
|
|
<MudPaper Class="pa-6 mb-4">
|
|
<MudText Typo="Typo.h5" Class="mb-4">Basic Information</MudText>
|
|
<MudGrid>
|
|
<MudItem xs="12" md="8">
|
|
<MudTextField @bind-Value="_settings.Name"
|
|
Label="Chapter Name"
|
|
Variant="Variant.Outlined"
|
|
HelperText="Full chapter name"
|
|
Required="true" />
|
|
</MudItem>
|
|
<MudItem xs="12" md="4">
|
|
<MudTextField @bind-Value="_settings.ShortName"
|
|
Label="Short Name"
|
|
Variant="Variant.Outlined"
|
|
HelperText="Abbreviation (e.g., YCN)"
|
|
MaxLength="10"
|
|
Required="true" />
|
|
</MudItem>
|
|
<MudItem xs="12" md="6">
|
|
<MudTextField @bind-Value="_settings.CompetitionYear"
|
|
Label="Competition Year"
|
|
Variant="Variant.Outlined"
|
|
HelperText="Year of competition (e.g., 2026)"
|
|
MaxLength="4"
|
|
Required="true" />
|
|
</MudItem>
|
|
</MudGrid>
|
|
</MudPaper>
|
|
|
|
<MudPaper Class="pa-6 mb-4">
|
|
<MudText Typo="Typo.h5" Class="mb-4">Chapter IDs</MudText>
|
|
<MudGrid>
|
|
<MudItem xs="12" md="4">
|
|
<MudTextField @bind-Value="_settings.NationalId"
|
|
Label="National ID"
|
|
Variant="Variant.Outlined"
|
|
HelperText="4-digit national chapter ID"
|
|
MaxLength="4" />
|
|
</MudItem>
|
|
<MudItem xs="12" md="4">
|
|
<MudTextField @bind-Value="_settings.StateId"
|
|
Label="State ID"
|
|
Variant="Variant.Outlined"
|
|
HelperText="5-digit state chapter ID"
|
|
MaxLength="5" />
|
|
</MudItem>
|
|
<MudItem xs="12" md="4">
|
|
<MudTextField @bind-Value="_settings.RegionalId"
|
|
Label="Regional ID"
|
|
Variant="Variant.Outlined"
|
|
HelperText="5-digit regional chapter ID"
|
|
MaxLength="5" />
|
|
</MudItem>
|
|
</MudGrid>
|
|
</MudPaper>
|
|
|
|
<MudPaper Class="pa-6">
|
|
<MudGrid>
|
|
<MudItem xs="12">
|
|
<MudButton Variant="Variant.Filled"
|
|
Color="Color.Primary"
|
|
StartIcon="@Icons.Material.Filled.Save"
|
|
OnClick="SaveSettings"
|
|
Disabled="_isSaving">
|
|
@if (_isSaving)
|
|
{
|
|
<MudProgressCircular Class="mr-2" Size="Size.Small" Indeterminate="true" />
|
|
<span>Saving...</span>
|
|
}
|
|
else
|
|
{
|
|
<span>Save Settings</span>
|
|
}
|
|
</MudButton>
|
|
</MudItem>
|
|
</MudGrid>
|
|
</MudPaper>
|
|
|
|
@if (!string.IsNullOrEmpty(_statusMessage))
|
|
{
|
|
<MudAlert Severity="@_statusSeverity" Class="mt-4">@_statusMessage</MudAlert>
|
|
}
|
|
}
|
|
else
|
|
{
|
|
<MudProgressCircular Indeterminate="true" />
|
|
}
|
|
</MudContainer>
|
|
|
|
@code {
|
|
private Models.ChapterSettings? _settings;
|
|
private bool _isSaving;
|
|
private string? _statusMessage;
|
|
private Severity _statusSeverity = Severity.Success;
|
|
|
|
protected override void OnInitialized()
|
|
{
|
|
// Load from IConfiguration
|
|
_settings = Configuration.GetSection("ChapterSettings").Get<Models.ChapterSettings>()
|
|
?? new Models.ChapterSettings();
|
|
}
|
|
|
|
private string GetAppSettingsPath()
|
|
{
|
|
return Path.Combine(
|
|
Environment.ContentRootPath,
|
|
"Data",
|
|
"appsettings.json");
|
|
}
|
|
|
|
private async Task SaveSettings()
|
|
{
|
|
if (_settings == null) return;
|
|
|
|
_isSaving = true;
|
|
_statusMessage = null;
|
|
|
|
try
|
|
{
|
|
var appSettingsPath = GetAppSettingsPath();
|
|
|
|
// Ensure Data directory exists
|
|
var dataDir = Path.GetDirectoryName(appSettingsPath);
|
|
if (dataDir != null && !Directory.Exists(dataDir))
|
|
{
|
|
Directory.CreateDirectory(dataDir);
|
|
}
|
|
|
|
// Read existing appsettings or create new
|
|
JsonDocument? existingDoc = null;
|
|
Dictionary<string, object?> settings;
|
|
|
|
if (File.Exists(appSettingsPath))
|
|
{
|
|
var existingJson = await File.ReadAllTextAsync(appSettingsPath);
|
|
existingDoc = JsonDocument.Parse(existingJson);
|
|
settings = JsonSerializer.Deserialize<Dictionary<string, object?>>(existingJson)
|
|
?? new Dictionary<string, object?>();
|
|
}
|
|
else
|
|
{
|
|
settings = new Dictionary<string, object?>();
|
|
}
|
|
|
|
// Update ChapterSettings section
|
|
settings["ChapterSettings"] = _settings;
|
|
|
|
// Write back to file
|
|
var options = new JsonSerializerOptions { WriteIndented = true };
|
|
var json = JsonSerializer.Serialize(settings, options);
|
|
await File.WriteAllTextAsync(appSettingsPath, json);
|
|
|
|
existingDoc?.Dispose();
|
|
|
|
_statusMessage = "Settings saved successfully! Changes will take effect on next application restart.";
|
|
_statusSeverity = Severity.Success;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_statusMessage = $"Error saving settings: {ex.Message}";
|
|
_statusSeverity = Severity.Error;
|
|
}
|
|
finally
|
|
{
|
|
_isSaving = false;
|
|
}
|
|
}
|
|
}
|