Group platform admin tools into hub pages

- add grouped platform admin hub pages, view models, and shared card UI\n- simplify the super admin nav and dashboard quick links around the new hubs\n- fix the AiQuoteService EstimatedMinutes assignment so the infrastructure project builds cleanly
This commit is contained in:
2026-05-12 09:03:18 -04:00
parent 42a8c089d5
commit 31d305b66a
12 changed files with 374 additions and 129 deletions
@@ -619,7 +619,7 @@ Respond with the JSON object only.";
CoatCount = request.CoatCount,
MaterialCost = Math.Round(materialCost, 2),
ConsumablesCost = Math.Round(consumablesSurcharge, 2),
EstimatedMinutes = (int)Math.Round(perItemMinutes),
EstimatedMinutes = perItemMinutes,
MaterialMinMinutes = materialMinMinutes,
MinFloorApplied = minFloorApplied,
LaborCost = Math.Round(laborCost, 2),
@@ -0,0 +1,130 @@
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using PowderCoating.Shared.Constants;
using PowderCoating.Web.ViewModels.PlatformAdmin;
namespace PowderCoating.Web.Controllers;
[Authorize(Policy = AppConstants.Policies.SuperAdminOnly)]
public class PlatformAdminController : Controller
{
private static readonly bool ShowRawLogFiles = string.IsNullOrEmpty(Environment.GetEnvironmentVariable("WEBSITE_SITE_NAME"));
public IActionResult TenantsBilling() => View(BuildTenantsBillingHub());
public IActionResult PeopleActivity() => View(BuildPeopleActivityHub());
public IActionResult ContentMessaging() => View(BuildContentMessagingHub());
public IActionResult Observability() => View(BuildObservabilityHub());
public IActionResult Maintenance() => View(BuildMaintenanceHub());
private static PlatformAdminHubViewModel BuildTenantsBillingHub() => new()
{
Title = "Tenants & Billing",
PageIcon = "bi-building-gear",
Intro = "Manage tenant accounts, subscription health, pricing plans, and payment-system signals from one place.",
Cards = new List<PlatformAdminLinkCardViewModel>
{
Card("Companies", "Browse all tenant companies, create new accounts, and jump into company-level details.", "Companies", "Index", "bi-building", "Daily", "text-bg-primary-subtle border border-primary-subtle"),
Card("Company Health", "Review readiness, setup gaps, and health signals across every company.", "CompanyHealth", "Index", "bi-heart-pulse", "Review", "text-bg-success-subtle border border-success-subtle"),
Card("Subscriptions", "Manage tenant subscriptions, grace periods, expirations, and billing status.", "SubscriptionManagement", "Index", "bi-credit-card", "Daily", "text-bg-warning-subtle border border-warning-subtle"),
Card("Subscription Plans", "Adjust platform plan definitions, limits, pricing, and feature packaging.", "PlatformSubscription", "Index", "bi-layers", "Config", "text-bg-info-subtle border border-info-subtle"),
Card("Revenue Dashboard", "Track platform-level revenue, plan mix, and commercial performance over time.", "Revenue", "Index", "bi-graph-up-arrow", "Monitor", "text-bg-secondary-subtle border border-secondary-subtle"),
Card("Stripe Events", "Inspect incoming Stripe webhook activity and payment-provider events.", "StripeEvents", "Index", "bi-lightning-charge", "Advanced", "text-bg-secondary-subtle border border-secondary-subtle"),
Card("SMS Agreements", "Audit company SMS agreement status and identify accounts blocked by compliance gating.", "SmsAgreements", "Index", "bi-file-earmark-check", "Compliance", "text-bg-danger-subtle border border-danger-subtle")
}
};
private static PlatformAdminHubViewModel BuildPeopleActivityHub() => new()
{
Title = "People & Activity",
PageIcon = "bi-people",
Intro = "See who is using the platform, what they are doing, and which tenants are still onboarding.",
Cards = new List<PlatformAdminLinkCardViewModel>
{
Card("Platform Users", "Manage SuperAdmin accounts and review platform-user access details.", "PlatformUsers", "Index", "bi-people-fill", "Daily", "text-bg-primary-subtle border border-primary-subtle"),
Card("User Activity", "Review cross-tenant usage history, filters, and behavioral trends.", "UserActivity", "Index", "bi-person-lines-fill", "Review", "text-bg-success-subtle border border-success-subtle"),
Card("Online Now", "Check who is currently active in the application right now.", "UserActivity", "Online", "bi-broadcast-pin", "Live", "text-bg-success-subtle border border-success-subtle"),
Card("Onboarding Progress", "Track which companies are still working through setup and activation milestones.", "OnboardingProgress", "Index", "bi-rocket-takeoff", "Support", "text-bg-warning-subtle border border-warning-subtle"),
Card("Platform Notifications", "Review platform-level in-app notifications and operational follow-ups.", "PlatformNotifications", "Index", "bi-bell", "Monitor", "text-bg-info-subtle border border-info-subtle")
}
};
private static PlatformAdminHubViewModel BuildContentMessagingHub() => new()
{
Title = "Content & Messaging",
PageIcon = "bi-megaphone",
Intro = "Manage the platform-facing announcements, release communication, admin outreach, and inbound support signals.",
Cards = new List<PlatformAdminLinkCardViewModel>
{
Card("Announcements", "Publish or retire platform-wide announcements shown to tenant users.", "Announcements", "Index", "bi-megaphone", "Publish", "text-bg-primary-subtle border border-primary-subtle"),
Card("Dashboard Tips", "Curate the rotating tips and educational nudges shown in the product dashboard.", "DashboardTips", "Index", "bi-lightbulb", "Content", "text-bg-warning-subtle border border-warning-subtle"),
Card("Email Broadcast", "Send platform-admin broadcast emails to selected tenant companies.", "EmailBroadcast", "Index", "bi-broadcast", "Support", "text-bg-info-subtle border border-info-subtle"),
Card("Release Notes", "Publish and maintain customer-facing release notes and change summaries.", "ReleaseNotes", "Manage", "bi-journal-text", "Publish", "text-bg-success-subtle border border-success-subtle"),
Card("Contact Submissions", "Review inbound contact requests, questions, and support follow-up items.", "Contact", "Submissions", "bi-envelope", "Inbox", "text-bg-secondary-subtle border border-secondary-subtle"),
Card("Bug Reports", "Triage submitted product bugs and inspect the supporting report detail.", "BugReport", "Index", "bi-bug", "Triage", "text-bg-danger-subtle border border-danger-subtle")
}
};
private static PlatformAdminHubViewModel BuildObservabilityHub()
{
var cards = new List<PlatformAdminLinkCardViewModel>
{
Card("Audit Log", "Review the platform audit trail for sensitive actions and administrative changes.", "AuditLog", "Index", "bi-shield-check", "Investigate", "text-bg-primary-subtle border border-primary-subtle"),
Card("System Logs", "Inspect application warning and error logs from the structured logging pipeline.", "SystemLogs", "Index", "bi-database-exclamation", "Investigate", "text-bg-danger-subtle border border-danger-subtle"),
Card("System Info", "Review environment, infrastructure, and runtime diagnostics for the platform.", "SystemInfo", "Index", "bi-cpu", "Advanced", "text-bg-secondary-subtle border border-secondary-subtle"),
Card("AI Usage", "Monitor AI feature usage, cross-tenant consumption, and feature mix.", "AiUsageReport", "Index", "bi-robot", "Monitor", "text-bg-info-subtle border border-info-subtle"),
Card("Usage & Quota", "Review platform-wide usage ceilings, quota trends, and consumption pressure.", "UsageQuota", "Index", "bi-speedometer2", "Monitor", "text-bg-warning-subtle border border-warning-subtle"),
Card("Banned IPs", "Manage blocked IP addresses and platform-level abuse controls.", "BannedIps", "Index", "bi-slash-circle", "Security", "text-bg-danger-subtle border border-danger-subtle")
};
if (ShowRawLogFiles)
{
cards.Add(Card("Raw Log Files", "Open raw log-file views for deeper troubleshooting in environments where file logs are available.", "Diagnostics", "ViewLogs", "bi-file-text", "Advanced", "text-bg-secondary-subtle border border-secondary-subtle"));
}
return new PlatformAdminHubViewModel
{
Title = "Observability",
PageIcon = "bi-binoculars",
Intro = "Investigate platform behavior across audit trails, system logs, AI usage, quotas, and security-oriented diagnostics.",
Cards = cards
};
}
private static PlatformAdminHubViewModel BuildMaintenanceHub() => new()
{
Title = "Maintenance",
PageIcon = "bi-wrench-adjustable-circle",
Intro = "Use these tools for exceptional maintenance work, migration tasks, and destructive admin operations. They are not routine day-to-day workflows.",
WarningTitle = "Use With Care",
WarningMessage = "These tools can expose bulk data, change platform state, or permanently remove records. Use them deliberately and preferably with a written reason or ticket.",
Cards = new List<PlatformAdminLinkCardViewModel>
{
Card("Data Export", "Export a tenant company's data set for audits, offboarding, support, or migration work.", "DataExport", "Index", "bi-file-earmark-arrow-down", "Maintenance", "text-bg-warning-subtle border border-warning-subtle"),
Card("Data Purge", "Permanently delete soft-deleted records after previewing impact and cutoff windows.", "DataPurge", "Index", "bi-trash3", "Dangerous", "text-bg-danger-subtle border border-danger-subtle"),
Card("Storage Migration", "Run one-off migration of local media files into cloud storage.", "StorageMigration", "Index", "bi-cloud-upload", "One-off", "text-bg-info-subtle border border-info-subtle"),
Card("Seed Data", "Seed or remove system and demo data for setup, QA, or controlled test scenarios.", "SeedData", "Index", "bi-database-fill-gear", "Restricted", "text-bg-danger-subtle border border-danger-subtle")
}
};
private static PlatformAdminLinkCardViewModel Card(
string title,
string description,
string controller,
string action,
string icon,
string badgeText,
string badgeStyle) => new()
{
Title = title,
Description = description,
Controller = controller,
Action = action,
Icon = icon,
BadgeText = badgeText,
BadgeStyle = badgeStyle
};
}
@@ -0,0 +1,11 @@
namespace PowderCoating.Web.ViewModels.PlatformAdmin;
public class PlatformAdminHubViewModel
{
public string Title { get; set; } = string.Empty;
public string PageIcon { get; set; } = string.Empty;
public string Intro { get; set; } = string.Empty;
public string? WarningTitle { get; set; }
public string? WarningMessage { get; set; }
public IReadOnlyList<PlatformAdminLinkCardViewModel> Cards { get; set; } = Array.Empty<PlatformAdminLinkCardViewModel>();
}
@@ -0,0 +1,13 @@
namespace PowderCoating.Web.ViewModels.PlatformAdmin;
public class PlatformAdminLinkCardViewModel
{
public string Title { get; set; } = string.Empty;
public string Description { get; set; } = string.Empty;
public string Controller { get; set; } = string.Empty;
public string Action { get; set; } = string.Empty;
public string Icon { get; set; } = string.Empty;
public string BadgeText { get; set; } = string.Empty;
public string BadgeStyle { get; set; } = string.Empty;
public Dictionary<string, string>? RouteValues { get; set; }
}
@@ -251,6 +251,21 @@
<a asp-controller="Companies" asp-action="Create" class="btn btn-sm btn-outline-primary">
<i class="bi bi-plus-circle me-1"></i>New Company
</a>
<a asp-controller="PlatformAdmin" asp-action="TenantsBilling" class="btn btn-sm btn-outline-secondary">
<i class="bi bi-building-gear me-1"></i>Tenants &amp; Billing
</a>
<a asp-controller="PlatformAdmin" asp-action="PeopleActivity" class="btn btn-sm btn-outline-secondary">
<i class="bi bi-people me-1"></i>People &amp; Activity
</a>
<a asp-controller="PlatformAdmin" asp-action="ContentMessaging" class="btn btn-sm btn-outline-secondary">
<i class="bi bi-megaphone me-1"></i>Content &amp; Messaging
</a>
<a asp-controller="PlatformAdmin" asp-action="Observability" class="btn btn-sm btn-outline-secondary">
<i class="bi bi-binoculars me-1"></i>Observability
</a>
<a asp-controller="PlatformAdmin" asp-action="Maintenance" class="btn btn-sm btn-outline-secondary">
<i class="bi bi-wrench-adjustable me-1"></i>Maintenance
</a>
<a asp-controller="PlatformUsers" asp-action="Index" class="btn btn-sm btn-outline-secondary">
<i class="bi bi-people me-1"></i>Platform Users
</a>
@@ -263,7 +278,7 @@
@if (string.IsNullOrEmpty(Environment.GetEnvironmentVariable("WEBSITE_SITE_NAME")))
{
<a asp-controller="Diagnostics" asp-action="ViewLogs" class="btn btn-sm btn-outline-secondary">
<i class="bi bi-file-text me-1"></i>View Logs
<i class="bi bi-file-text me-1"></i>Raw Logs
</a>
}
<a asp-controller="SeedData" asp-action="Index" class="btn btn-sm btn-outline-secondary">
@@ -0,0 +1,9 @@
@using PowderCoating.Web.ViewModels.PlatformAdmin
@model PlatformAdminHubViewModel
@{
ViewData["Title"] = Model.Title;
ViewData["PageIcon"] = Model.PageIcon;
}
@await Html.PartialAsync("_HubCardGrid", Model)
@@ -0,0 +1,9 @@
@using PowderCoating.Web.ViewModels.PlatformAdmin
@model PlatformAdminHubViewModel
@{
ViewData["Title"] = Model.Title;
ViewData["PageIcon"] = Model.PageIcon;
}
@await Html.PartialAsync("_HubCardGrid", Model)
@@ -0,0 +1,9 @@
@using PowderCoating.Web.ViewModels.PlatformAdmin
@model PlatformAdminHubViewModel
@{
ViewData["Title"] = Model.Title;
ViewData["PageIcon"] = Model.PageIcon;
}
@await Html.PartialAsync("_HubCardGrid", Model)
@@ -0,0 +1,9 @@
@using PowderCoating.Web.ViewModels.PlatformAdmin
@model PlatformAdminHubViewModel
@{
ViewData["Title"] = Model.Title;
ViewData["PageIcon"] = Model.PageIcon;
}
@await Html.PartialAsync("_HubCardGrid", Model)
@@ -0,0 +1,9 @@
@using PowderCoating.Web.ViewModels.PlatformAdmin
@model PlatformAdminHubViewModel
@{
ViewData["Title"] = Model.Title;
ViewData["PageIcon"] = Model.PageIcon;
}
@await Html.PartialAsync("_HubCardGrid", Model)
@@ -0,0 +1,138 @@
@using PowderCoating.Web.ViewModels.PlatformAdmin
@model PlatformAdminHubViewModel
@{
bool hasWarning = !string.IsNullOrWhiteSpace(Model.WarningTitle) || !string.IsNullOrWhiteSpace(Model.WarningMessage);
}
<style>
.platform-admin-hub {
max-width: 1240px;
}
.platform-admin-intro {
max-width: 820px;
}
.platform-admin-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(260px, 1fr));
gap: 1rem;
}
.platform-admin-card {
display: flex;
flex-direction: column;
gap: 0.9rem;
min-height: 100%;
background: var(--bs-body-bg);
border: 1px solid var(--bs-border-color);
border-radius: 0.95rem;
padding: 1.2rem;
text-decoration: none;
color: inherit;
transition: border-color 0.15s, box-shadow 0.15s, transform 0.15s;
}
.platform-admin-card:hover {
border-color: var(--bs-primary);
box-shadow: 0 6px 18px rgba(13, 110, 253, 0.12);
transform: translateY(-2px);
color: inherit;
}
.platform-admin-card-head {
display: flex;
align-items: flex-start;
justify-content: space-between;
gap: 0.75rem;
}
.platform-admin-card-icon {
width: 46px;
height: 46px;
border-radius: 0.9rem;
display: flex;
align-items: center;
justify-content: center;
flex-shrink: 0;
background: rgba(13, 110, 253, 0.12);
color: #0d6efd;
font-size: 1.25rem;
}
.platform-admin-card-title {
font-size: 0.98rem;
font-weight: 700;
margin: 0;
}
.platform-admin-card-copy {
color: var(--bs-secondary-color);
font-size: 0.84rem;
line-height: 1.5;
margin: 0;
flex: 1;
}
.platform-admin-card-arrow {
display: inline-flex;
align-items: center;
gap: 0.35rem;
font-size: 0.8rem;
font-weight: 600;
color: var(--bs-primary);
}
</style>
<div class="platform-admin-hub container-fluid py-2">
<div class="mb-4">
<h4 class="fw-bold mb-2">
<i class="bi @Model.PageIcon me-2 text-primary"></i>@Model.Title
</h4>
<p class="platform-admin-intro text-muted mb-0">@Model.Intro</p>
</div>
@if (hasWarning)
{
<div class="alert alert-warning alert-permanent border mb-4">
@if (!string.IsNullOrWhiteSpace(Model.WarningTitle))
{
<div class="fw-semibold mb-1">
<i class="bi bi-exclamation-triangle me-2"></i>@Model.WarningTitle
</div>
}
@if (!string.IsNullOrWhiteSpace(Model.WarningMessage))
{
<div class="small">@Model.WarningMessage</div>
}
</div>
}
<div class="platform-admin-grid">
@foreach (var card in Model.Cards)
{
<a asp-controller="@card.Controller"
asp-action="@card.Action"
asp-all-route-data="@card.RouteValues"
class="platform-admin-card">
<div class="platform-admin-card-head">
<div class="platform-admin-card-icon">
<i class="bi @card.Icon"></i>
</div>
@if (!string.IsNullOrWhiteSpace(card.BadgeText))
{
<span class="badge @card.BadgeStyle">@card.BadgeText</span>
}
</div>
<div>
<h5 class="platform-admin-card-title">@card.Title</h5>
</div>
<p class="platform-admin-card-copy">@card.Description</p>
<div class="platform-admin-card-arrow">
Open area <i class="bi bi-arrow-right"></i>
</div>
</a>
}
</div>
</div>
+20 -127
View File
@@ -1281,140 +1281,37 @@
@* Multi-tenancy: SuperAdmin Platform Management (hidden while impersonating) *@
@if (User.IsInRole("SuperAdmin") && !isImpersonating)
{
<div class="nav-section-title">Tenants &amp; Billing</div>
<a asp-controller="Companies" asp-action="Index" class="nav-link">
<i class="bi bi-building"></i>
<span>Companies</span>
</a>
<a asp-controller="CompanyHealth" asp-action="Index" class="nav-link">
<i class="bi bi-heart-pulse"></i>
<span>Company Health</span>
</a>
<a asp-controller="SubscriptionManagement" asp-action="Index" class="nav-link">
<i class="bi bi-credit-card"></i>
<span>Subscriptions</span>
</a>
<a asp-controller="PlatformSubscription" asp-action="Index" class="nav-link">
<i class="bi bi-layers"></i>
<span>Subscription Plans</span>
</a>
<a asp-controller="Revenue" asp-action="Index" class="nav-link">
<i class="bi bi-graph-up-arrow"></i>
<span>Revenue Dashboard</span>
</a>
<a asp-controller="StripeEvents" asp-action="Index" class="nav-link">
<i class="bi bi-lightning-charge"></i>
<span>Stripe Events</span>
</a>
<a asp-controller="SmsAgreements" asp-action="Index" class="nav-link">
<i class="bi bi-file-earmark-check"></i>
<span>SMS Agreements</span>
</a>
<div class="nav-section-title">Content &amp; Communication</div>
<a asp-controller="Announcements" asp-action="Index" class="nav-link">
<i class="bi bi-megaphone"></i>
<span>Announcements</span>
</a>
<a asp-controller="DashboardTips" asp-action="Index" class="nav-link">
<i class="bi bi-lightbulb"></i>
<span>Dashboard Tips</span>
</a>
<a asp-controller="EmailBroadcast" asp-action="Index" class="nav-link">
<i class="bi bi-broadcast"></i>
<span>Email Broadcast</span>
</a>
<div class="nav-section-title">Users &amp; Activity</div>
<a asp-controller="OnboardingProgress" asp-action="Index" class="nav-link">
<i class="bi bi-rocket-takeoff"></i>
<span>Onboarding Progress</span>
</a>
<a asp-controller="PlatformUsers" asp-action="Index" class="nav-link">
<i class="bi bi-people-fill"></i>
<span>Platform Users</span>
</a>
<a asp-controller="UserActivity" asp-action="Index" class="nav-link">
<i class="bi bi-person-lines-fill"></i>
<span>User Activity</span>
</a>
<a asp-controller="UserActivity" asp-action="Online" class="nav-link d-flex align-items-center justify-content-between">
<span><i class="bi bi-circle-fill me-2" style="color:#22c55e;font-size:.55rem;vertical-align:middle;"></i>Online Now</span>
@{ var _onlineCount = OnlineUserTracker.GetActiveCount(15); }
@if (_onlineCount > 0)
{
<span class="badge rounded-pill" style="background:#22c55e;color:#fff;font-size:.7rem;">@_onlineCount</span>
}
</a>
<a asp-controller="PlatformNotifications" asp-action="Index" class="nav-link">
<i class="bi bi-bell"></i>
<span>Notification Log</span>
</a>
<div class="nav-section-title">AI &amp; Usage</div>
<a asp-controller="AiUsageReport" asp-action="Index" class="nav-link">
<i class="bi bi-robot"></i>
<span>AI Usage</span>
</a>
<a asp-controller="UsageQuota" asp-action="Index" class="nav-link">
<i class="bi bi-speedometer2"></i>
<span>Usage &amp; Quota</span>
</a>
<div class="nav-section-title">Logs &amp; Support</div>
<a asp-controller="Contact" asp-action="Submissions" class="nav-link">
<i class="bi bi-envelope"></i>
<span>Contact Submissions</span>
</a>
<a asp-controller="BugReport" asp-action="Index" class="nav-link">
<i class="bi bi-bug"></i>
<span>Bug Reports</span>
</a>
<a asp-controller="AuditLog" asp-action="Index" class="nav-link">
<div class="nav-section-title">Platform Admin</div>
<a asp-controller="Dashboard" asp-action="SuperAdminDashboard" class="nav-link">
<i class="bi bi-shield-check"></i>
<span>Audit Log</span>
<span>Platform Overview</span>
</a>
<a asp-controller="BannedIps" asp-action="Index" class="nav-link">
<i class="bi bi-slash-circle"></i>
<span>Banned IPs</span>
<a asp-controller="PlatformAdmin" asp-action="TenantsBilling" class="nav-link">
<i class="bi bi-building-gear"></i>
<span>Tenants &amp; Billing</span>
</a>
<a asp-controller="SystemLogs" asp-action="Index" class="nav-link">
<i class="bi bi-database-exclamation"></i>
<span>System Logs</span>
<a asp-controller="PlatformAdmin" asp-action="PeopleActivity" class="nav-link">
<i class="bi bi-people"></i>
<span>People &amp; Activity</span>
</a>
@if (string.IsNullOrEmpty(Environment.GetEnvironmentVariable("WEBSITE_SITE_NAME")))
{
<a asp-controller="Diagnostics" asp-action="ViewLogs" class="nav-link">
<i class="bi bi-file-text"></i>
<span>Raw Log Files</span>
<a asp-controller="PlatformAdmin" asp-action="ContentMessaging" class="nav-link">
<i class="bi bi-megaphone"></i>
<span>Content &amp; Messaging</span>
</a>
}
<a asp-controller="SystemInfo" asp-action="Index" class="nav-link">
<i class="bi bi-cpu"></i>
<span>System Info</span>
<a asp-controller="PlatformAdmin" asp-action="Observability" class="nav-link">
<i class="bi bi-binoculars"></i>
<span>Observability</span>
</a>
<a asp-controller="PlatformAdmin" asp-action="Maintenance" class="nav-link">
<i class="bi bi-wrench-adjustable-circle"></i>
<span>Maintenance</span>
</a>
<div class="nav-section-title">Data &amp; Tools</div>
<a asp-controller="DataExport" asp-action="Index" class="nav-link">
<i class="bi bi-file-earmark-arrow-down"></i>
<span>Data Export</span>
</a>
<a asp-controller="DataPurge" asp-action="Index" class="nav-link">
<i class="bi bi-trash3"></i>
<span>Data Purge</span>
</a>
<a asp-controller="StorageMigration" asp-action="Index" class="nav-link">
<i class="bi bi-cloud-upload"></i>
<span>Storage Migration</span>
</a>
<div class="nav-section-title">Platform Configuration</div>
<a asp-controller="PlatformSettings" asp-action="Index" class="nav-link">
<i class="bi bi-sliders"></i>
<span>Platform Settings</span>
</a>
<a asp-controller="SeedData" asp-action="Index" class="nav-link">
<i class="bi bi-database-fill-gear"></i>
<span>Seed Data</span>
</a>
<a asp-controller="PowderCatalog" asp-action="Index" class="nav-link">
<i class="bi bi-palette2"></i>
<span>Powder Catalog</span>
@@ -1423,10 +1320,6 @@
<i class="bi bi-link-45deg"></i>
<span>Manufacturer Lookup Patterns</span>
</a>
<a asp-controller="ReleaseNotes" asp-action="Manage" class="nav-link">
<i class="bi bi-journal-text"></i>
<span>Release Notes</span>
</a>
}
}