diff --git a/Directory.Packages.props b/Directory.Packages.props
index 903f739..e8262ad 100644
--- a/Directory.Packages.props
+++ b/Directory.Packages.props
@@ -46,5 +46,7 @@
+
+
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/App.razor b/RomRepoMgr.Blazor/Components/App.razor
new file mode 100644
index 0000000..be1e43f
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/App.razor
@@ -0,0 +1,20 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/Layout/MainLayout.razor b/RomRepoMgr.Blazor/Components/Layout/MainLayout.razor
new file mode 100644
index 0000000..fe28016
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/Layout/MainLayout.razor
@@ -0,0 +1,26 @@
+@inherits LayoutComponentBase
+
+
+
+ RomRepoMgr.Blazor
+
+
+
+
+
+ @Body
+
+
+
+
+ Documentation and demos
+
+ About Blazor
+
+
+
+
+ An unhandled error has occurred.
+
Reload
+
🗙
+
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/Layout/NavMenu.razor b/RomRepoMgr.Blazor/Components/Layout/NavMenu.razor
new file mode 100644
index 0000000..963bc94
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/Layout/NavMenu.razor
@@ -0,0 +1,19 @@
+@rendermode InteractiveServer
+
+
+
+@code {
+ private bool expanded = true;
+}
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/Pages/Counter.razor b/RomRepoMgr.Blazor/Components/Pages/Counter.razor
new file mode 100644
index 0000000..c0215cc
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/Pages/Counter.razor
@@ -0,0 +1,22 @@
+@page "/counter"
+@rendermode InteractiveServer
+
+Counter
+
+Counter
+
+
+ Current count: @currentCount
+
+
+Click me
+
+@code {
+ private int currentCount = 0;
+
+ private void IncrementCount()
+ {
+ currentCount++;
+ }
+
+}
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/Pages/Error.razor b/RomRepoMgr.Blazor/Components/Pages/Error.razor
new file mode 100644
index 0000000..adab1c1
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/Pages/Error.razor
@@ -0,0 +1,35 @@
+@page "/Error"
+@using System.Diagnostics
+
+Error
+
+Error.
+An error occurred while processing your request.
+
+@if(ShowRequestId)
+{
+
+ Request ID: @RequestId
+
+}
+
+Development Mode
+
+ Swapping to Development environment will display more detailed information about the error that occurred.
+
+
+ The Development environment shouldn't be enabled for deployed applications.
+ It can result in displaying sensitive information from exceptions to end users.
+ For local debugging, enable the Development environment by setting the ASPNETCORE_ENVIRONMENT environment variable to Development
+ and restarting the app.
+
+
+@code{
+ [CascadingParameter]
+ private HttpContext? HttpContext { get; set; }
+
+ private string? RequestId { get; set; }
+ private bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
+
+ protected override void OnInitialized() => RequestId = Activity.Current?.Id ?? HttpContext?.TraceIdentifier;
+}
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/Pages/Home.razor b/RomRepoMgr.Blazor/Components/Pages/Home.razor
new file mode 100644
index 0000000..96714a2
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/Pages/Home.razor
@@ -0,0 +1,7 @@
+@page "/"
+
+Home
+
+Hello, world!
+
+Welcome to your new Fluent Blazor app.
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/Pages/Weather.razor b/RomRepoMgr.Blazor/Components/Pages/Weather.razor
new file mode 100644
index 0000000..20f4ec8
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/Pages/Weather.razor
@@ -0,0 +1,51 @@
+@page "/weather"
+@attribute [StreamRendering]
+
+Weather
+
+Weather
+
+This component demonstrates showing data.
+
+
+
+
+
+
+
+
+
+@code {
+ private IQueryable? forecasts;
+
+ protected override async Task OnInitializedAsync()
+ {
+ // Simulate asynchronous loading to demonstrate streaming rendering
+ await Task.Delay(500);
+
+ var startDate = DateOnly.FromDateTime(DateTime.Now);
+
+ var summaries = new[]
+ {
+ "Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering", "Scorching"
+ };
+
+ forecasts = Enumerable.Range(1, 5)
+ .Select(index => new WeatherForecast
+ {
+ Date = startDate.AddDays(index),
+ TemperatureC = Random.Shared.Next(-20, 55),
+ Summary = summaries[Random.Shared.Next(summaries.Length)]
+ })
+ .AsQueryable();
+ }
+
+ private class WeatherForecast
+ {
+ public DateOnly Date { get; set; }
+ public int TemperatureC { get; set; }
+ public string? Summary { get; set; }
+ public int TemperatureF => 32 + (int)(TemperatureC / 0.5556);
+ }
+
+}
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/Routes.razor b/RomRepoMgr.Blazor/Components/Routes.razor
new file mode 100644
index 0000000..22b036e
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/Routes.razor
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Components/_Imports.razor b/RomRepoMgr.Blazor/Components/_Imports.razor
new file mode 100644
index 0000000..33aea7d
--- /dev/null
+++ b/RomRepoMgr.Blazor/Components/_Imports.razor
@@ -0,0 +1,12 @@
+@using System.Net.Http
+@using System.Net.Http.Json
+@using Microsoft.AspNetCore.Components.Forms
+@using Microsoft.AspNetCore.Components.Routing
+@using Microsoft.AspNetCore.Components.Web
+@using static Microsoft.AspNetCore.Components.Web.RenderMode
+@using Microsoft.AspNetCore.Components.Web.Virtualization
+@using Microsoft.FluentUI.AspNetCore.Components
+@using Icons = Microsoft.FluentUI.AspNetCore.Components.Icons
+@using Microsoft.JSInterop
+@using RomRepoMgr.Blazor
+@using RomRepoMgr.Blazor.Components
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Program.cs b/RomRepoMgr.Blazor/Program.cs
new file mode 100644
index 0000000..1a2cb7f
--- /dev/null
+++ b/RomRepoMgr.Blazor/Program.cs
@@ -0,0 +1,28 @@
+using Microsoft.FluentUI.AspNetCore.Components;
+using RomRepoMgr.Blazor.Components;
+
+var builder = WebApplication.CreateBuilder(args);
+
+// Add services to the container.
+builder.Services.AddRazorComponents().AddInteractiveServerComponents();
+builder.Services.AddFluentUIComponents();
+
+var app = builder.Build();
+
+// Configure the HTTP request pipeline.
+if(!app.Environment.IsDevelopment())
+{
+ app.UseExceptionHandler("/Error", createScopeForErrors: true);
+
+ // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
+ app.UseHsts();
+}
+
+app.UseHttpsRedirection();
+
+app.UseAntiforgery();
+
+app.MapStaticAssets();
+app.MapRazorComponents().AddInteractiveServerRenderMode();
+
+app.Run();
\ No newline at end of file
diff --git a/RomRepoMgr.Blazor/Properties/launchSettings.json b/RomRepoMgr.Blazor/Properties/launchSettings.json
new file mode 100644
index 0000000..3597ff3
--- /dev/null
+++ b/RomRepoMgr.Blazor/Properties/launchSettings.json
@@ -0,0 +1,23 @@
+{
+ "$schema": "https://json.schemastore.org/launchsettings.json",
+ "profiles": {
+ "http": {
+ "commandName": "Project",
+ "dotnetRunMessages": true,
+ "launchBrowser": true,
+ "applicationUrl": "http://localhost:5079",
+ "environmentVariables": {
+ "ASPNETCORE_ENVIRONMENT": "Development"
+ }
+ },
+ "https": {
+ "commandName": "Project",
+ "dotnetRunMessages": true,
+ "launchBrowser": true,
+ "applicationUrl": "https://localhost:7057;http://localhost:5079",
+ "environmentVariables": {
+ "ASPNETCORE_ENVIRONMENT": "Development"
+ }
+ }
+ }
+ }
diff --git a/RomRepoMgr.Blazor/RomRepoMgr.Blazor.csproj b/RomRepoMgr.Blazor/RomRepoMgr.Blazor.csproj
new file mode 100644
index 0000000..35ea963
--- /dev/null
+++ b/RomRepoMgr.Blazor/RomRepoMgr.Blazor.csproj
@@ -0,0 +1,13 @@
+
+
+
+ net9.0
+ enable
+ enable
+
+
+
+
+
+
+
diff --git a/RomRepoMgr.Blazor/appsettings.Development.json b/RomRepoMgr.Blazor/appsettings.Development.json
new file mode 100644
index 0000000..0c208ae
--- /dev/null
+++ b/RomRepoMgr.Blazor/appsettings.Development.json
@@ -0,0 +1,8 @@
+{
+ "Logging": {
+ "LogLevel": {
+ "Default": "Information",
+ "Microsoft.AspNetCore": "Warning"
+ }
+ }
+}
diff --git a/RomRepoMgr.Blazor/appsettings.json b/RomRepoMgr.Blazor/appsettings.json
new file mode 100644
index 0000000..10f68b8
--- /dev/null
+++ b/RomRepoMgr.Blazor/appsettings.json
@@ -0,0 +1,9 @@
+{
+ "Logging": {
+ "LogLevel": {
+ "Default": "Information",
+ "Microsoft.AspNetCore": "Warning"
+ }
+ },
+ "AllowedHosts": "*"
+}
diff --git a/RomRepoMgr.Blazor/wwwroot/app.css b/RomRepoMgr.Blazor/wwwroot/app.css
new file mode 100644
index 0000000..77cf978
--- /dev/null
+++ b/RomRepoMgr.Blazor/wwwroot/app.css
@@ -0,0 +1,191 @@
+@import '_content/Microsoft.FluentUI.AspNetCore.Components/css/reboot.css';
+
+body {
+ --body-font: "Segoe UI Variable", "Segoe UI", sans-serif;
+ font-family: var(--body-font);
+ font-size: var(--type-ramp-base-font-size);
+ line-height: var(--type-ramp-base-line-height);
+ margin: 0;
+}
+
+.navmenu-icon {
+ display: none;
+}
+
+.main {
+ min-height: calc(100dvh - 86px);
+ color: var(--neutral-foreground-rest);
+ align-items: stretch !important;
+}
+
+.body-content {
+ align-self: stretch;
+ height: calc(100dvh - 86px) !important;
+ display: flex;
+}
+
+.content {
+ padding: 0.5rem 1.5rem;
+ align-self: stretch !important;
+ width: 100%;
+}
+
+.manage {
+ width: 100dvw;
+}
+
+footer {
+ background: var(--neutral-layer-4);
+ color: var(--neutral-foreground-rest);
+ align-items: center;
+ padding: 10px 10px;
+}
+
+ footer a {
+ color: var(--neutral-foreground-rest);
+ text-decoration: none;
+ }
+
+ footer a:focus {
+ outline: 1px dashed;
+ outline-offset: 3px;
+ }
+
+ footer a:hover {
+ text-decoration: underline;
+ }
+
+.alert {
+ border: 1px dashed var(--accent-fill-rest);
+ padding: 5px;
+}
+
+
+#blazor-error-ui {
+ background: lightyellow;
+ bottom: 0;
+ box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2);
+ display: none;
+ left: 0;
+ padding: 0.6rem 1.25rem 0.7rem 1.25rem;
+ position: fixed;
+ width: 100%;
+ z-index: 1000;
+ margin: 20px 0;
+}
+
+ #blazor-error-ui .dismiss {
+ cursor: pointer;
+ position: absolute;
+ right: 0.75rem;
+ top: 0.5rem;
+ }
+
+.blazor-error-boundary {
+ background: url(data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNTYiIGhlaWdodD0iNDkiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIG92ZXJmbG93PSJoaWRkZW4iPjxkZWZzPjxjbGlwUGF0aCBpZD0iY2xpcDAiPjxyZWN0IHg9IjIzNSIgeT0iNTEiIHdpZHRoPSI1NiIgaGVpZ2h0PSI0OSIvPjwvY2xpcFBhdGg+PC9kZWZzPjxnIGNsaXAtcGF0aD0idXJsKCNjbGlwMCkiIHRyYW5zZm9ybT0idHJhbnNsYXRlKC0yMzUgLTUxKSI+PHBhdGggZD0iTTI2My41MDYgNTFDMjY0LjcxNyA1MSAyNjUuODEzIDUxLjQ4MzcgMjY2LjYwNiA1Mi4yNjU4TDI2Ny4wNTIgNTIuNzk4NyAyNjcuNTM5IDUzLjYyODMgMjkwLjE4NSA5Mi4xODMxIDI5MC41NDUgOTIuNzk1IDI5MC42NTYgOTIuOTk2QzI5MC44NzcgOTMuNTEzIDI5MSA5NC4wODE1IDI5MSA5NC42NzgyIDI5MSA5Ny4wNjUxIDI4OS4wMzggOTkgMjg2LjYxNyA5OUwyNDAuMzgzIDk5QzIzNy45NjMgOTkgMjM2IDk3LjA2NTEgMjM2IDk0LjY3ODIgMjM2IDk0LjM3OTkgMjM2LjAzMSA5NC4wODg2IDIzNi4wODkgOTMuODA3MkwyMzYuMzM4IDkzLjAxNjIgMjM2Ljg1OCA5Mi4xMzE0IDI1OS40NzMgNTMuNjI5NCAyNTkuOTYxIDUyLjc5ODUgMjYwLjQwNyA1Mi4yNjU4QzI2MS4yIDUxLjQ4MzcgMjYyLjI5NiA1MSAyNjMuNTA2IDUxWk0yNjMuNTg2IDY2LjAxODNDMjYwLjczNyA2Ni4wMTgzIDI1OS4zMTMgNjcuMTI0NSAyNTkuMzEzIDY5LjMzNyAyNTkuMzEzIDY5LjYxMDIgMjU5LjMzMiA2OS44NjA4IDI1OS4zNzEgNzAuMDg4N0wyNjEuNzk1IDg0LjAxNjEgMjY1LjM4IDg0LjAxNjEgMjY3LjgyMSA2OS43NDc1QzI2Ny44NiA2OS43MzA5IDI2Ny44NzkgNjkuNTg3NyAyNjcuODc5IDY5LjMxNzkgMjY3Ljg3OSA2Ny4xMTgyIDI2Ni40NDggNjYuMDE4MyAyNjMuNTg2IDY2LjAxODNaTTI2My41NzYgODYuMDU0N0MyNjEuMDQ5IDg2LjA1NDcgMjU5Ljc4NiA4Ny4zMDA1IDI1OS43ODYgODkuNzkyMSAyNTkuNzg2IDkyLjI4MzcgMjYxLjA0OSA5My41Mjk1IDI2My41NzYgOTMuNTI5NSAyNjYuMTE2IDkzLjUyOTUgMjY3LjM4NyA5Mi4yODM3IDI2Ny4zODcgODkuNzkyMSAyNjcuMzg3IDg3LjMwMDUgMjY2LjExNiA4Ni4wNTQ3IDI2My41NzYgODYuMDU0N1oiIGZpbGw9IiNGRkU1MDAiIGZpbGwtcnVsZT0iZXZlbm9kZCIvPjwvZz48L3N2Zz4=) no-repeat 1rem/1.8rem, #b32121;
+ padding: 1rem 1rem 1rem 3.7rem;
+ color: white;
+}
+
+ .blazor-error-boundary::before {
+ content: "An error has occurred. "
+ }
+
+.loading-progress {
+ position: relative;
+ display: block;
+ width: 8rem;
+ height: 8rem;
+ margin: 20vh auto 1rem auto;
+}
+
+ .loading-progress circle {
+ fill: none;
+ stroke: #e0e0e0;
+ stroke-width: 0.6rem;
+ transform-origin: 50% 50%;
+ transform: rotate(-90deg);
+ }
+
+ .loading-progress circle:last-child {
+ stroke: #1b6ec2;
+ stroke-dasharray: calc(3.141 * var(--blazor-load-percentage, 0%) * 0.8), 500%;
+ transition: stroke-dasharray 0.05s ease-in-out;
+ }
+
+.loading-progress-text {
+ position: absolute;
+ text-align: center;
+ font-weight: bold;
+ inset: calc(20vh + 3.25rem) 0 auto 0.2rem;
+}
+
+ .loading-progress-text:after {
+ content: var(--blazor-load-percentage-text, "Loading");
+ }
+
+code {
+ color: #c02d76;
+}
+
+@media (max-width: 600px) {
+ .header-gutters {
+ margin: 0.5rem 3rem 0.5rem 1.5rem !important;
+ }
+
+ [dir="rtl"] .header-gutters {
+ margin: 0.5rem 1.5rem 0.5rem 3rem !important;
+ }
+
+ .main {
+ flex-direction: column !important;
+ row-gap: 0 !important;
+ }
+
+ nav.sitenav {
+ width: 100%;
+ height: 100%;
+ }
+
+ #main-menu {
+ width: 100% !important;
+ }
+
+ #main-menu > div:first-child:is(.expander) {
+ display: none;
+ }
+
+ .navmenu {
+ width: 100%;
+ }
+
+ #navmenu-toggle {
+ appearance: none;
+ }
+
+ #navmenu-toggle ~ nav {
+ display: none;
+ }
+
+ #navmenu-toggle:checked ~ nav {
+ display: block;
+ }
+
+ .navmenu-icon {
+ cursor: pointer;
+ z-index: 10;
+ display: block;
+ position: absolute;
+ top: 15px;
+ left: unset;
+ right: 20px;
+ width: 20px;
+ height: 20px;
+ border: none;
+ }
+
+ [dir="rtl"] .navmenu-icon {
+ left: 20px;
+ right: unset;
+ }
+}
diff --git a/RomRepoMgr.Blazor/wwwroot/favicon.ico b/RomRepoMgr.Blazor/wwwroot/favicon.ico
new file mode 100644
index 0000000..e189d8e
Binary files /dev/null and b/RomRepoMgr.Blazor/wwwroot/favicon.ico differ
diff --git a/RomRepoMgr.sln b/RomRepoMgr.sln
index b47832c..8084cb4 100644
--- a/RomRepoMgr.sln
+++ b/RomRepoMgr.sln
@@ -32,6 +32,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SabreTools.DatTools", "Sabr
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SabreTools.Reports", "SabreTools\SabreTools.Reports\SabreTools.Reports.csproj", "{E73767A7-0A65-4F89-B149-A520874F7B32}"
EndProject
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "RomRepoMgr.Blazor", "RomRepoMgr.Blazor\RomRepoMgr.Blazor.csproj", "{30DA0637-76C5-43DE-8203-403AECF5F859}"
+EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
@@ -78,5 +80,9 @@ Global
{E73767A7-0A65-4F89-B149-A520874F7B32}.Debug|Any CPU.Build.0 = Debug|Any CPU
{E73767A7-0A65-4F89-B149-A520874F7B32}.Release|Any CPU.ActiveCfg = Release|Any CPU
{E73767A7-0A65-4F89-B149-A520874F7B32}.Release|Any CPU.Build.0 = Release|Any CPU
+ {30DA0637-76C5-43DE-8203-403AECF5F859}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {30DA0637-76C5-43DE-8203-403AECF5F859}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {30DA0637-76C5-43DE-8203-403AECF5F859}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {30DA0637-76C5-43DE-8203-403AECF5F859}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
EndGlobal