diff --git a/CLAUDE.md b/CLAUDE.md index afb6b8f..8c8a147 100644 --- a/CLAUDE.md +++ b/CLAUDE.md @@ -6,6 +6,10 @@ This file provides guidance to Claude Code (claude.ai/code) when working with co PlanTempus is a .NET 9 web application built with ASP.NET Core Razor Pages. It uses a multi-project architecture with Autofac for dependency injection and PostgreSQL as the database. +## Related Projects + +- **calpoc** = Calendar POC projekt located at `../Calendar` (TypeScript calendar component with offline-first architecture, drag-and-drop, NovaDI, EventBus). When user mentions "calpoc", refer to this folder. + ## Build and Development Commands ### Prerequisites diff --git a/Core/SeqLogging/SeqBackgroundService.cs b/Core/SeqLogging/SeqBackgroundService.cs deleted file mode 100644 index cef2cc1..0000000 --- a/Core/SeqLogging/SeqBackgroundService.cs +++ /dev/null @@ -1,88 +0,0 @@ -using Microsoft.ApplicationInsights; -using Microsoft.ApplicationInsights.Channel; -using Microsoft.ApplicationInsights.DataContracts; -using Microsoft.Extensions.Hosting; -using PlanTempus.Core.Telemetry; - -namespace PlanTempus.Core.SeqLogging -{ - public class SeqBackgroundService : BackgroundService - { - private readonly IMessageChannel _messageChannel; - private readonly TelemetryClient _telemetryClient; - private readonly SeqLogger _seqLogger; - - public SeqBackgroundService(TelemetryClient telemetryClient, - IMessageChannel messageChannel, - SeqLogger seqlogger) - { - _telemetryClient = telemetryClient; - _messageChannel = messageChannel; - _seqLogger = seqlogger; - - _telemetryClient.TrackTrace("SeqBackgroundService started"); - - } - - protected override async Task ExecuteAsync(CancellationToken stoppingToken) - { - try - { - while (!stoppingToken.IsCancellationRequested) - await foreach (var telemetry in _messageChannel.Reader.ReadAllAsync(stoppingToken)) - { - try - { - switch (telemetry) - { - case ExceptionTelemetry et: - await _seqLogger.LogAsync(et); - break; - - case TraceTelemetry et: - await _seqLogger.LogAsync(et); - break; - - case DependencyTelemetry et: - await _seqLogger.LogAsync(et); - break; - - case RequestTelemetry et: - await _seqLogger.LogAsync(et); - break; - - case EventTelemetry et: - await _seqLogger.LogAsync(et); - break; - - - default: - throw new NotSupportedException(telemetry.GetType().Name); - } - } - catch - { - throw; - //_telemetryClient.TrackException(ex); this is disabled for now, we need to think about the channel structure first - } - } - } - catch (Exception ex) - { - if (ex is not OperationCanceledException) - { - _telemetryClient.TrackException(ex); - throw; - } - - } - } - - public override async Task StopAsync(CancellationToken cancellationToken) - { - _telemetryClient.TrackTrace("StopAsync called: Service shutdown started"); - _messageChannel.Dispose(); - await base.StopAsync(cancellationToken); - } - } -} diff --git a/PTWork.code-workspace b/PTWork.code-workspace new file mode 100644 index 0000000..2ac8eb2 --- /dev/null +++ b/PTWork.code-workspace @@ -0,0 +1,13 @@ +{ + "folders": [ + { + "path": "." + }, + { + "path": "../Calendar" + } + ], + "settings": { + "liveServer.settings.port": 5501 + } +} \ No newline at end of file diff --git a/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml b/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml new file mode 100644 index 0000000..9e3b36c --- /dev/null +++ b/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml @@ -0,0 +1,94 @@ +@page "/" +@using PlanTempus.Application.Features.Dashboard.Pages +@model PlanTempus.Application.Features.Dashboard.Pages.IndexModel +@{ + ViewData["Title"] = "Dashboard"; +} + + + + + + 12 + Bookinger i dag + + + 4 gennemført, 2 i gang + + + + 8.450 kr + Forventet omsætning + + + +12% vs. gennemsnit + + + + 78% + Belægningsgrad + + + God kapacitet + + + + 4 + Kræver opmærksomhed + + + + + + + + + + + + AI Analyse + + + Godt i gang! 4 af 12 bookinger er gennemført. 2 er i gang nu, og 6 venter. + Forventet omsætning: 8.450 kr – allerede realiseret 2.150 kr. + + + + + + + + + + Dagens bookinger + + Se alle + + +

Booking oversigt kommer her...

+
+
+
+ + + + + + Hurtige handlinger + + + + + + Ny booking + + + + Ny kunde + + + + + +
+
diff --git a/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml.cs b/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml.cs new file mode 100644 index 0000000..c822af3 --- /dev/null +++ b/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml.cs @@ -0,0 +1,10 @@ +using Microsoft.AspNetCore.Mvc.RazorPages; + +namespace PlanTempus.Application.Features.Dashboard.Pages; + +public class IndexModel : PageModel +{ + public void OnGet() + { + } +} diff --git a/PlanTempus.Application/Features/Localization/Models/SupportedCulture.cs b/PlanTempus.Application/Features/Localization/Models/SupportedCulture.cs new file mode 100644 index 0000000..da758e7 --- /dev/null +++ b/PlanTempus.Application/Features/Localization/Models/SupportedCulture.cs @@ -0,0 +1,8 @@ +namespace PlanTempus.Application.Features.Localization.Models; + +public class SupportedCulture +{ + public required string Code { get; set; } + public required string Name { get; set; } + public required string NativeName { get; set; } +} diff --git a/PlanTempus.Application/Features/Localization/Services/ILocalizationService.cs b/PlanTempus.Application/Features/Localization/Services/ILocalizationService.cs new file mode 100644 index 0000000..c689ec8 --- /dev/null +++ b/PlanTempus.Application/Features/Localization/Services/ILocalizationService.cs @@ -0,0 +1,10 @@ +using PlanTempus.Application.Features.Localization.Models; + +namespace PlanTempus.Application.Features.Localization.Services; + +public interface ILocalizationService +{ + string Get(string key, string? culture = null); + string CurrentCulture { get; } + IEnumerable GetSupportedCultures(); +} diff --git a/PlanTempus.Application/Features/Localization/Services/JsonLocalizationService.cs b/PlanTempus.Application/Features/Localization/Services/JsonLocalizationService.cs new file mode 100644 index 0000000..0c96748 --- /dev/null +++ b/PlanTempus.Application/Features/Localization/Services/JsonLocalizationService.cs @@ -0,0 +1,46 @@ +using System.Text.Json; +using PlanTempus.Application.Features.Localization.Models; + +namespace PlanTempus.Application.Features.Localization.Services; + +public class JsonLocalizationService : ILocalizationService +{ + private readonly string _translationsPath; + + public JsonLocalizationService(IWebHostEnvironment env) + { + _translationsPath = Path.Combine(env.ContentRootPath, "Features", "Localization", "Translations"); + } + + public string CurrentCulture => "da"; + + public string Get(string key, string? culture = null) + { + culture ??= CurrentCulture; + var filePath = Path.Combine(_translationsPath, $"{culture}.json"); + + if (!File.Exists(filePath)) + return key; + + var json = File.ReadAllText(filePath); + var doc = JsonDocument.Parse(json); + + var parts = key.Split('.'); + JsonElement current = doc.RootElement; + + foreach (var part in parts) + if (!current.TryGetProperty(part, out current)) + return key; + + return current.GetString() ?? key; + } + + public IEnumerable GetSupportedCultures() + { + return new List + { + new() { Code = "da", Name = "Danish", NativeName = "Dansk" }, + new() { Code = "en", Name = "English", NativeName = "English" } + }; + } +} diff --git a/PlanTempus.Application/Features/Localization/Translations/da.json b/PlanTempus.Application/Features/Localization/Translations/da.json new file mode 100644 index 0000000..c569ae6 --- /dev/null +++ b/PlanTempus.Application/Features/Localization/Translations/da.json @@ -0,0 +1,33 @@ +{ + "menu": { + "home": "Dashboard", + "calendar": "Kalender", + "pos": "Kasse", + "products": "Produkter & Lager", + "suppliers": "Leverandører", + "customers": "Kunder", + "employees": "Medarbejdere", + "reports": "Statistik & Rapporter", + "settings": "Indstillinger", + "account": "Abonnement & Konto" + }, + "groups": { + "dashboard": "Dashboard", + "data": "Data", + "analytics": "Analyse", + "system": "System" + }, + "common": { + "save": "Gem", + "cancel": "Annuller", + "search": "Søg", + "close": "Luk", + "delete": "Slet", + "edit": "Rediger", + "add": "Tilføj" + }, + "sidebar": { + "lockScreen": "Lås skærm", + "appName": "Salon OS" + } +} diff --git a/PlanTempus.Application/Features/Localization/Translations/en.json b/PlanTempus.Application/Features/Localization/Translations/en.json new file mode 100644 index 0000000..a66a7b7 --- /dev/null +++ b/PlanTempus.Application/Features/Localization/Translations/en.json @@ -0,0 +1,33 @@ +{ + "menu": { + "home": "Dashboard", + "calendar": "Calendar", + "pos": "Point of Sale", + "products": "Products & Inventory", + "suppliers": "Suppliers", + "customers": "Customers", + "employees": "Employees", + "reports": "Statistics & Reports", + "settings": "Settings", + "account": "Subscription & Account" + }, + "groups": { + "dashboard": "Dashboard", + "data": "Data", + "analytics": "Analytics", + "system": "System" + }, + "common": { + "save": "Save", + "cancel": "Cancel", + "search": "Search", + "close": "Close", + "delete": "Delete", + "edit": "Edit", + "add": "Add" + }, + "sidebar": { + "lockScreen": "Lock screen", + "appName": "Salon OS" + } +} diff --git a/PlanTempus.Application/Features/Menu/Models/MenuGroup.cs b/PlanTempus.Application/Features/Menu/Models/MenuGroup.cs new file mode 100644 index 0000000..2b68088 --- /dev/null +++ b/PlanTempus.Application/Features/Menu/Models/MenuGroup.cs @@ -0,0 +1,12 @@ +namespace PlanTempus.Application.Features.Menu.Models; + +/// +/// Represents a group of menu items (e.g., "Dashboard", "Data", "System"). +/// +public class MenuGroup +{ + public required string Id { get; set; } + public required string Label { get; set; } + public int SortOrder { get; set; } + public List Items { get; set; } = new(); +} diff --git a/PlanTempus.Application/Features/Menu/Models/MenuItem.cs b/PlanTempus.Application/Features/Menu/Models/MenuItem.cs new file mode 100644 index 0000000..6996726 --- /dev/null +++ b/PlanTempus.Application/Features/Menu/Models/MenuItem.cs @@ -0,0 +1,15 @@ +namespace PlanTempus.Application.Features.Menu.Models; + +/// +/// Represents a single menu item in the sidebar. +/// +public class MenuItem +{ + public required string Id { get; set; } + public required string Label { get; set; } + public required string Icon { get; set; } + public required string Url { get; set; } + public UserRole MinimumRole { get; set; } = UserRole.Staff; + public int SortOrder { get; set; } + public bool IsActive { get; set; } +} diff --git a/PlanTempus.Application/Features/Menu/Models/UserRole.cs b/PlanTempus.Application/Features/Menu/Models/UserRole.cs new file mode 100644 index 0000000..44d5aea --- /dev/null +++ b/PlanTempus.Application/Features/Menu/Models/UserRole.cs @@ -0,0 +1,11 @@ +namespace PlanTempus.Application.Features.Menu.Models; + +/// +/// User roles for menu visibility. Higher value = more access. +/// +public enum UserRole +{ + Staff = 0, + Manager = 1, + Admin = 2 +} diff --git a/PlanTempus.Application/Features/Menu/Services/IMenuService.cs b/PlanTempus.Application/Features/Menu/Services/IMenuService.cs new file mode 100644 index 0000000..a69c850 --- /dev/null +++ b/PlanTempus.Application/Features/Menu/Services/IMenuService.cs @@ -0,0 +1,14 @@ +using PlanTempus.Application.Features.Menu.Models; + +namespace PlanTempus.Application.Features.Menu.Services; + +/// +/// Service for retrieving menu structure based on user role. +/// +public interface IMenuService +{ + /// + /// Get menu groups filtered by user role. + /// + List GetMenuForRole(UserRole role, string? currentUrl = null); +} diff --git a/PlanTempus.Application/Features/Menu/Services/MockMenuService.cs b/PlanTempus.Application/Features/Menu/Services/MockMenuService.cs new file mode 100644 index 0000000..d42f628 --- /dev/null +++ b/PlanTempus.Application/Features/Menu/Services/MockMenuService.cs @@ -0,0 +1,187 @@ +using PlanTempus.Application.Features.Localization.Services; +using PlanTempus.Application.Features.Menu.Models; + +namespace PlanTempus.Application.Features.Menu.Services; + +/// +/// Mock implementation of IMenuService with hardcoded menu data. +/// +public class MockMenuService : IMenuService +{ + private readonly ILocalizationService _localize; + + public MockMenuService(ILocalizationService localize) + { + _localize = localize; + } + + public List GetMenuForRole(UserRole role, string? currentUrl = null) + { + var allGroups = GetAllMenuGroups(); + + return allGroups + .Select(g => new MenuGroup + { + Id = g.Id, + Label = _localize.Get($"groups.{g.Id}"), + SortOrder = g.SortOrder, + Items = g.Items + .Where(i => role >= i.MinimumRole) + .Select(i => new MenuItem + { + Id = i.Id, + Label = _localize.Get($"menu.{i.Id}"), + Icon = i.Icon, + Url = i.Url, + MinimumRole = i.MinimumRole, + SortOrder = i.SortOrder, + IsActive = currentUrl != null && i.Url.Equals(currentUrl, StringComparison.OrdinalIgnoreCase) + }) + .OrderBy(i => i.SortOrder) + .ToList() + }) + .Where(g => g.Items.Any()) + .OrderBy(g => g.SortOrder) + .ToList(); + } + + private static List GetAllMenuGroups() + { + return new List + { + // DASHBOARD GROUP + new MenuGroup + { + Id = "dashboard", + Label = "Dashboard", + SortOrder = 1, + Items = new List + { + new MenuItem + { + Id = "home", + Label = "Dashboard", + Icon = "ph-squares-four", + Url = "/", + MinimumRole = UserRole.Staff, + SortOrder = 1 + }, + new MenuItem + { + Id = "calendar", + Label = "Kalender", + Icon = "ph-calendar", + Url = "/poc-calendar.html", + MinimumRole = UserRole.Staff, + SortOrder = 2 + }, + new MenuItem + { + Id = "pos", + Label = "Kasse", + Icon = "ph-device-mobile", + Url = "/pos", + MinimumRole = UserRole.Staff, + SortOrder = 3 + } + } + }, + + // DATA GROUP + new MenuGroup + { + Id = "data", + Label = "Data", + SortOrder = 2, + Items = new List + { + new MenuItem + { + Id = "products", + Label = "Produkter & Lager", + Icon = "ph-package", + Url = "/poc-produkter.html", + MinimumRole = UserRole.Manager, + SortOrder = 1 + }, + new MenuItem + { + Id = "suppliers", + Label = "Leverandører", + Icon = "ph-truck", + Url = "/poc-leverandoerer.html", + MinimumRole = UserRole.Manager, + SortOrder = 2 + }, + new MenuItem + { + Id = "customers", + Label = "Kunder", + Icon = "ph-users", + Url = "/customers", + MinimumRole = UserRole.Staff, + SortOrder = 3 + }, + new MenuItem + { + Id = "employees", + Label = "Medarbejdere", + Icon = "ph-user", + Url = "/poc-medarbejdere.html", + MinimumRole = UserRole.Manager, + SortOrder = 4 + } + } + }, + + // ANALYSE GROUP + new MenuGroup + { + Id = "analytics", + Label = "Analyse", + SortOrder = 3, + Items = new List + { + new MenuItem + { + Id = "reports", + Label = "Statistik & Rapporter", + Icon = "ph-chart-bar", + Url = "/reports", + MinimumRole = UserRole.Manager, + SortOrder = 1 + } + } + }, + + // SYSTEM GROUP + new MenuGroup + { + Id = "system", + Label = "System", + SortOrder = 4, + Items = new List + { + new MenuItem + { + Id = "settings", + Label = "Indstillinger", + Icon = "ph-gear", + Url = "/poc-indstillinger.html", + MinimumRole = UserRole.Admin, + SortOrder = 1 + }, + new MenuItem + { + Id = "account", + Label = "Abonnement & Konto", + Icon = "ph-credit-card", + Url = "/poc-konto.html", + MinimumRole = UserRole.Admin, + SortOrder = 2 + } + } + } + }; + } +} diff --git a/PlanTempus.Application/Features/Menu/SideMenuViewComponent.cs b/PlanTempus.Application/Features/Menu/SideMenuViewComponent.cs new file mode 100644 index 0000000..5c3c75e --- /dev/null +++ b/PlanTempus.Application/Features/Menu/SideMenuViewComponent.cs @@ -0,0 +1,36 @@ +using Microsoft.AspNetCore.Mvc; +using PlanTempus.Application.Features.Menu.Services; +using PlanTempus.Application.Features.Menu; +using PlanTempus.Application.Features.Menu.Models; + +namespace PlanTempus.Application.Features.Menu; + +/// +/// ViewComponent for rendering the side menu based on user role. +/// +public class SideMenuViewComponent : ViewComponent +{ + private readonly IMenuService _menuService; + + public SideMenuViewComponent(IMenuService menuService) + { + _menuService = menuService; + } + + public IViewComponentResult Invoke(UserRole? role = null) + { + // Default to Admin for demo (in real app, get from auth) + var userRole = role ?? UserRole.Admin; + + var currentUrl = HttpContext.Request.Path.Value; + var groups = _menuService.GetMenuForRole(userRole, currentUrl); + + var viewModel = new SideMenuViewModel + { + Groups = groups, + CurrentUserRole = userRole + }; + + return View(viewModel); + } +} diff --git a/PlanTempus.Application/Features/Menu/SideMenuViewModel.cs b/PlanTempus.Application/Features/Menu/SideMenuViewModel.cs new file mode 100644 index 0000000..1a5c408 --- /dev/null +++ b/PlanTempus.Application/Features/Menu/SideMenuViewModel.cs @@ -0,0 +1,12 @@ +using PlanTempus.Application.Features.Menu.Models; + +namespace PlanTempus.Application.Features.Menu; + +/// +/// ViewModel for the side menu partial view. +/// +public class SideMenuViewModel +{ + public required List Groups { get; set; } + public UserRole CurrentUserRole { get; set; } +} diff --git a/PlanTempus.Application/Features/Shared/Components/SideMenu/Default.cshtml b/PlanTempus.Application/Features/Shared/Components/SideMenu/Default.cshtml new file mode 100644 index 0000000..2514a4b --- /dev/null +++ b/PlanTempus.Application/Features/Shared/Components/SideMenu/Default.cshtml @@ -0,0 +1,40 @@ +@using PlanTempus.Application.Features.Menu +@model SideMenuViewModel + + + + + Salon OS + + + + + + + @foreach (var group in Model.Groups) + { + + @group.Label + @foreach (var item in group.Items) + { + + + @item.Label + + } + + } + + + + + + Lås skærm + + + + + + diff --git a/PlanTempus.Application/Features/Shared/_ProfileDrawer.cshtml b/PlanTempus.Application/Features/Shared/_ProfileDrawer.cshtml new file mode 100644 index 0000000..9e42620 --- /dev/null +++ b/PlanTempus.Application/Features/Shared/_ProfileDrawer.cshtml @@ -0,0 +1,49 @@ + + + Profil + + + + + + + + MJ + Maria Jensen + maria@salon.dk + + + + + + + + Min profil + + + + Indstillinger + + + + + + + + + Mørk tilstand + + + + + + + + + + + + Log ud + + + diff --git a/PlanTempus.Application/Features/Shared/_TopBar.cshtml b/PlanTempus.Application/Features/Shared/_TopBar.cshtml new file mode 100644 index 0000000..41b140a --- /dev/null +++ b/PlanTempus.Application/Features/Shared/_TopBar.cshtml @@ -0,0 +1,26 @@ + + + + + ⌘K + + + + + + + 3 + + + + + + + MJ + + Maria Jensen + Administrator + + + + diff --git a/PlanTempus.Application/Features/_Shared/Components/SideMenu/Default.cshtml b/PlanTempus.Application/Features/_Shared/Components/SideMenu/Default.cshtml new file mode 100644 index 0000000..ae7de86 --- /dev/null +++ b/PlanTempus.Application/Features/_Shared/Components/SideMenu/Default.cshtml @@ -0,0 +1,39 @@ +@model PlanTempus.Application.Features.Menu.SideMenuViewModel + + + + + Salon OS + + + + + + + @foreach (var group in Model.Groups) + { + + @group.Label + @foreach (var item in group.Items) + { + + + @item.Label + + } + + } + + + + + + Lås skærm + + + + + + diff --git a/PlanTempus.Application/Features/_Shared/Pages/_Layout.cshtml b/PlanTempus.Application/Features/_Shared/Pages/_Layout.cshtml new file mode 100644 index 0000000..e8bb2e3 --- /dev/null +++ b/PlanTempus.Application/Features/_Shared/Pages/_Layout.cshtml @@ -0,0 +1,38 @@ + + + + + + @ViewData["Title"] - Salon OS + + + + + + + + + + + + + + @await RenderSectionAsync("Styles", required: false) + + + + @await Component.InvokeAsync("SideMenu") + + + + @RenderBody() + + + + + + + + @await RenderSectionAsync("Scripts", required: false) + + diff --git a/PlanTempus.Application/Features/_Shared/Pages/_ProfileDrawer.cshtml b/PlanTempus.Application/Features/_Shared/Pages/_ProfileDrawer.cshtml new file mode 100644 index 0000000..9e42620 --- /dev/null +++ b/PlanTempus.Application/Features/_Shared/Pages/_ProfileDrawer.cshtml @@ -0,0 +1,49 @@ + + + Profil + + + + + + + + MJ + Maria Jensen + maria@salon.dk + + + + + + + + Min profil + + + + Indstillinger + + + + + + + + + Mørk tilstand + + + + + + + + + + + + Log ud + + + diff --git a/PlanTempus.Application/Features/_Shared/Pages/_TopBar.cshtml b/PlanTempus.Application/Features/_Shared/Pages/_TopBar.cshtml new file mode 100644 index 0000000..41b140a --- /dev/null +++ b/PlanTempus.Application/Features/_Shared/Pages/_TopBar.cshtml @@ -0,0 +1,26 @@ + + + + + ⌘K + + + + + + + 3 + + + + + + + MJ + + Maria Jensen + Administrator + + + + diff --git a/PlanTempus.Application/Features/_Shared/Pages/_ViewImports.cshtml b/PlanTempus.Application/Features/_Shared/Pages/_ViewImports.cshtml new file mode 100644 index 0000000..bfd56f8 --- /dev/null +++ b/PlanTempus.Application/Features/_Shared/Pages/_ViewImports.cshtml @@ -0,0 +1,4 @@ +@using PlanTempus.Application +@namespace PlanTempus.Application.Features +@addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers +@addTagHelper *, PlanTempus.Application diff --git a/PlanTempus.Application/Features/_Shared/Pages/_ViewStart.cshtml b/PlanTempus.Application/Features/_Shared/Pages/_ViewStart.cshtml new file mode 100644 index 0000000..f9f7d10 --- /dev/null +++ b/PlanTempus.Application/Features/_Shared/Pages/_ViewStart.cshtml @@ -0,0 +1,3 @@ +@{ + Layout = "/Features/_Shared/Pages/_Layout.cshtml"; +} diff --git a/PlanTempus.Application/Features/_Shared/TagHelpers/LocalizeTagHelper.cs b/PlanTempus.Application/Features/_Shared/TagHelpers/LocalizeTagHelper.cs new file mode 100644 index 0000000..9e77cf0 --- /dev/null +++ b/PlanTempus.Application/Features/_Shared/TagHelpers/LocalizeTagHelper.cs @@ -0,0 +1,28 @@ +using Microsoft.AspNetCore.Razor.TagHelpers; +using PlanTempus.Application.Features.Localization.Services; + +namespace PlanTempus.Application.Features._Shared.TagHelpers; + +[HtmlTargetElement(Attributes = "localize")] +public class LocalizeTagHelper : TagHelper +{ + private readonly ILocalizationService _localize; + + public LocalizeTagHelper(ILocalizationService localize) + { + _localize = localize; + } + + [HtmlAttributeName("localize")] + public string Key { get; set; } = string.Empty; + + public override void Process(TagHelperContext context, TagHelperOutput output) + { + var translated = _localize.Get(Key); + + if (!string.IsNullOrEmpty(translated) && translated != Key) + output.Content.SetContent(translated); + + output.Attributes.RemoveAll("localize"); + } +} diff --git a/PlanTempus.Application/Features/_ViewImports.cshtml b/PlanTempus.Application/Features/_ViewImports.cshtml new file mode 100644 index 0000000..bfd56f8 --- /dev/null +++ b/PlanTempus.Application/Features/_ViewImports.cshtml @@ -0,0 +1,4 @@ +@using PlanTempus.Application +@namespace PlanTempus.Application.Features +@addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers +@addTagHelper *, PlanTempus.Application diff --git a/PlanTempus.Application/Features/_ViewStart.cshtml b/PlanTempus.Application/Features/_ViewStart.cshtml new file mode 100644 index 0000000..f9f7d10 --- /dev/null +++ b/PlanTempus.Application/Features/_ViewStart.cshtml @@ -0,0 +1,3 @@ +@{ + Layout = "/Features/_Shared/Pages/_Layout.cshtml"; +} diff --git a/PlanTempus.Application/PlanTempus.Application.csproj b/PlanTempus.Application/PlanTempus.Application.csproj new file mode 100644 index 0000000..b9ee8e2 --- /dev/null +++ b/PlanTempus.Application/PlanTempus.Application.csproj @@ -0,0 +1,19 @@ + + + + net9.0 + enable + enable + + + + + + + + + + + + + \ No newline at end of file diff --git a/PlanTempus.Application/Program.cs b/PlanTempus.Application/Program.cs new file mode 100644 index 0000000..e839d00 --- /dev/null +++ b/PlanTempus.Application/Program.cs @@ -0,0 +1,47 @@ +using Microsoft.AspNetCore.Builder; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using Microsoft.AspNetCore.Mvc.Razor; +using Microsoft.AspNetCore.Mvc.RazorPages; +using PlanTempus.Application.Features.Localization.Services; +using PlanTempus.Application.Features.Menu.Services; + +var builder = WebApplication.CreateBuilder(args); + +// Add Razor Pages with feature-based structure +builder.Services.AddRazorPages(options => +{ + options.RootDirectory = "/Features"; +}) +.AddRazorOptions(options => +{ + // View locations for partials and ViewComponents + options.ViewLocationFormats.Add("/Features/_Shared/Pages/{0}.cshtml"); + options.ViewLocationFormats.Add("/Features/_Shared/Components/{1}/{0}.cshtml"); + +}); + +// Register application services +builder.Services.AddScoped(); +builder.Services.AddScoped(); + +var app = builder.Build(); + +// Developer exception page for debugging +if (app.Environment.IsDevelopment()) +{ + app.UseDeveloperExceptionPage(); +} + +// Serve static files from wwwroot +app.UseStaticFiles(); + +// Configure routing +app.UseRouting(); + +// Map Razor Pages +app.MapRazorPages(); + +app.Run("http://localhost:8000"); + +// Note: Set ASPNETCORE_ENVIRONMENT=Development for detailed error pages diff --git a/PlanTempus.Application/Properties/launchSettings.json b/PlanTempus.Application/Properties/launchSettings.json new file mode 100644 index 0000000..8f87adf --- /dev/null +++ b/PlanTempus.Application/Properties/launchSettings.json @@ -0,0 +1,12 @@ +{ + "profiles": { + "PlanTempus.Application": { + "commandName": "Project", + "launchBrowser": true, + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + }, + "applicationUrl": "https://localhost:55726;http://localhost:55727" + } + } +} \ No newline at end of file diff --git a/PlanTempus.Application/build.js b/PlanTempus.Application/build.js new file mode 100644 index 0000000..4aa944e --- /dev/null +++ b/PlanTempus.Application/build.js @@ -0,0 +1,24 @@ +import * as esbuild from 'esbuild'; + +async function build() { + try { + await esbuild.build({ + entryPoints: ['wwwroot/ts/app.ts'], + bundle: true, + outfile: 'wwwroot/js/app.js', + format: 'esm', + sourcemap: 'inline', + target: 'es2020', + minify: false, + keepNames: true, + platform: 'browser' + }); + + console.log('App bundle created: wwwroot/js/app.js'); + } catch (error) { + console.error('Build failed:', error); + process.exit(1); + } +} + +build(); diff --git a/PlanTempus.Application/wwwroot/css/app-layout.css b/PlanTempus.Application/wwwroot/css/app-layout.css new file mode 100644 index 0000000..2f57b74 --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/app-layout.css @@ -0,0 +1,50 @@ +/** + * App Layout - Main Grid Structure + * + * Definerer den overordnede app-struktur med sidebar og main content + */ + +/* =========================================== + MAIN APP GRID + =========================================== */ +swp-app-layout { + display: grid; + grid-template-columns: var(--side-menu-width) 1fr; + grid-template-rows: var(--topbar-height) 1fr; + height: 100vh; + transition: grid-template-columns var(--transition-normal); +} + +/* =========================================== + COLLAPSED MENU STATE + =========================================== */ +swp-app-layout.menu-collapsed { + grid-template-columns: var(--side-menu-width-collapsed) 1fr; +} + +/* =========================================== + MAIN CONTENT AREA + =========================================== */ +swp-main-content { + display: block; + overflow-y: auto; + background: var(--color-background); +} + +/* =========================================== + DRAWER OVERLAY + =========================================== */ +swp-drawer-overlay { + position: fixed; + inset: 0; + background: rgba(0, 0, 0, 0.5); + z-index: var(--z-overlay); + opacity: 0; + visibility: hidden; + transition: opacity var(--transition-normal), visibility var(--transition-normal); +} + +swp-drawer-overlay.active { + opacity: 1; + visibility: visible; +} diff --git a/PlanTempus.Application/wwwroot/css/base.css b/PlanTempus.Application/wwwroot/css/base.css new file mode 100644 index 0000000..00669dc --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/base.css @@ -0,0 +1,118 @@ +/** + * Base Styles - Reset & Global Elements + * + * Normalization og grundlæggende styling + */ + +/* =========================================== + FONT FACES + =========================================== */ +@font-face { + font-family: 'Poppins'; + src: url('../fonts/Poppins-Regular.woff') format('woff'); + font-weight: 400; + font-style: normal; + font-display: swap; +} + +@font-face { + font-family: 'Poppins'; + src: url('../fonts/Poppins-Medium.woff') format('woff'); + font-weight: 500; + font-style: normal; + font-display: swap; +} + +@font-face { + font-family: 'Poppins'; + src: url('../fonts/Poppins-SemiBold.woff') format('woff'); + font-weight: 600; + font-style: normal; + font-display: swap; +} + +@font-face { + font-family: 'Poppins'; + src: url('../fonts/Poppins-Bold.woff') format('woff'); + font-weight: 700; + font-style: normal; + font-display: swap; +} + +/* =========================================== + RESET + =========================================== */ +*, +*::before, +*::after { + margin: 0; + padding: 0; + box-sizing: border-box; +} + +html, +body { + height: 100%; +} + +/* =========================================== + BASE ELEMENTS + =========================================== */ +body { + font-family: var(--font-family); + font-size: var(--font-size-base); + color: var(--color-text); + background: var(--color-background); + line-height: var(--line-height-normal); + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} + +/* Links */ +a { + color: var(--color-teal); + text-decoration: none; + transition: color var(--transition-fast); +} + +a:hover { + text-decoration: underline; +} + +/* Lists */ +ul, ol { + list-style: none; +} + +/* Images */ +img { + max-width: 100%; + height: auto; + display: block; +} + +/* Buttons */ +button { + font-family: inherit; + cursor: pointer; +} + +/* Inputs */ +input, +textarea, +select { + font-family: inherit; + font-size: inherit; +} + +/* Focus visible */ +:focus-visible { + outline: 2px solid var(--color-teal); + outline-offset: 2px; +} + +/* Selection */ +::selection { + background: var(--color-teal-light); + color: var(--color-text); +} diff --git a/PlanTempus.Application/wwwroot/css/design-system.css b/PlanTempus.Application/wwwroot/css/design-system.css new file mode 100644 index 0000000..beb61ec --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/design-system.css @@ -0,0 +1,163 @@ +/** + * SWP Design System - CSS Variables + * + * Dette er den centrale definition af alle design tokens. + * Alle farver, fonts og layout-variabler defineres her. + */ + +/* =========================================== + COLOR PALETTE - Light Mode (Default) + =========================================== */ +:root { + /* Surfaces */ + --color-surface: #fff; + --color-background: #f5f5f5; + --color-background-hover: #f0f0f0; + --color-background-alt: #fafafa; + + /* Borders */ + --color-border: #e0e0e0; + --color-border-light: #f0f0f0; + + /* Text */ + --color-text: #333; + --color-text-secondary: #666; + --color-text-muted: #999; + + /* Brand Colors */ + --color-teal: #00897b; + --color-teal-light: color-mix(in srgb, var(--color-teal) 10%, transparent); + + /* Semantic Colors */ + --color-blue: #1976d2; + --color-green: #43a047; + --color-amber: #f59e0b; + --color-red: #e53935; + --color-purple: #8b5cf6; +} + +/* =========================================== + COLOR PALETTE - Dark Mode (System) + =========================================== */ +@media (prefers-color-scheme: dark) { + :root:not(.light-mode) { + --color-surface: #1e1e1e; + --color-background: #121212; + --color-background-hover: #2a2a2a; + --color-background-alt: #1a1a1a; + + --color-border: #333; + --color-border-light: #2a2a2a; + + --color-text: #e0e0e0; + --color-text-secondary: #999; + --color-text-muted: #666; + + --color-teal: #26a69a; + --color-blue: #42a5f5; + --color-green: #66bb6a; + --color-amber: #ffb74d; + --color-red: #ef5350; + --color-purple: #a78bfa; + } +} + +/* =========================================== + COLOR PALETTE - Dark Mode (Manual) + =========================================== */ +:root.dark-mode { + --color-surface: #1e1e1e; + --color-background: #121212; + --color-background-hover: #2a2a2a; + --color-background-alt: #1a1a1a; + + --color-border: #333; + --color-border-light: #2a2a2a; + + --color-text: #e0e0e0; + --color-text-secondary: #999; + --color-text-muted: #666; + + --color-teal: #26a69a; + --color-blue: #42a5f5; + --color-green: #66bb6a; + --color-amber: #ffb74d; + --color-red: #ef5350; + --color-purple: #a78bfa; +} + +/* =========================================== + TYPOGRAPHY + =========================================== */ +:root { + --font-family: 'Poppins', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; + --font-mono: 'JetBrains Mono', monospace; + + /* Font Sizes */ + --font-size-xs: 11px; + --font-size-sm: 12px; + --font-size-base: 14px; + --font-size-md: 13px; + --font-size-lg: 16px; + --font-size-xl: 22px; + + /* Line Heights */ + --line-height-tight: 1.25; + --line-height-normal: 1.5; + --line-height-relaxed: 1.75; +} + +/* =========================================== + SPACING + =========================================== */ +:root { + --spacing-1: 4px; + --spacing-2: 8px; + --spacing-3: 12px; + --spacing-4: 16px; + --spacing-5: 20px; + --spacing-6: 24px; + --spacing-8: 32px; +} + +/* =========================================== + LAYOUT + =========================================== */ +:root { + --side-menu-width: 240px; + --side-menu-width-collapsed: 64px; + --topbar-height: 56px; + --page-max-width: 1400px; + --border-radius: 6px; + --border-radius-lg: 8px; +} + +/* =========================================== + TRANSITIONS + =========================================== */ +:root { + --transition-fast: 150ms ease; + --transition-normal: 200ms ease; + --transition-slow: 300ms ease; +} + +/* =========================================== + Z-INDEX LAYERS + =========================================== */ +:root { + --z-dropdown: 100; + --z-sticky: 200; + --z-overlay: 900; + --z-drawer: 1000; + --z-modal: 1100; + --z-tooltip: 1200; +} + +/* =========================================== + SHADOWS + =========================================== */ +:root { + --shadow-sm: 0 1px 2px rgba(0, 0, 0, 0.05); + --shadow-md: 0 2px 8px rgba(0, 0, 0, 0.1); + --shadow-lg: 0 4px 16px rgba(0, 0, 0, 0.15); +} diff --git a/PlanTempus.Application/wwwroot/css/drawers.css b/PlanTempus.Application/wwwroot/css/drawers.css new file mode 100644 index 0000000..2805da5 --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/drawers.css @@ -0,0 +1,258 @@ +/** + * Drawers - Slide-in Panels + * + * Profile drawer, notifications drawer, etc. + */ + +/* =========================================== + BASE DRAWER + =========================================== */ +swp-profile-drawer, +swp-notification-drawer, +swp-todo-drawer { + position: fixed; + top: 0; + right: 0; + width: 320px; + height: 100vh; + background: var(--color-surface); + border-left: 1px solid var(--color-border); + box-shadow: var(--shadow-lg); + z-index: var(--z-drawer); + display: flex; + flex-direction: column; + transform: translateX(100%); + transition: transform var(--transition-normal); +} + +swp-profile-drawer.active, +swp-notification-drawer.active, +swp-todo-drawer.active { + transform: translateX(0); +} + +/* =========================================== + DRAWER HEADER + =========================================== */ +swp-drawer-header { + display: flex; + align-items: center; + justify-content: space-between; + padding: var(--spacing-4) var(--spacing-5); + border-bottom: 1px solid var(--color-border); + flex-shrink: 0; +} + +swp-drawer-title { + font-size: var(--font-size-lg); + font-weight: 600; + color: var(--color-text); +} + +swp-drawer-close { + width: 32px; + height: 32px; + display: flex; + align-items: center; + justify-content: center; + border: none; + background: transparent; + border-radius: var(--border-radius); + cursor: pointer; + color: var(--color-text-secondary); + transition: all var(--transition-fast); +} + +swp-drawer-close:hover { + background: var(--color-background-hover); + color: var(--color-text); +} + +swp-drawer-close i { + font-size: 20px; +} + +/* =========================================== + DRAWER CONTENT + =========================================== */ +swp-drawer-content { + flex: 1; + overflow-y: auto; + padding: var(--spacing-5); +} + +swp-drawer-divider { + height: 1px; + background: var(--color-border); + margin: var(--spacing-4) 0; +} + +/* =========================================== + PROFILE SECTION + =========================================== */ +swp-profile-section { + display: flex; + flex-direction: column; + align-items: center; + text-align: center; + padding: var(--spacing-4) 0; +} + +swp-profile-avatar-large { + width: 64px; + height: 64px; + border-radius: 50%; + background: var(--color-teal); + color: white; + font-size: 24px; + font-weight: 600; + display: flex; + align-items: center; + justify-content: center; + margin-bottom: var(--spacing-3); +} + +swp-profile-name-large { + font-size: var(--font-size-lg); + font-weight: 600; + color: var(--color-text); + margin-bottom: var(--spacing-1); +} + +swp-profile-email { + font-size: var(--font-size-sm); + color: var(--color-text-secondary); +} + +/* =========================================== + DRAWER MENU + =========================================== */ +swp-drawer-menu { + display: flex; + flex-direction: column; + gap: var(--spacing-1); +} + +swp-drawer-menu-item { + display: flex; + align-items: center; + gap: var(--spacing-3); + padding: var(--spacing-3) var(--spacing-3); + border-radius: var(--border-radius); + cursor: pointer; + transition: background var(--transition-fast); + color: var(--color-text); +} + +swp-drawer-menu-item:hover { + background: var(--color-background-hover); +} + +swp-drawer-menu-item i { + font-size: 20px; + color: var(--color-text-secondary); +} + +/* =========================================== + THEME TOGGLE + =========================================== */ +swp-theme-toggle { + display: flex; + align-items: center; + justify-content: space-between; + padding: var(--spacing-3); + border-radius: var(--border-radius); + background: var(--color-background); +} + +swp-theme-label { + display: flex; + align-items: center; + gap: var(--spacing-3); + color: var(--color-text); +} + +swp-theme-label i { + font-size: 20px; + color: var(--color-text-secondary); +} + +swp-toggle-switch { + position: relative; + width: 44px; + height: 24px; +} + +swp-toggle-switch input { + opacity: 0; + width: 0; + height: 0; +} + +swp-toggle-slider { + position: absolute; + cursor: pointer; + inset: 0; + background: var(--color-border); + border-radius: 12px; + transition: background var(--transition-fast); +} + +swp-toggle-slider::before { + content: ''; + position: absolute; + width: 18px; + height: 18px; + left: 3px; + bottom: 3px; + background: white; + border-radius: 50%; + transition: transform var(--transition-fast); +} + +swp-toggle-switch input:checked + swp-toggle-slider { + background: var(--color-teal); +} + +swp-toggle-switch input:checked + swp-toggle-slider::before { + transform: translateX(20px); +} + +/* =========================================== + DRAWER FOOTER + =========================================== */ +swp-drawer-footer { + padding: var(--spacing-4) var(--spacing-5); + border-top: 1px solid var(--color-border); + flex-shrink: 0; +} + +swp-drawer-action { + display: flex; + align-items: center; + justify-content: center; + gap: var(--spacing-2); + width: 100%; + padding: var(--spacing-3); + font-size: var(--font-size-base); + font-family: var(--font-family); + color: var(--color-text-secondary); + background: transparent; + border: 1px solid var(--color-border); + border-radius: var(--border-radius); + cursor: pointer; + transition: all var(--transition-fast); +} + +swp-drawer-action:hover { + background: var(--color-background-hover); +} + +swp-drawer-action.logout:hover { + color: var(--color-red); + border-color: var(--color-red); +} + +swp-drawer-action i { + font-size: 18px; +} diff --git a/PlanTempus.Application/wwwroot/css/page.css b/PlanTempus.Application/wwwroot/css/page.css new file mode 100644 index 0000000..cf4b9c7 --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/page.css @@ -0,0 +1,204 @@ +/** + * Page Layout - Content Area Structure + * + * Page container, headers, cards og grid layouts + */ + +/* =========================================== + PAGE CONTAINER + =========================================== */ +swp-page-container { + display: block; + max-width: var(--page-max-width); + margin: 0 auto; + padding: var(--spacing-6); +} + +/* =========================================== + PAGE HEADER + =========================================== */ +swp-page-header { + display: flex; + align-items: flex-start; + justify-content: space-between; + margin-bottom: var(--spacing-6); +} + +swp-page-title h1 { + font-size: 24px; + font-weight: 600; + color: var(--color-text); + margin-bottom: var(--spacing-1); +} + +swp-page-title p { + font-size: var(--font-size-base); + color: var(--color-text-secondary); +} + +swp-page-actions { + display: flex; + gap: var(--spacing-2); +} + +/* =========================================== + CARDS + =========================================== */ +swp-card { + display: block; + background: var(--color-surface); + border: 1px solid var(--color-border); + border-radius: var(--border-radius-lg); + margin-bottom: var(--spacing-5); +} + +swp-card-header { + display: flex; + align-items: center; + justify-content: space-between; + padding: var(--spacing-4) var(--spacing-5); + border-bottom: 1px solid var(--color-border); +} + +swp-card-title { + display: flex; + align-items: center; + gap: var(--spacing-2); + font-size: var(--font-size-base); + font-weight: 600; + color: var(--color-text); +} + +swp-card-title i { + font-size: 20px; + color: var(--color-text-secondary); +} + +swp-card-action { + font-size: var(--font-size-md); + color: var(--color-teal); + cursor: pointer; + transition: opacity var(--transition-fast); +} + +swp-card-action:hover { + opacity: 0.8; +} + +swp-card-content { + padding: var(--spacing-5); +} + +/* =========================================== + DASHBOARD GRID + =========================================== */ +swp-dashboard-grid { + display: grid; + grid-template-columns: 1fr 350px; + gap: var(--spacing-5); +} + +swp-main-column { + display: flex; + flex-direction: column; +} + +swp-side-column { + display: flex; + flex-direction: column; +} + +/* =========================================== + AI INSIGHT + =========================================== */ +swp-ai-insight { + display: block; + padding: var(--spacing-4) var(--spacing-5); + background: linear-gradient(135deg, + color-mix(in srgb, var(--color-purple) 8%, transparent), + color-mix(in srgb, var(--color-teal) 8%, transparent) + ); + border-radius: var(--border-radius); +} + +swp-ai-header { + display: flex; + align-items: center; + gap: var(--spacing-2); + margin-bottom: var(--spacing-2); + font-size: var(--font-size-sm); + font-weight: 500; + color: var(--color-purple); +} + +swp-ai-header i { + font-size: 16px; +} + +swp-ai-text { + font-size: var(--font-size-base); + color: var(--color-text); + line-height: var(--line-height-relaxed); +} + +/* =========================================== + QUICK ACTIONS + =========================================== */ +swp-quick-actions { + display: flex; + flex-direction: column; + gap: var(--spacing-2); +} + +swp-quick-action-btn { + display: flex; + align-items: center; + gap: var(--spacing-2); + padding: var(--spacing-3); + font-size: var(--font-size-base); + font-family: var(--font-family); + color: var(--color-text); + background: var(--color-background); + border: 1px solid var(--color-border); + border-radius: var(--border-radius); + cursor: pointer; + transition: all var(--transition-fast); +} + +swp-quick-action-btn:hover { + background: var(--color-background-hover); + border-color: var(--color-teal); + color: var(--color-teal); +} + +swp-quick-action-btn i { + font-size: 18px; +} + +/* =========================================== + RESPONSIVE + =========================================== */ +@media (max-width: 1200px) { + swp-dashboard-grid { + grid-template-columns: 1fr; + } + + swp-side-column { + order: -1; + } +} + +@media (max-width: 768px) { + swp-page-container { + padding: var(--spacing-4); + } + + swp-page-header { + flex-direction: column; + gap: var(--spacing-4); + } + + swp-page-actions { + width: 100%; + } +} diff --git a/PlanTempus.Application/wwwroot/css/sidebar.css b/PlanTempus.Application/wwwroot/css/sidebar.css new file mode 100644 index 0000000..cc9354e --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/sidebar.css @@ -0,0 +1,246 @@ +/** + * Sidebar - Side Menu Component + * + * Navigation sidebar med collapse-funktionalitet + */ + +/* =========================================== + SIDE MENU CONTAINER + =========================================== */ +swp-side-menu { + grid-row: 1 / -1; + display: flex; + flex-direction: column; + background: var(--color-surface); + border-right: 1px solid var(--color-border); + overflow-y: auto; + overflow-x: hidden; +} + +/* =========================================== + HEADER + =========================================== */ +swp-side-menu-header { + display: flex; + align-items: center; + gap: 10px; + height: var(--topbar-height); + padding: 0 var(--spacing-4); + border-bottom: 1px solid var(--color-border); + flex-shrink: 0; +} + +swp-side-menu-header > i { + font-size: 26px; + color: var(--color-teal); +} + +swp-side-menu-logo { + font-size: var(--font-size-lg); + font-weight: 600; + color: var(--color-text); +} + +/* Toggle Button */ +swp-menu-toggle { + margin-left: auto; + width: 28px; + height: 28px; + display: flex; + align-items: center; + justify-content: center; + border: none; + background: transparent; + border-radius: var(--border-radius); + cursor: pointer; + color: var(--color-text-secondary); + transition: all var(--transition-fast); +} + +swp-menu-toggle:hover { + background: var(--color-background-hover); + color: var(--color-text); +} + +swp-menu-toggle i { + font-size: 18px; + color: inherit; + transition: transform var(--transition-normal); +} + +/* =========================================== + NAVIGATION + =========================================== */ +swp-side-menu-nav { + flex: 1; + padding: var(--spacing-3) 0; + overflow-y: auto; +} + +swp-side-menu-group { + display: block; + margin-bottom: var(--spacing-2); +} + +swp-side-menu-label { + display: block; + padding: var(--spacing-2) var(--spacing-4) 6px; + font-size: var(--font-size-xs); + font-weight: 600; + text-transform: uppercase; + letter-spacing: 0.5px; + color: var(--color-text-secondary); +} + +/* =========================================== + MENU ITEMS + =========================================== */ +swp-side-menu-item, +a[is="swp-side-menu-item"] { + display: flex; + align-items: center; + gap: var(--spacing-3); + padding: 10px var(--spacing-4); + color: var(--color-text); + cursor: pointer; + transition: all var(--transition-fast); + border-left: 3px solid transparent; + text-decoration: none; +} + +swp-side-menu-item:hover, +a[is="swp-side-menu-item"]:hover { + background: var(--color-background-hover); + text-decoration: none; +} + +swp-side-menu-item[data-active="true"], +a[is="swp-side-menu-item"][data-active="true"] { + background: var(--color-teal-light); + border-left-color: var(--color-teal); + color: var(--color-teal); + font-weight: 500; +} + +swp-side-menu-item i, +a[is="swp-side-menu-item"] i { + font-size: 20px; + flex-shrink: 0; +} + +/* =========================================== + FOOTER + =========================================== */ +swp-side-menu-footer { + display: flex; + flex-direction: column; + gap: var(--spacing-2); + padding: var(--spacing-3) var(--spacing-4); + border-top: 1px solid var(--color-border); + margin-top: auto; + flex-shrink: 0; +} + +swp-side-menu-action { + display: flex; + align-items: center; + justify-content: center; + gap: var(--spacing-2); + padding: 10px; + font-size: var(--font-size-md); + color: var(--color-text-secondary); + background: transparent; + border: 1px solid var(--color-border); + border-radius: var(--border-radius); + cursor: pointer; + transition: all var(--transition-fast); + font-family: var(--font-family); +} + +swp-side-menu-action:hover { + background: var(--color-background-hover); +} + +swp-side-menu-action.lock:hover { + color: var(--color-amber); + border-color: var(--color-amber); +} + +swp-side-menu-action.logout:hover { + color: var(--color-red); + border-color: var(--color-red); +} + +swp-side-menu-action i { + font-size: 18px; +} + +/* =========================================== + COLLAPSED STATE + =========================================== */ +swp-app-layout.menu-collapsed swp-side-menu { + overflow: visible; +} + +swp-app-layout.menu-collapsed swp-side-menu-logo, +swp-app-layout.menu-collapsed swp-side-menu-label, +swp-app-layout.menu-collapsed swp-side-menu-item span, +swp-app-layout.menu-collapsed swp-side-menu-action span, +swp-app-layout.menu-collapsed a[is="swp-side-menu-item"] span { + display: none; +} + +swp-app-layout.menu-collapsed swp-side-menu-header { + justify-content: center; + padding: 0; +} + +swp-app-layout.menu-collapsed swp-menu-toggle { + margin-left: 0; +} + +swp-app-layout.menu-collapsed swp-menu-toggle i { + transform: rotate(180deg); +} + +swp-app-layout.menu-collapsed swp-side-menu-item, +swp-app-layout.menu-collapsed a[is="swp-side-menu-item"] { + justify-content: center; + padding: var(--spacing-3); + border-left: none; +} + +swp-app-layout.menu-collapsed swp-side-menu-item[data-active="true"], +swp-app-layout.menu-collapsed a[is="swp-side-menu-item"][data-active="true"] { + border-left: none; + border-radius: var(--border-radius-lg); + margin: 0 var(--spacing-2); +} + +swp-app-layout.menu-collapsed swp-side-menu-action { + justify-content: center; + padding: var(--spacing-3); +} + +swp-app-layout.menu-collapsed swp-side-menu-footer { + padding: var(--spacing-3) var(--spacing-2); +} + +/* =========================================== + TOOLTIP (Collapsed State) + =========================================== */ +.swp-menu-tooltip { + position: fixed; + margin: 0; + padding: 6px var(--spacing-3); + background: var(--color-surface); + border: 1px solid var(--color-border); + border-radius: var(--border-radius); + font-size: var(--font-size-md); + font-family: var(--font-family); + color: var(--color-text); + white-space: nowrap; + box-shadow: var(--shadow-md); + pointer-events: none; + z-index: var(--z-tooltip); +} diff --git a/PlanTempus.Application/wwwroot/css/stats.css b/PlanTempus.Application/wwwroot/css/stats.css new file mode 100644 index 0000000..2d50a5f --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/stats.css @@ -0,0 +1,258 @@ +/** + * Stats - Statistics Components + * + * Stat bars, cards, values og trends + */ + +/* =========================================== + STATS CONTAINER (Grid/Bar/Row) + =========================================== */ +swp-stats-bar, +swp-stats-grid { + display: grid; + grid-template-columns: repeat(4, 1fr); + gap: var(--spacing-4); + margin-bottom: var(--spacing-5); +} + +swp-stats-row { + display: grid; + grid-template-columns: repeat(3, 1fr); + gap: var(--spacing-4); + margin-bottom: var(--spacing-5); +} + +/* =========================================== + STAT CARD + =========================================== */ +swp-stat-card { + display: flex; + flex-direction: column; + background: var(--color-surface); + border-radius: var(--border-radius-lg); + padding: var(--spacing-4) var(--spacing-5); + border: 1px solid var(--color-border); +} + +swp-stat-box { + display: flex; + flex-direction: column; + gap: var(--spacing-1); + padding: var(--spacing-4); + background: var(--color-surface); + border-radius: var(--border-radius); + border: 1px solid var(--color-border); +} + +/* =========================================== + STAT VALUE + =========================================== */ +swp-stat-value { + display: block; + font-size: 22px; + font-weight: 600; + font-family: var(--font-mono); + color: var(--color-text); + line-height: var(--line-height-tight); +} + +/* Larger variant for emphasis */ +swp-stat-card swp-stat-value, +swp-stat-box swp-stat-value { + font-size: 22px; + font-weight: 600; + font-family: var(--font-mono); + color: var(--color-text); +} + +/* =========================================== + STAT LABEL + =========================================== */ +swp-stat-label { + display: block; + font-size: var(--font-size-sm); + color: var(--color-text-secondary); + margin-top: var(--spacing-1); +} + +swp-stat-box swp-stat-label { + font-size: 11px; + font-weight: 500; + text-transform: uppercase; + letter-spacing: 0.5px; + color: var(--color-text-secondary); +} + +/* =========================================== + STAT SUBTITLE + =========================================== */ +swp-stat-subtitle { + display: block; + font-size: 11px; + color: var(--color-text-muted); + margin-top: var(--spacing-1); +} + +/* =========================================== + STAT TREND / CHANGE + =========================================== */ +swp-stat-trend, +swp-stat-change { + display: inline-flex; + align-items: center; + gap: var(--spacing-1); + font-size: var(--font-size-xs); + margin-top: var(--spacing-2); +} + +swp-stat-trend i, +swp-stat-change i { + font-size: 14px; +} + +/* Trend Up (positive) */ +swp-stat-trend.up, +swp-stat-change.positive { + color: var(--color-green); +} + +/* Trend Down (negative) */ +swp-stat-trend.down, +swp-stat-change.negative { + color: var(--color-red); +} + +/* Neutral trend */ +swp-stat-trend.neutral { + color: var(--color-text-secondary); +} + +/* =========================================== + COLOR MODIFIERS + =========================================== */ + +/* Highlight (Primary/Teal) */ +swp-stat-card.highlight swp-stat-value, +swp-stat-box.highlight swp-stat-value, +swp-stat-card.teal swp-stat-value { + color: var(--color-teal); +} + +/* Success (Green) */ +swp-stat-card.success swp-stat-value { + color: var(--color-green); +} + +/* Warning (Amber) */ +swp-stat-card.warning swp-stat-value, +swp-stat-card.amber swp-stat-value { + color: var(--color-amber); +} + +/* Danger (Red) */ +swp-stat-card.danger swp-stat-value, +swp-stat-card.negative swp-stat-value, +swp-stat-card.red swp-stat-value { + color: var(--color-red); +} + +/* Purple */ +swp-stat-card.purple swp-stat-value { + color: var(--color-purple); +} + +/* =========================================== + HIGHLIGHT CARD (Filled Background) + =========================================== */ +swp-stat-card.highlight.filled { + background: linear-gradient(135deg, var(--color-teal) 0%, #00695c 100%); + color: white; + border-color: transparent; +} + +swp-stat-card.highlight.filled swp-stat-value { + color: white; +} + +swp-stat-card.highlight.filled swp-stat-label { + color: rgba(255, 255, 255, 0.8); +} + +swp-stat-card.highlight.filled swp-stat-change { + color: rgba(255, 255, 255, 0.9); +} + +/* =========================================== + QUICK STATS (Compact Variant) + =========================================== */ +swp-quick-stats { + display: grid; + grid-template-columns: repeat(4, 1fr); + gap: var(--spacing-3); +} + +swp-quick-stat { + display: flex; + flex-direction: column; + text-align: center; + padding: var(--spacing-3); + background: var(--color-background); + border-radius: var(--border-radius); +} + +swp-quick-stat swp-stat-value { + font-size: 18px; +} + +swp-quick-stat swp-stat-label { + font-size: 11px; + margin-top: var(--spacing-1); +} + +/* =========================================== + STAT ITEM (Inline Variant) + =========================================== */ +swp-stat-item { + display: flex; + flex-direction: column; + gap: 2px; + padding: var(--spacing-2) var(--spacing-3); + background: var(--color-background); + border-radius: var(--border-radius); +} + +swp-stat-item swp-stat-value { + font-size: 16px; + font-weight: 600; +} + +swp-stat-item swp-stat-value.mono { + font-family: var(--font-mono); +} + +swp-stat-item swp-stat-label { + font-size: 11px; + margin-top: 0; +} + +/* =========================================== + RESPONSIVE + =========================================== */ +@media (max-width: 1200px) { + swp-stats-bar, + swp-stats-grid { + grid-template-columns: repeat(2, 1fr); + } +} + +@media (max-width: 768px) { + swp-stats-bar, + swp-stats-grid, + swp-stats-row { + grid-template-columns: 1fr; + } + + swp-quick-stats { + grid-template-columns: repeat(2, 1fr); + } +} diff --git a/PlanTempus.Application/wwwroot/css/topbar.css b/PlanTempus.Application/wwwroot/css/topbar.css new file mode 100644 index 0000000..6c9d182 --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/topbar.css @@ -0,0 +1,180 @@ +/** + * Topbar - App Header Bar + * + * Search, notifications og profil-menu + */ + +/* =========================================== + TOPBAR CONTAINER + =========================================== */ +swp-app-topbar { + display: flex; + align-items: center; + justify-content: space-between; + padding: 0 var(--spacing-5); + background: var(--color-surface); + border-bottom: 1px solid var(--color-border); + position: sticky; + top: 0; + z-index: var(--z-sticky); +} + +/* =========================================== + SEARCH + =========================================== */ +swp-topbar-search { + display: flex; + align-items: center; + gap: 10px; + padding: var(--spacing-2) var(--spacing-3); + background: var(--color-background); + border: 1px solid var(--color-border); + border-radius: var(--border-radius); + width: 320px; + transition: border-color var(--transition-fast); +} + +swp-topbar-search:focus-within { + border-color: var(--color-teal); +} + +swp-topbar-search i { + font-size: 18px; + color: var(--color-text-secondary); + flex-shrink: 0; +} + +swp-topbar-search input { + flex: 1; + border: none; + outline: none; + background: transparent; + font-size: var(--font-size-md); + font-family: var(--font-family); + color: var(--color-text); +} + +swp-topbar-search input::placeholder { + color: var(--color-text-secondary); +} + +swp-topbar-search kbd { + font-size: var(--font-size-xs); + font-family: var(--font-mono); + padding: 2px 6px; + background: var(--color-surface); + border: 1px solid var(--color-border); + border-radius: 4px; + color: var(--color-text-secondary); +} + +/* =========================================== + ACTIONS + =========================================== */ +swp-topbar-actions { + display: flex; + align-items: center; + gap: var(--spacing-2); +} + +/* Action Buttons */ +swp-topbar-btn { + display: flex; + align-items: center; + justify-content: center; + width: 38px; + height: 38px; + border: none; + background: transparent; + border-radius: var(--border-radius); + cursor: pointer; + transition: background var(--transition-fast); + position: relative; + color: var(--color-text-secondary); +} + +swp-topbar-btn:hover { + background: var(--color-background-hover); +} + +swp-topbar-btn i { + font-size: 22px; +} + +/* Notification Badge */ +swp-notification-badge { + position: absolute; + top: 6px; + right: 6px; + min-width: 16px; + height: 16px; + padding: 0 4px; + font-size: 10px; + font-weight: 600; + background: var(--color-red); + color: white; + border-radius: 8px; + display: flex; + align-items: center; + justify-content: center; +} + +/* Divider */ +swp-topbar-divider { + width: 1px; + height: 24px; + background: var(--color-border); + margin: 0 var(--spacing-2); +} + +/* =========================================== + PROFILE TRIGGER + =========================================== */ +swp-topbar-profile { + display: flex; + align-items: center; + gap: 10px; + padding: 6px var(--spacing-3) 6px 6px; + background: transparent; + border: 1px solid transparent; + border-radius: var(--border-radius); + cursor: pointer; + transition: all var(--transition-fast); +} + +swp-topbar-profile:hover { + background: var(--color-background-hover); + border-color: var(--color-border); +} + +swp-profile-avatar { + width: 32px; + height: 32px; + border-radius: 50%; + background: var(--color-teal); + color: white; + font-size: var(--font-size-sm); + font-weight: 600; + display: flex; + align-items: center; + justify-content: center; +} + +swp-profile-info { + display: flex; + flex-direction: column; + align-items: flex-start; +} + +swp-profile-name { + font-size: var(--font-size-md); + font-weight: 500; + color: var(--color-text); + line-height: var(--line-height-tight); +} + +swp-profile-role { + font-size: var(--font-size-xs); + color: var(--color-text-secondary); + line-height: var(--line-height-tight); +} diff --git a/PlanTempus.Application/wwwroot/email-templates/verify-email.html b/PlanTempus.Application/wwwroot/email-templates/verify-email.html new file mode 100644 index 0000000..8290a3f --- /dev/null +++ b/PlanTempus.Application/wwwroot/email-templates/verify-email.html @@ -0,0 +1,308 @@ + + + + + + + Bekræft din email - Plan Tempus + + + + + + + + + + + + diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-Black.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-Black.woff new file mode 100644 index 0000000..65f3801 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-Black.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-BlackItalic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-BlackItalic.woff new file mode 100644 index 0000000..63eb02e Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-BlackItalic.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-Bold.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-Bold.woff new file mode 100644 index 0000000..28fe6b6 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-Bold.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-BoldItalic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-BoldItalic.woff new file mode 100644 index 0000000..8ec438d Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-BoldItalic.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraBold.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraBold.woff new file mode 100644 index 0000000..10df1ac Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraBold.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraBoldItalic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraBoldItalic.woff new file mode 100644 index 0000000..38090a0 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraBoldItalic.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraLight.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraLight.woff new file mode 100644 index 0000000..8eb4095 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraLight.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraLightItalic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraLightItalic.woff new file mode 100644 index 0000000..560f65e Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-ExtraLightItalic.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-Italic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-Italic.woff new file mode 100644 index 0000000..341cbb3 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-Italic.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-Light.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-Light.woff new file mode 100644 index 0000000..61f6a5c Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-Light.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-LightItalic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-LightItalic.woff new file mode 100644 index 0000000..2706cd4 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-LightItalic.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-Medium.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-Medium.woff new file mode 100644 index 0000000..e2b717d Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-Medium.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-MediumItalic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-MediumItalic.woff new file mode 100644 index 0000000..d8101a7 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-MediumItalic.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-Regular.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-Regular.woff new file mode 100644 index 0000000..609eb3d Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-Regular.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-SemiBold.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-SemiBold.woff new file mode 100644 index 0000000..c097e14 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-SemiBold.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-SemiBoldItalic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-SemiBoldItalic.woff new file mode 100644 index 0000000..902dcfd Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-SemiBoldItalic.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-Thin.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-Thin.woff new file mode 100644 index 0000000..d21fbd3 Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-Thin.woff differ diff --git a/PlanTempus.Application/wwwroot/fonts/Poppins-ThinItalic.woff b/PlanTempus.Application/wwwroot/fonts/Poppins-ThinItalic.woff new file mode 100644 index 0000000..ea3a42f Binary files /dev/null and b/PlanTempus.Application/wwwroot/fonts/Poppins-ThinItalic.woff differ diff --git a/PlanTempus.Application/wwwroot/images/poclogo.png b/PlanTempus.Application/wwwroot/images/poclogo.png new file mode 100644 index 0000000..d363158 Binary files /dev/null and b/PlanTempus.Application/wwwroot/images/poclogo.png differ diff --git a/PlanTempus.Application/wwwroot/js/app.js b/PlanTempus.Application/wwwroot/js/app.js new file mode 100644 index 0000000..701f2c5 --- /dev/null +++ b/PlanTempus.Application/wwwroot/js/app.js @@ -0,0 +1,604 @@ +var __defProp = Object.defineProperty; +var __name = (target, value) => __defProp(target, "name", { value, configurable: true }); + +// wwwroot/ts/modules/sidebar.ts +var _SidebarController = class _SidebarController { + constructor() { + this.menuToggle = null; + this.appLayout = null; + this.menuTooltip = null; + this.menuToggle = document.getElementById("menuToggle"); + this.appLayout = document.querySelector("swp-app-layout"); + this.menuTooltip = document.getElementById("menuTooltip"); + this.setupListeners(); + this.setupTooltips(); + this.restoreState(); + } + /** + * Check if sidebar is collapsed + */ + get isCollapsed() { + return this.appLayout?.classList.contains("menu-collapsed") ?? false; + } + /** + * Toggle sidebar collapsed state + */ + toggle() { + if (!this.appLayout) + return; + this.appLayout.classList.toggle("menu-collapsed"); + localStorage.setItem("sidebar-collapsed", String(this.isCollapsed)); + } + /** + * Collapse the sidebar + */ + collapse() { + this.appLayout?.classList.add("menu-collapsed"); + localStorage.setItem("sidebar-collapsed", "true"); + } + /** + * Expand the sidebar + */ + expand() { + this.appLayout?.classList.remove("menu-collapsed"); + localStorage.setItem("sidebar-collapsed", "false"); + } + setupListeners() { + this.menuToggle?.addEventListener("click", () => this.toggle()); + } + setupTooltips() { + if (!this.menuTooltip) + return; + const menuItems = document.querySelectorAll("swp-side-menu-item[data-tooltip]"); + menuItems.forEach((item) => { + item.addEventListener("mouseenter", () => this.showTooltip(item)); + item.addEventListener("mouseleave", () => this.hideTooltip()); + }); + } + showTooltip(item) { + if (!this.isCollapsed || !this.menuTooltip) + return; + const rect = item.getBoundingClientRect(); + const tooltipText = item.dataset.tooltip; + if (!tooltipText) + return; + this.menuTooltip.textContent = tooltipText; + this.menuTooltip.style.left = `${rect.right + 8}px`; + this.menuTooltip.style.top = `${rect.top + rect.height / 2}px`; + this.menuTooltip.style.transform = "translateY(-50%)"; + this.menuTooltip.showPopover(); + } + hideTooltip() { + this.menuTooltip?.hidePopover(); + } + restoreState() { + if (!this.appLayout) + return; + if (localStorage.getItem("sidebar-collapsed") === "true") { + this.appLayout.classList.add("menu-collapsed"); + } + } +}; +__name(_SidebarController, "SidebarController"); +var SidebarController = _SidebarController; + +// wwwroot/ts/modules/drawers.ts +var _DrawerController = class _DrawerController { + constructor() { + this.profileDrawer = null; + this.notificationDrawer = null; + this.todoDrawer = null; + this.newTodoDrawer = null; + this.overlay = null; + this.activeDrawer = null; + this.profileDrawer = document.getElementById("profileDrawer"); + this.notificationDrawer = document.getElementById("notificationDrawer"); + this.todoDrawer = document.getElementById("todoDrawer"); + this.newTodoDrawer = document.getElementById("newTodoDrawer"); + this.overlay = document.getElementById("drawerOverlay"); + this.setupListeners(); + } + /** + * Get currently active drawer name + */ + get active() { + return this.activeDrawer; + } + /** + * Open a drawer by name + */ + open(name) { + this.closeAll(); + const drawer = this.getDrawer(name); + if (drawer && this.overlay) { + drawer.classList.add("active"); + this.overlay.classList.add("active"); + document.body.style.overflow = "hidden"; + this.activeDrawer = name; + } + } + /** + * Close a specific drawer + */ + close(name) { + const drawer = this.getDrawer(name); + drawer?.classList.remove("active"); + if (this.overlay && !document.querySelector('.active[class*="drawer"]')) { + this.overlay.classList.remove("active"); + document.body.style.overflow = ""; + } + if (this.activeDrawer === name) { + this.activeDrawer = null; + } + } + /** + * Close all drawers + */ + closeAll() { + [this.profileDrawer, this.notificationDrawer, this.todoDrawer, this.newTodoDrawer].forEach((drawer) => drawer?.classList.remove("active")); + this.overlay?.classList.remove("active"); + document.body.style.overflow = ""; + this.activeDrawer = null; + } + /** + * Open profile drawer + */ + openProfile() { + this.open("profile"); + } + /** + * Open notification drawer + */ + openNotification() { + this.open("notification"); + } + /** + * Open todo drawer (slides on top of profile) + */ + openTodo() { + this.todoDrawer?.classList.add("active"); + } + /** + * Close todo drawer + */ + closeTodo() { + this.todoDrawer?.classList.remove("active"); + this.closeNewTodo(); + } + /** + * Open new todo drawer + */ + openNewTodo() { + this.newTodoDrawer?.classList.add("active"); + } + /** + * Close new todo drawer + */ + closeNewTodo() { + this.newTodoDrawer?.classList.remove("active"); + } + /** + * Mark all notifications as read + */ + markAllNotificationsRead() { + if (!this.notificationDrawer) + return; + const unreadItems = this.notificationDrawer.querySelectorAll( + 'swp-notification-item[data-unread="true"]' + ); + unreadItems.forEach((item) => item.removeAttribute("data-unread")); + const badge = document.querySelector("swp-notification-badge"); + if (badge) { + badge.style.display = "none"; + } + } + getDrawer(name) { + switch (name) { + case "profile": + return this.profileDrawer; + case "notification": + return this.notificationDrawer; + case "todo": + return this.todoDrawer; + case "newTodo": + return this.newTodoDrawer; + } + } + setupListeners() { + document.getElementById("profileTrigger")?.addEventListener("click", () => this.openProfile()); + document.getElementById("drawerClose")?.addEventListener("click", () => this.close("profile")); + document.getElementById("notificationsBtn")?.addEventListener("click", () => this.openNotification()); + document.getElementById("notificationDrawerClose")?.addEventListener("click", () => this.close("notification")); + document.getElementById("markAllRead")?.addEventListener("click", () => this.markAllNotificationsRead()); + document.getElementById("openTodoDrawer")?.addEventListener("click", () => this.openTodo()); + document.getElementById("todoDrawerBack")?.addEventListener("click", () => this.closeTodo()); + document.getElementById("addTodoBtn")?.addEventListener("click", () => this.openNewTodo()); + document.getElementById("newTodoDrawerBack")?.addEventListener("click", () => this.closeNewTodo()); + document.getElementById("cancelNewTodo")?.addEventListener("click", () => this.closeNewTodo()); + document.getElementById("saveNewTodo")?.addEventListener("click", () => this.closeNewTodo()); + this.overlay?.addEventListener("click", () => this.closeAll()); + document.addEventListener("keydown", (e) => { + if (e.key === "Escape") + this.closeAll(); + }); + this.todoDrawer?.addEventListener("click", (e) => this.handleTodoClick(e)); + document.addEventListener("click", (e) => this.handleVisibilityClick(e)); + } + handleTodoClick(e) { + const target = e.target; + const todoItem = target.closest("swp-todo-item"); + const checkbox = target.closest("swp-todo-checkbox"); + if (checkbox && todoItem) { + const isCompleted = todoItem.dataset.completed === "true"; + if (isCompleted) { + todoItem.removeAttribute("data-completed"); + } else { + todoItem.dataset.completed = "true"; + } + } + const sectionHeader = target.closest("swp-todo-section-header"); + if (sectionHeader) { + const section = sectionHeader.closest("swp-todo-section"); + section?.classList.toggle("collapsed"); + } + } + handleVisibilityClick(e) { + const target = e.target; + const option = target.closest("swp-visibility-option"); + if (option) { + document.querySelectorAll("swp-visibility-option").forEach((o) => o.classList.remove("active")); + option.classList.add("active"); + } + } +}; +__name(_DrawerController, "DrawerController"); +var DrawerController = _DrawerController; + +// wwwroot/ts/modules/theme.ts +var _ThemeController = class _ThemeController { + constructor() { + this.root = document.documentElement; + this.themeOptions = document.querySelectorAll("swp-theme-option"); + this.applyTheme(this.current); + this.updateUI(); + this.setupListeners(); + } + /** + * Get the current theme setting + */ + get current() { + const stored = localStorage.getItem(_ThemeController.STORAGE_KEY); + if (stored === "dark" || stored === "light" || stored === "system") { + return stored; + } + return "system"; + } + /** + * Check if dark mode is currently active + */ + get isDark() { + return this.root.classList.contains(_ThemeController.DARK_CLASS) || this.systemPrefersDark && !this.root.classList.contains(_ThemeController.LIGHT_CLASS); + } + /** + * Check if system prefers dark mode + */ + get systemPrefersDark() { + return window.matchMedia("(prefers-color-scheme: dark)").matches; + } + /** + * Set theme and persist preference + */ + set(theme) { + localStorage.setItem(_ThemeController.STORAGE_KEY, theme); + this.applyTheme(theme); + this.updateUI(); + } + /** + * Toggle between light and dark themes + */ + toggle() { + this.set(this.isDark ? "light" : "dark"); + } + applyTheme(theme) { + this.root.classList.remove(_ThemeController.DARK_CLASS, _ThemeController.LIGHT_CLASS); + if (theme === "dark") { + this.root.classList.add(_ThemeController.DARK_CLASS); + } else if (theme === "light") { + this.root.classList.add(_ThemeController.LIGHT_CLASS); + } + } + updateUI() { + if (!this.themeOptions) + return; + const darkActive = this.isDark; + this.themeOptions.forEach((option) => { + const theme = option.dataset.theme; + const isActive = theme === "dark" && darkActive || theme === "light" && !darkActive; + option.classList.toggle("active", isActive); + }); + } + setupListeners() { + this.themeOptions.forEach((option) => { + option.addEventListener("click", (e) => this.handleOptionClick(e)); + }); + window.matchMedia("(prefers-color-scheme: dark)").addEventListener("change", () => this.handleSystemChange()); + } + handleOptionClick(e) { + const target = e.target; + const option = target.closest("swp-theme-option"); + if (option) { + const theme = option.dataset.theme; + if (theme) { + this.set(theme); + } + } + } + handleSystemChange() { + if (this.current === "system") { + this.updateUI(); + } + } +}; +__name(_ThemeController, "ThemeController"); +_ThemeController.STORAGE_KEY = "theme-preference"; +_ThemeController.DARK_CLASS = "dark-mode"; +_ThemeController.LIGHT_CLASS = "light-mode"; +var ThemeController = _ThemeController; + +// wwwroot/ts/modules/search.ts +var _SearchController = class _SearchController { + constructor() { + this.input = null; + this.container = null; + this.input = document.getElementById("globalSearch"); + this.container = document.querySelector("swp-topbar-search"); + this.setupListeners(); + } + /** + * Get current search value + */ + get value() { + return this.input?.value ?? ""; + } + /** + * Set search value + */ + set value(val) { + if (this.input) { + this.input.value = val; + } + } + /** + * Focus the search input + */ + focus() { + this.input?.focus(); + } + /** + * Blur the search input + */ + blur() { + this.input?.blur(); + } + /** + * Clear the search input + */ + clear() { + this.value = ""; + } + setupListeners() { + document.addEventListener("keydown", (e) => this.handleKeyboard(e)); + if (this.input) { + this.input.addEventListener("input", (e) => this.handleInput(e)); + const form = this.input.closest("form"); + form?.addEventListener("submit", (e) => this.handleSubmit(e)); + } + } + handleKeyboard(e) { + if ((e.metaKey || e.ctrlKey) && e.key === "k") { + e.preventDefault(); + this.focus(); + return; + } + if (e.key === "Escape" && document.activeElement === this.input) { + this.blur(); + } + } + handleInput(e) { + const target = e.target; + const query = target.value.trim(); + document.dispatchEvent(new CustomEvent("app:search", { + detail: { query }, + bubbles: true + })); + } + handleSubmit(e) { + e.preventDefault(); + const query = this.value.trim(); + if (!query) + return; + document.dispatchEvent(new CustomEvent("app:search-submit", { + detail: { query }, + bubbles: true + })); + } +}; +__name(_SearchController, "SearchController"); +var SearchController = _SearchController; + +// wwwroot/ts/modules/lockscreen.ts +var _LockScreenController = class _LockScreenController { + constructor(drawers) { + // Demo PIN + this.lockScreen = null; + this.pinInput = null; + this.pinKeypad = null; + this.lockTimeEl = null; + this.pinDigits = null; + this.currentPin = ""; + this.drawers = null; + this.drawers = drawers ?? null; + this.lockScreen = document.getElementById("lockScreen"); + this.pinInput = document.getElementById("pinInput"); + this.pinKeypad = document.getElementById("pinKeypad"); + this.lockTimeEl = document.getElementById("lockTime"); + this.pinDigits = this.pinInput?.querySelectorAll("swp-pin-digit") ?? null; + this.setupListeners(); + } + /** + * Check if lock screen is active + */ + get isActive() { + return this.lockScreen?.classList.contains("active") ?? false; + } + /** + * Show the lock screen + */ + show() { + this.drawers?.closeAll(); + if (this.lockScreen) { + this.lockScreen.classList.add("active"); + document.body.style.overflow = "hidden"; + } + this.currentPin = ""; + this.updateDisplay(); + if (this.lockTimeEl) { + this.lockTimeEl.textContent = `L\xE5st kl. ${this.formatTime()}`; + } + } + /** + * Hide the lock screen + */ + hide() { + if (this.lockScreen) { + this.lockScreen.classList.remove("active"); + document.body.style.overflow = ""; + } + this.currentPin = ""; + this.updateDisplay(); + } + formatTime() { + const now = /* @__PURE__ */ new Date(); + const hours = now.getHours().toString().padStart(2, "0"); + const minutes = now.getMinutes().toString().padStart(2, "0"); + return `${hours}:${minutes}`; + } + updateDisplay() { + if (!this.pinDigits) + return; + this.pinDigits.forEach((digit, index) => { + digit.classList.remove("filled", "error"); + if (index < this.currentPin.length) { + digit.textContent = "\u2022"; + digit.classList.add("filled"); + } else { + digit.textContent = ""; + } + }); + } + showError() { + if (!this.pinDigits) + return; + this.pinDigits.forEach((digit) => digit.classList.add("error")); + this.pinInput?.classList.add("shake"); + setTimeout(() => { + this.currentPin = ""; + this.updateDisplay(); + this.pinInput?.classList.remove("shake"); + }, 500); + } + verify() { + if (this.currentPin === _LockScreenController.CORRECT_PIN) { + this.hide(); + } else { + this.showError(); + } + } + addDigit(digit) { + if (this.currentPin.length >= 4) + return; + this.currentPin += digit; + this.updateDisplay(); + if (this.currentPin.length === 4) { + setTimeout(() => this.verify(), 200); + } + } + removeDigit() { + if (this.currentPin.length === 0) + return; + this.currentPin = this.currentPin.slice(0, -1); + this.updateDisplay(); + } + clearPin() { + this.currentPin = ""; + this.updateDisplay(); + } + setupListeners() { + this.pinKeypad?.addEventListener("click", (e) => this.handleKeypadClick(e)); + document.addEventListener("keydown", (e) => this.handleKeyboard(e)); + document.querySelector("swp-side-menu-action.lock")?.addEventListener("click", () => this.show()); + } + handleKeypadClick(e) { + const target = e.target; + const key = target.closest("swp-pin-key"); + if (!key) + return; + const digit = key.dataset.digit; + const action = key.dataset.action; + if (digit) { + this.addDigit(digit); + } else if (action === "backspace") { + this.removeDigit(); + } else if (action === "clear") { + this.clearPin(); + } + } + handleKeyboard(e) { + if (!this.isActive) + return; + e.preventDefault(); + if (e.key >= "0" && e.key <= "9") { + this.addDigit(e.key); + } else if (e.key === "Backspace") { + this.removeDigit(); + } else if (e.key === "Escape") { + this.clearPin(); + } + } +}; +__name(_LockScreenController, "LockScreenController"); +_LockScreenController.CORRECT_PIN = "1234"; +var LockScreenController = _LockScreenController; + +// wwwroot/ts/app.ts +var _App = class _App { + constructor() { + this.sidebar = new SidebarController(); + this.drawers = new DrawerController(); + this.theme = new ThemeController(); + this.search = new SearchController(); + this.lockScreen = new LockScreenController(this.drawers); + } +}; +__name(_App, "App"); +var App = _App; +var app; +function init() { + app = new App(); + if (typeof window !== "undefined") { + window.app = app; + } +} +__name(init, "init"); +if (document.readyState === "loading") { + document.addEventListener("DOMContentLoaded", init); +} else { + init(); +} +var app_default = App; +export { + App, + app, + app_default as default +}; +//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../ts/modules/sidebar.ts", "../ts/modules/drawers.ts", "../ts/modules/theme.ts", "../ts/modules/search.ts", "../ts/modules/lockscreen.ts", "../ts/app.ts"],
  "sourcesContent": ["/**\n * Sidebar Controller\n *\n * Handles sidebar collapse/expand and tooltip functionality\n */\n\nexport class SidebarController {\n  private menuToggle: HTMLElement | null = null;\n  private appLayout: HTMLElement | null = null;\n  private menuTooltip: HTMLElement | null = null;\n\n  constructor() {\n    this.menuToggle = document.getElementById('menuToggle');\n    this.appLayout = document.querySelector('swp-app-layout');\n    this.menuTooltip = document.getElementById('menuTooltip');\n\n    this.setupListeners();\n    this.setupTooltips();\n    this.restoreState();\n  }\n\n  /**\n   * Check if sidebar is collapsed\n   */\n  get isCollapsed(): boolean {\n    return this.appLayout?.classList.contains('menu-collapsed') ?? false;\n  }\n\n  /**\n   * Toggle sidebar collapsed state\n   */\n  toggle(): void {\n    if (!this.appLayout) return;\n\n    this.appLayout.classList.toggle('menu-collapsed');\n    localStorage.setItem('sidebar-collapsed', String(this.isCollapsed));\n  }\n\n  /**\n   * Collapse the sidebar\n   */\n  collapse(): void {\n    this.appLayout?.classList.add('menu-collapsed');\n    localStorage.setItem('sidebar-collapsed', 'true');\n  }\n\n  /**\n   * Expand the sidebar\n   */\n  expand(): void {\n    this.appLayout?.classList.remove('menu-collapsed');\n    localStorage.setItem('sidebar-collapsed', 'false');\n  }\n\n  private setupListeners(): void {\n    this.menuToggle?.addEventListener('click', () => this.toggle());\n  }\n\n  private setupTooltips(): void {\n    if (!this.menuTooltip) return;\n\n    const menuItems = document.querySelectorAll<HTMLElement>('swp-side-menu-item[data-tooltip]');\n\n    menuItems.forEach(item => {\n      item.addEventListener('mouseenter', () => this.showTooltip(item));\n      item.addEventListener('mouseleave', () => this.hideTooltip());\n    });\n  }\n\n  private showTooltip(item: HTMLElement): void {\n    if (!this.isCollapsed || !this.menuTooltip) return;\n\n    const rect = item.getBoundingClientRect();\n    const tooltipText = item.dataset.tooltip;\n\n    if (!tooltipText) return;\n\n    this.menuTooltip.textContent = tooltipText;\n    this.menuTooltip.style.left = `${rect.right + 8}px`;\n    this.menuTooltip.style.top = `${rect.top + rect.height / 2}px`;\n    this.menuTooltip.style.transform = 'translateY(-50%)';\n    this.menuTooltip.showPopover();\n  }\n\n  private hideTooltip(): void {\n    this.menuTooltip?.hidePopover();\n  }\n\n  private restoreState(): void {\n    if (!this.appLayout) return;\n\n    if (localStorage.getItem('sidebar-collapsed') === 'true') {\n      this.appLayout.classList.add('menu-collapsed');\n    }\n  }\n}\n", "/**\n * Drawer Controller\n *\n * Handles all drawer functionality including profile, notifications, and todo drawers\n */\n\nexport type DrawerName = 'profile' | 'notification' | 'todo' | 'newTodo';\n\nexport class DrawerController {\n  private profileDrawer: HTMLElement | null = null;\n  private notificationDrawer: HTMLElement | null = null;\n  private todoDrawer: HTMLElement | null = null;\n  private newTodoDrawer: HTMLElement | null = null;\n  private overlay: HTMLElement | null = null;\n  private activeDrawer: DrawerName | null = null;\n\n  constructor() {\n    this.profileDrawer = document.getElementById('profileDrawer');\n    this.notificationDrawer = document.getElementById('notificationDrawer');\n    this.todoDrawer = document.getElementById('todoDrawer');\n    this.newTodoDrawer = document.getElementById('newTodoDrawer');\n    this.overlay = document.getElementById('drawerOverlay');\n\n    this.setupListeners();\n  }\n\n  /**\n   * Get currently active drawer name\n   */\n  get active(): DrawerName | null {\n    return this.activeDrawer;\n  }\n\n  /**\n   * Open a drawer by name\n   */\n  open(name: DrawerName): void {\n    this.closeAll();\n\n    const drawer = this.getDrawer(name);\n    if (drawer && this.overlay) {\n      drawer.classList.add('active');\n      this.overlay.classList.add('active');\n      document.body.style.overflow = 'hidden';\n      this.activeDrawer = name;\n    }\n  }\n\n  /**\n   * Close a specific drawer\n   */\n  close(name: DrawerName): void {\n    const drawer = this.getDrawer(name);\n    drawer?.classList.remove('active');\n\n    // Only hide overlay if no drawers are active\n    if (this.overlay && !document.querySelector('.active[class*=\"drawer\"]')) {\n      this.overlay.classList.remove('active');\n      document.body.style.overflow = '';\n    }\n\n    if (this.activeDrawer === name) {\n      this.activeDrawer = null;\n    }\n  }\n\n  /**\n   * Close all drawers\n   */\n  closeAll(): void {\n    [this.profileDrawer, this.notificationDrawer, this.todoDrawer, this.newTodoDrawer]\n      .forEach(drawer => drawer?.classList.remove('active'));\n\n    this.overlay?.classList.remove('active');\n    document.body.style.overflow = '';\n    this.activeDrawer = null;\n  }\n\n  /**\n   * Open profile drawer\n   */\n  openProfile(): void {\n    this.open('profile');\n  }\n\n  /**\n   * Open notification drawer\n   */\n  openNotification(): void {\n    this.open('notification');\n  }\n\n  /**\n   * Open todo drawer (slides on top of profile)\n   */\n  openTodo(): void {\n    this.todoDrawer?.classList.add('active');\n  }\n\n  /**\n   * Close todo drawer\n   */\n  closeTodo(): void {\n    this.todoDrawer?.classList.remove('active');\n    this.closeNewTodo();\n  }\n\n  /**\n   * Open new todo drawer\n   */\n  openNewTodo(): void {\n    this.newTodoDrawer?.classList.add('active');\n  }\n\n  /**\n   * Close new todo drawer\n   */\n  closeNewTodo(): void {\n    this.newTodoDrawer?.classList.remove('active');\n  }\n\n  /**\n   * Mark all notifications as read\n   */\n  markAllNotificationsRead(): void {\n    if (!this.notificationDrawer) return;\n\n    const unreadItems = this.notificationDrawer.querySelectorAll<HTMLElement>(\n      'swp-notification-item[data-unread=\"true\"]'\n    );\n    unreadItems.forEach(item => item.removeAttribute('data-unread'));\n\n    const badge = document.querySelector<HTMLElement>('swp-notification-badge');\n    if (badge) {\n      badge.style.display = 'none';\n    }\n  }\n\n  private getDrawer(name: DrawerName): HTMLElement | null {\n    switch (name) {\n      case 'profile': return this.profileDrawer;\n      case 'notification': return this.notificationDrawer;\n      case 'todo': return this.todoDrawer;\n      case 'newTodo': return this.newTodoDrawer;\n    }\n  }\n\n  private setupListeners(): void {\n    // Profile drawer triggers\n    document.getElementById('profileTrigger')\n      ?.addEventListener('click', () => this.openProfile());\n    document.getElementById('drawerClose')\n      ?.addEventListener('click', () => this.close('profile'));\n\n    // Notification drawer triggers\n    document.getElementById('notificationsBtn')\n      ?.addEventListener('click', () => this.openNotification());\n    document.getElementById('notificationDrawerClose')\n      ?.addEventListener('click', () => this.close('notification'));\n    document.getElementById('markAllRead')\n      ?.addEventListener('click', () => this.markAllNotificationsRead());\n\n    // Todo drawer triggers\n    document.getElementById('openTodoDrawer')\n      ?.addEventListener('click', () => this.openTodo());\n    document.getElementById('todoDrawerBack')\n      ?.addEventListener('click', () => this.closeTodo());\n\n    // New todo drawer triggers\n    document.getElementById('addTodoBtn')\n      ?.addEventListener('click', () => this.openNewTodo());\n    document.getElementById('newTodoDrawerBack')\n      ?.addEventListener('click', () => this.closeNewTodo());\n    document.getElementById('cancelNewTodo')\n      ?.addEventListener('click', () => this.closeNewTodo());\n    document.getElementById('saveNewTodo')\n      ?.addEventListener('click', () => this.closeNewTodo());\n\n    // Overlay click closes all\n    this.overlay?.addEventListener('click', () => this.closeAll());\n\n    // Escape key closes all\n    document.addEventListener('keydown', (e: KeyboardEvent) => {\n      if (e.key === 'Escape') this.closeAll();\n    });\n\n    // Todo interactions\n    this.todoDrawer?.addEventListener('click', (e) => this.handleTodoClick(e));\n\n    // Visibility options\n    document.addEventListener('click', (e) => this.handleVisibilityClick(e));\n  }\n\n  private handleTodoClick(e: Event): void {\n    const target = e.target as HTMLElement;\n    const todoItem = target.closest<HTMLElement>('swp-todo-item');\n    const checkbox = target.closest<HTMLElement>('swp-todo-checkbox');\n\n    if (checkbox && todoItem) {\n      const isCompleted = todoItem.dataset.completed === 'true';\n      if (isCompleted) {\n        todoItem.removeAttribute('data-completed');\n      } else {\n        todoItem.dataset.completed = 'true';\n      }\n    }\n\n    // Toggle section collapse\n    const sectionHeader = target.closest<HTMLElement>('swp-todo-section-header');\n    if (sectionHeader) {\n      const section = sectionHeader.closest<HTMLElement>('swp-todo-section');\n      section?.classList.toggle('collapsed');\n    }\n  }\n\n  private handleVisibilityClick(e: Event): void {\n    const target = e.target as HTMLElement;\n    const option = target.closest<HTMLElement>('swp-visibility-option');\n\n    if (option) {\n      document.querySelectorAll<HTMLElement>('swp-visibility-option')\n        .forEach(o => o.classList.remove('active'));\n      option.classList.add('active');\n    }\n  }\n}\n", "/**\n * Theme Controller\n *\n * Handles dark/light mode switching and system preference detection\n */\n\nexport type Theme = 'light' | 'dark' | 'system';\n\nexport class ThemeController {\n  private static readonly STORAGE_KEY = 'theme-preference';\n  private static readonly DARK_CLASS = 'dark-mode';\n  private static readonly LIGHT_CLASS = 'light-mode';\n\n  private root: HTMLElement;\n  private themeOptions: NodeListOf<HTMLElement>;\n\n  constructor() {\n    this.root = document.documentElement;\n    this.themeOptions = document.querySelectorAll<HTMLElement>('swp-theme-option');\n\n    this.applyTheme(this.current);\n    this.updateUI();\n    this.setupListeners();\n  }\n\n  /**\n   * Get the current theme setting\n   */\n  get current(): Theme {\n    const stored = localStorage.getItem(ThemeController.STORAGE_KEY) as Theme | null;\n    if (stored === 'dark' || stored === 'light' || stored === 'system') {\n      return stored;\n    }\n    return 'system';\n  }\n\n  /**\n   * Check if dark mode is currently active\n   */\n  get isDark(): boolean {\n    return this.root.classList.contains(ThemeController.DARK_CLASS) ||\n      (this.systemPrefersDark && !this.root.classList.contains(ThemeController.LIGHT_CLASS));\n  }\n\n  /**\n   * Check if system prefers dark mode\n   */\n  get systemPrefersDark(): boolean {\n    return window.matchMedia('(prefers-color-scheme: dark)').matches;\n  }\n\n  /**\n   * Set theme and persist preference\n   */\n  set(theme: Theme): void {\n    localStorage.setItem(ThemeController.STORAGE_KEY, theme);\n    this.applyTheme(theme);\n    this.updateUI();\n  }\n\n  /**\n   * Toggle between light and dark themes\n   */\n  toggle(): void {\n    this.set(this.isDark ? 'light' : 'dark');\n  }\n\n  private applyTheme(theme: Theme): void {\n    this.root.classList.remove(ThemeController.DARK_CLASS, ThemeController.LIGHT_CLASS);\n\n    if (theme === 'dark') {\n      this.root.classList.add(ThemeController.DARK_CLASS);\n    } else if (theme === 'light') {\n      this.root.classList.add(ThemeController.LIGHT_CLASS);\n    }\n    // 'system' leaves both classes off, letting CSS media query handle it\n  }\n\n  private updateUI(): void {\n    if (!this.themeOptions) return;\n\n    const darkActive = this.isDark;\n\n    this.themeOptions.forEach(option => {\n      const theme = option.dataset.theme as Theme;\n      const isActive = (theme === 'dark' && darkActive) || (theme === 'light' && !darkActive);\n      option.classList.toggle('active', isActive);\n    });\n  }\n\n  private setupListeners(): void {\n    // Theme option clicks\n    this.themeOptions.forEach(option => {\n      option.addEventListener('click', (e) => this.handleOptionClick(e));\n    });\n\n    // System theme changes\n    window.matchMedia('(prefers-color-scheme: dark)')\n      .addEventListener('change', () => this.handleSystemChange());\n  }\n\n  private handleOptionClick(e: Event): void {\n    const target = e.target as HTMLElement;\n    const option = target.closest<HTMLElement>('swp-theme-option');\n\n    if (option) {\n      const theme = option.dataset.theme as Theme;\n      if (theme) {\n        this.set(theme);\n      }\n    }\n  }\n\n  private handleSystemChange(): void {\n    // Only react to system changes if we're using system preference\n    if (this.current === 'system') {\n      this.updateUI();\n    }\n  }\n}\n", "/**\n * Search Controller\n *\n * Handles global search functionality and keyboard shortcuts\n */\n\nexport class SearchController {\n  private input: HTMLInputElement | null = null;\n  private container: HTMLElement | null = null;\n\n  constructor() {\n    this.input = document.getElementById('globalSearch') as HTMLInputElement | null;\n    this.container = document.querySelector<HTMLElement>('swp-topbar-search');\n\n    this.setupListeners();\n  }\n\n  /**\n   * Get current search value\n   */\n  get value(): string {\n    return this.input?.value ?? '';\n  }\n\n  /**\n   * Set search value\n   */\n  set value(val: string) {\n    if (this.input) {\n      this.input.value = val;\n    }\n  }\n\n  /**\n   * Focus the search input\n   */\n  focus(): void {\n    this.input?.focus();\n  }\n\n  /**\n   * Blur the search input\n   */\n  blur(): void {\n    this.input?.blur();\n  }\n\n  /**\n   * Clear the search input\n   */\n  clear(): void {\n    this.value = '';\n  }\n\n  private setupListeners(): void {\n    // Keyboard shortcuts\n    document.addEventListener('keydown', (e) => this.handleKeyboard(e));\n\n    // Input handlers\n    if (this.input) {\n      this.input.addEventListener('input', (e) => this.handleInput(e));\n\n      // Prevent form submission if wrapped in form\n      const form = this.input.closest('form');\n      form?.addEventListener('submit', (e) => this.handleSubmit(e));\n    }\n  }\n\n  private handleKeyboard(e: KeyboardEvent): void {\n    // Cmd/Ctrl + K to focus search\n    if ((e.metaKey || e.ctrlKey) && e.key === 'k') {\n      e.preventDefault();\n      this.focus();\n      return;\n    }\n\n    // Escape to blur search when focused\n    if (e.key === 'Escape' && document.activeElement === this.input) {\n      this.blur();\n    }\n  }\n\n  private handleInput(e: Event): void {\n    const target = e.target as HTMLInputElement;\n    const query = target.value.trim();\n\n    // Emit custom event for search\n    document.dispatchEvent(new CustomEvent('app:search', {\n      detail: { query },\n      bubbles: true\n    }));\n  }\n\n  private handleSubmit(e: Event): void {\n    e.preventDefault();\n\n    const query = this.value.trim();\n    if (!query) return;\n\n    // Emit custom event for search submit\n    document.dispatchEvent(new CustomEvent('app:search-submit', {\n      detail: { query },\n      bubbles: true\n    }));\n  }\n}\n", "/**\n * Lock Screen Controller\n *\n * Handles PIN-based lock screen functionality\n */\n\nimport { DrawerController } from './drawers';\n\nexport class LockScreenController {\n  private static readonly CORRECT_PIN = '1234'; // Demo PIN\n\n  private lockScreen: HTMLElement | null = null;\n  private pinInput: HTMLElement | null = null;\n  private pinKeypad: HTMLElement | null = null;\n  private lockTimeEl: HTMLElement | null = null;\n  private pinDigits: NodeListOf<HTMLElement> | null = null;\n  private currentPin = '';\n  private drawers: DrawerController | null = null;\n\n  constructor(drawers?: DrawerController) {\n    this.drawers = drawers ?? null;\n    this.lockScreen = document.getElementById('lockScreen');\n    this.pinInput = document.getElementById('pinInput');\n    this.pinKeypad = document.getElementById('pinKeypad');\n    this.lockTimeEl = document.getElementById('lockTime');\n    this.pinDigits = this.pinInput?.querySelectorAll<HTMLElement>('swp-pin-digit') ?? null;\n\n    this.setupListeners();\n  }\n\n  /**\n   * Check if lock screen is active\n   */\n  get isActive(): boolean {\n    return this.lockScreen?.classList.contains('active') ?? false;\n  }\n\n  /**\n   * Show the lock screen\n   */\n  show(): void {\n    this.drawers?.closeAll();\n\n    if (this.lockScreen) {\n      this.lockScreen.classList.add('active');\n      document.body.style.overflow = 'hidden';\n    }\n\n    this.currentPin = '';\n    this.updateDisplay();\n\n    // Update lock time\n    if (this.lockTimeEl) {\n      this.lockTimeEl.textContent = `L\u00E5st kl. ${this.formatTime()}`;\n    }\n  }\n\n  /**\n   * Hide the lock screen\n   */\n  hide(): void {\n    if (this.lockScreen) {\n      this.lockScreen.classList.remove('active');\n      document.body.style.overflow = '';\n    }\n\n    this.currentPin = '';\n    this.updateDisplay();\n  }\n\n  private formatTime(): string {\n    const now = new Date();\n    const hours = now.getHours().toString().padStart(2, '0');\n    const minutes = now.getMinutes().toString().padStart(2, '0');\n    return `${hours}:${minutes}`;\n  }\n\n  private updateDisplay(): void {\n    if (!this.pinDigits) return;\n\n    this.pinDigits.forEach((digit, index) => {\n      digit.classList.remove('filled', 'error');\n      if (index < this.currentPin.length) {\n        digit.textContent = '\u2022';\n        digit.classList.add('filled');\n      } else {\n        digit.textContent = '';\n      }\n    });\n  }\n\n  private showError(): void {\n    if (!this.pinDigits) return;\n\n    this.pinDigits.forEach(digit => digit.classList.add('error'));\n\n    // Shake animation\n    this.pinInput?.classList.add('shake');\n\n    setTimeout(() => {\n      this.currentPin = '';\n      this.updateDisplay();\n      this.pinInput?.classList.remove('shake');\n    }, 500);\n  }\n\n  private verify(): void {\n    if (this.currentPin === LockScreenController.CORRECT_PIN) {\n      this.hide();\n    } else {\n      this.showError();\n    }\n  }\n\n  private addDigit(digit: string): void {\n    if (this.currentPin.length >= 4) return;\n\n    this.currentPin += digit;\n    this.updateDisplay();\n\n    // Auto-verify when 4 digits entered\n    if (this.currentPin.length === 4) {\n      setTimeout(() => this.verify(), 200);\n    }\n  }\n\n  private removeDigit(): void {\n    if (this.currentPin.length === 0) return;\n    this.currentPin = this.currentPin.slice(0, -1);\n    this.updateDisplay();\n  }\n\n  private clearPin(): void {\n    this.currentPin = '';\n    this.updateDisplay();\n  }\n\n  private setupListeners(): void {\n    // Keypad click handler\n    this.pinKeypad?.addEventListener('click', (e) => this.handleKeypadClick(e));\n\n    // Keyboard input\n    document.addEventListener('keydown', (e) => this.handleKeyboard(e));\n\n    // Lock button in sidebar\n    document.querySelector<HTMLElement>('swp-side-menu-action.lock')\n      ?.addEventListener('click', () => this.show());\n  }\n\n  private handleKeypadClick(e: Event): void {\n    const target = e.target as HTMLElement;\n    const key = target.closest<HTMLElement>('swp-pin-key');\n\n    if (!key) return;\n\n    const digit = key.dataset.digit;\n    const action = key.dataset.action;\n\n    if (digit) {\n      this.addDigit(digit);\n    } else if (action === 'backspace') {\n      this.removeDigit();\n    } else if (action === 'clear') {\n      this.clearPin();\n    }\n  }\n\n  private handleKeyboard(e: KeyboardEvent): void {\n    if (!this.isActive) return;\n\n    // Prevent default to avoid other interactions\n    e.preventDefault();\n\n    if (e.key >= '0' && e.key <= '9') {\n      this.addDigit(e.key);\n    } else if (e.key === 'Backspace') {\n      this.removeDigit();\n    } else if (e.key === 'Escape') {\n      this.clearPin();\n    }\n  }\n}\n", "/**\n * Salon OS App\n *\n * Main application class that orchestrates all UI controllers\n */\n\nimport { SidebarController } from './modules/sidebar';\nimport { DrawerController } from './modules/drawers';\nimport { ThemeController } from './modules/theme';\nimport { SearchController } from './modules/search';\nimport { LockScreenController } from './modules/lockscreen';\n\n/**\n * Main application class\n */\nexport class App {\n  readonly sidebar: SidebarController;\n  readonly drawers: DrawerController;\n  readonly theme: ThemeController;\n  readonly search: SearchController;\n  readonly lockScreen: LockScreenController;\n\n  constructor() {\n    // Initialize controllers\n    this.sidebar = new SidebarController();\n    this.drawers = new DrawerController();\n    this.theme = new ThemeController();\n    this.search = new SearchController();\n    this.lockScreen = new LockScreenController(this.drawers);\n  }\n}\n\n/**\n * Global app instance\n */\nlet app: App;\n\n/**\n * Initialize the application\n */\nfunction init(): void {\n  app = new App();\n\n  // Expose to window for debugging\n  if (typeof window !== 'undefined') {\n    (window as unknown as { app: App }).app = app;\n  }\n}\n\n// Wait for DOM ready\nif (document.readyState === 'loading') {\n  document.addEventListener('DOMContentLoaded', init);\n} else {\n  init();\n}\n\nexport { app };\nexport default App;\n"],
  "mappings": "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",
  "names": []
}
 diff --git a/PlanTempus.Application/wwwroot/ts/app.ts b/PlanTempus.Application/wwwroot/ts/app.ts new file mode 100644 index 0000000..16735f5 --- /dev/null +++ b/PlanTempus.Application/wwwroot/ts/app.ts @@ -0,0 +1,58 @@ +/** + * Salon OS App + * + * Main application class that orchestrates all UI controllers + */ + +import { SidebarController } from './modules/sidebar'; +import { DrawerController } from './modules/drawers'; +import { ThemeController } from './modules/theme'; +import { SearchController } from './modules/search'; +import { LockScreenController } from './modules/lockscreen'; + +/** + * Main application class + */ +export class App { + readonly sidebar: SidebarController; + readonly drawers: DrawerController; + readonly theme: ThemeController; + readonly search: SearchController; + readonly lockScreen: LockScreenController; + + constructor() { + // Initialize controllers + this.sidebar = new SidebarController(); + this.drawers = new DrawerController(); + this.theme = new ThemeController(); + this.search = new SearchController(); + this.lockScreen = new LockScreenController(this.drawers); + } +} + +/** + * Global app instance + */ +let app: App; + +/** + * Initialize the application + */ +function init(): void { + app = new App(); + + // Expose to window for debugging + if (typeof window !== 'undefined') { + (window as unknown as { app: App }).app = app; + } +} + +// Wait for DOM ready +if (document.readyState === 'loading') { + document.addEventListener('DOMContentLoaded', init); +} else { + init(); +} + +export { app }; +export default App; diff --git a/PlanTempus.Application/wwwroot/ts/modules/drawers.ts b/PlanTempus.Application/wwwroot/ts/modules/drawers.ts new file mode 100644 index 0000000..7ee36ad --- /dev/null +++ b/PlanTempus.Application/wwwroot/ts/modules/drawers.ts @@ -0,0 +1,226 @@ +/** + * Drawer Controller + * + * Handles all drawer functionality including profile, notifications, and todo drawers + */ + +export type DrawerName = 'profile' | 'notification' | 'todo' | 'newTodo'; + +export class DrawerController { + private profileDrawer: HTMLElement | null = null; + private notificationDrawer: HTMLElement | null = null; + private todoDrawer: HTMLElement | null = null; + private newTodoDrawer: HTMLElement | null = null; + private overlay: HTMLElement | null = null; + private activeDrawer: DrawerName | null = null; + + constructor() { + this.profileDrawer = document.getElementById('profileDrawer'); + this.notificationDrawer = document.getElementById('notificationDrawer'); + this.todoDrawer = document.getElementById('todoDrawer'); + this.newTodoDrawer = document.getElementById('newTodoDrawer'); + this.overlay = document.getElementById('drawerOverlay'); + + this.setupListeners(); + } + + /** + * Get currently active drawer name + */ + get active(): DrawerName | null { + return this.activeDrawer; + } + + /** + * Open a drawer by name + */ + open(name: DrawerName): void { + this.closeAll(); + + const drawer = this.getDrawer(name); + if (drawer && this.overlay) { + drawer.classList.add('active'); + this.overlay.classList.add('active'); + document.body.style.overflow = 'hidden'; + this.activeDrawer = name; + } + } + + /** + * Close a specific drawer + */ + close(name: DrawerName): void { + const drawer = this.getDrawer(name); + drawer?.classList.remove('active'); + + // Only hide overlay if no drawers are active + if (this.overlay && !document.querySelector('.active[class*="drawer"]')) { + this.overlay.classList.remove('active'); + document.body.style.overflow = ''; + } + + if (this.activeDrawer === name) { + this.activeDrawer = null; + } + } + + /** + * Close all drawers + */ + closeAll(): void { + [this.profileDrawer, this.notificationDrawer, this.todoDrawer, this.newTodoDrawer] + .forEach(drawer => drawer?.classList.remove('active')); + + this.overlay?.classList.remove('active'); + document.body.style.overflow = ''; + this.activeDrawer = null; + } + + /** + * Open profile drawer + */ + openProfile(): void { + this.open('profile'); + } + + /** + * Open notification drawer + */ + openNotification(): void { + this.open('notification'); + } + + /** + * Open todo drawer (slides on top of profile) + */ + openTodo(): void { + this.todoDrawer?.classList.add('active'); + } + + /** + * Close todo drawer + */ + closeTodo(): void { + this.todoDrawer?.classList.remove('active'); + this.closeNewTodo(); + } + + /** + * Open new todo drawer + */ + openNewTodo(): void { + this.newTodoDrawer?.classList.add('active'); + } + + /** + * Close new todo drawer + */ + closeNewTodo(): void { + this.newTodoDrawer?.classList.remove('active'); + } + + /** + * Mark all notifications as read + */ + markAllNotificationsRead(): void { + if (!this.notificationDrawer) return; + + const unreadItems = this.notificationDrawer.querySelectorAll( + 'swp-notification-item[data-unread="true"]' + ); + unreadItems.forEach(item => item.removeAttribute('data-unread')); + + const badge = document.querySelector('swp-notification-badge'); + if (badge) { + badge.style.display = 'none'; + } + } + + private getDrawer(name: DrawerName): HTMLElement | null { + switch (name) { + case 'profile': return this.profileDrawer; + case 'notification': return this.notificationDrawer; + case 'todo': return this.todoDrawer; + case 'newTodo': return this.newTodoDrawer; + } + } + + private setupListeners(): void { + // Profile drawer triggers + document.getElementById('profileTrigger') + ?.addEventListener('click', () => this.openProfile()); + document.getElementById('drawerClose') + ?.addEventListener('click', () => this.close('profile')); + + // Notification drawer triggers + document.getElementById('notificationsBtn') + ?.addEventListener('click', () => this.openNotification()); + document.getElementById('notificationDrawerClose') + ?.addEventListener('click', () => this.close('notification')); + document.getElementById('markAllRead') + ?.addEventListener('click', () => this.markAllNotificationsRead()); + + // Todo drawer triggers + document.getElementById('openTodoDrawer') + ?.addEventListener('click', () => this.openTodo()); + document.getElementById('todoDrawerBack') + ?.addEventListener('click', () => this.closeTodo()); + + // New todo drawer triggers + document.getElementById('addTodoBtn') + ?.addEventListener('click', () => this.openNewTodo()); + document.getElementById('newTodoDrawerBack') + ?.addEventListener('click', () => this.closeNewTodo()); + document.getElementById('cancelNewTodo') + ?.addEventListener('click', () => this.closeNewTodo()); + document.getElementById('saveNewTodo') + ?.addEventListener('click', () => this.closeNewTodo()); + + // Overlay click closes all + this.overlay?.addEventListener('click', () => this.closeAll()); + + // Escape key closes all + document.addEventListener('keydown', (e: KeyboardEvent) => { + if (e.key === 'Escape') this.closeAll(); + }); + + // Todo interactions + this.todoDrawer?.addEventListener('click', (e) => this.handleTodoClick(e)); + + // Visibility options + document.addEventListener('click', (e) => this.handleVisibilityClick(e)); + } + + private handleTodoClick(e: Event): void { + const target = e.target as HTMLElement; + const todoItem = target.closest('swp-todo-item'); + const checkbox = target.closest('swp-todo-checkbox'); + + if (checkbox && todoItem) { + const isCompleted = todoItem.dataset.completed === 'true'; + if (isCompleted) { + todoItem.removeAttribute('data-completed'); + } else { + todoItem.dataset.completed = 'true'; + } + } + + // Toggle section collapse + const sectionHeader = target.closest('swp-todo-section-header'); + if (sectionHeader) { + const section = sectionHeader.closest('swp-todo-section'); + section?.classList.toggle('collapsed'); + } + } + + private handleVisibilityClick(e: Event): void { + const target = e.target as HTMLElement; + const option = target.closest('swp-visibility-option'); + + if (option) { + document.querySelectorAll('swp-visibility-option') + .forEach(o => o.classList.remove('active')); + option.classList.add('active'); + } + } +} diff --git a/PlanTempus.Application/wwwroot/ts/modules/lockscreen.ts b/PlanTempus.Application/wwwroot/ts/modules/lockscreen.ts new file mode 100644 index 0000000..f6a5878 --- /dev/null +++ b/PlanTempus.Application/wwwroot/ts/modules/lockscreen.ts @@ -0,0 +1,182 @@ +/** + * Lock Screen Controller + * + * Handles PIN-based lock screen functionality + */ + +import { DrawerController } from './drawers'; + +export class LockScreenController { + private static readonly CORRECT_PIN = '1234'; // Demo PIN + + private lockScreen: HTMLElement | null = null; + private pinInput: HTMLElement | null = null; + private pinKeypad: HTMLElement | null = null; + private lockTimeEl: HTMLElement | null = null; + private pinDigits: NodeListOf | null = null; + private currentPin = ''; + private drawers: DrawerController | null = null; + + constructor(drawers?: DrawerController) { + this.drawers = drawers ?? null; + this.lockScreen = document.getElementById('lockScreen'); + this.pinInput = document.getElementById('pinInput'); + this.pinKeypad = document.getElementById('pinKeypad'); + this.lockTimeEl = document.getElementById('lockTime'); + this.pinDigits = this.pinInput?.querySelectorAll('swp-pin-digit') ?? null; + + this.setupListeners(); + } + + /** + * Check if lock screen is active + */ + get isActive(): boolean { + return this.lockScreen?.classList.contains('active') ?? false; + } + + /** + * Show the lock screen + */ + show(): void { + this.drawers?.closeAll(); + + if (this.lockScreen) { + this.lockScreen.classList.add('active'); + document.body.style.overflow = 'hidden'; + } + + this.currentPin = ''; + this.updateDisplay(); + + // Update lock time + if (this.lockTimeEl) { + this.lockTimeEl.textContent = `Låst kl. ${this.formatTime()}`; + } + } + + /** + * Hide the lock screen + */ + hide(): void { + if (this.lockScreen) { + this.lockScreen.classList.remove('active'); + document.body.style.overflow = ''; + } + + this.currentPin = ''; + this.updateDisplay(); + } + + private formatTime(): string { + const now = new Date(); + const hours = now.getHours().toString().padStart(2, '0'); + const minutes = now.getMinutes().toString().padStart(2, '0'); + return `${hours}:${minutes}`; + } + + private updateDisplay(): void { + if (!this.pinDigits) return; + + this.pinDigits.forEach((digit, index) => { + digit.classList.remove('filled', 'error'); + if (index < this.currentPin.length) { + digit.textContent = '•'; + digit.classList.add('filled'); + } else { + digit.textContent = ''; + } + }); + } + + private showError(): void { + if (!this.pinDigits) return; + + this.pinDigits.forEach(digit => digit.classList.add('error')); + + // Shake animation + this.pinInput?.classList.add('shake'); + + setTimeout(() => { + this.currentPin = ''; + this.updateDisplay(); + this.pinInput?.classList.remove('shake'); + }, 500); + } + + private verify(): void { + if (this.currentPin === LockScreenController.CORRECT_PIN) { + this.hide(); + } else { + this.showError(); + } + } + + private addDigit(digit: string): void { + if (this.currentPin.length >= 4) return; + + this.currentPin += digit; + this.updateDisplay(); + + // Auto-verify when 4 digits entered + if (this.currentPin.length === 4) { + setTimeout(() => this.verify(), 200); + } + } + + private removeDigit(): void { + if (this.currentPin.length === 0) return; + this.currentPin = this.currentPin.slice(0, -1); + this.updateDisplay(); + } + + private clearPin(): void { + this.currentPin = ''; + this.updateDisplay(); + } + + private setupListeners(): void { + // Keypad click handler + this.pinKeypad?.addEventListener('click', (e) => this.handleKeypadClick(e)); + + // Keyboard input + document.addEventListener('keydown', (e) => this.handleKeyboard(e)); + + // Lock button in sidebar + document.querySelector('swp-side-menu-action.lock') + ?.addEventListener('click', () => this.show()); + } + + private handleKeypadClick(e: Event): void { + const target = e.target as HTMLElement; + const key = target.closest('swp-pin-key'); + + if (!key) return; + + const digit = key.dataset.digit; + const action = key.dataset.action; + + if (digit) { + this.addDigit(digit); + } else if (action === 'backspace') { + this.removeDigit(); + } else if (action === 'clear') { + this.clearPin(); + } + } + + private handleKeyboard(e: KeyboardEvent): void { + if (!this.isActive) return; + + // Prevent default to avoid other interactions + e.preventDefault(); + + if (e.key >= '0' && e.key <= '9') { + this.addDigit(e.key); + } else if (e.key === 'Backspace') { + this.removeDigit(); + } else if (e.key === 'Escape') { + this.clearPin(); + } + } +} diff --git a/PlanTempus.Application/wwwroot/ts/modules/search.ts b/PlanTempus.Application/wwwroot/ts/modules/search.ts new file mode 100644 index 0000000..d6eac21 --- /dev/null +++ b/PlanTempus.Application/wwwroot/ts/modules/search.ts @@ -0,0 +1,106 @@ +/** + * Search Controller + * + * Handles global search functionality and keyboard shortcuts + */ + +export class SearchController { + private input: HTMLInputElement | null = null; + private container: HTMLElement | null = null; + + constructor() { + this.input = document.getElementById('globalSearch') as HTMLInputElement | null; + this.container = document.querySelector('swp-topbar-search'); + + this.setupListeners(); + } + + /** + * Get current search value + */ + get value(): string { + return this.input?.value ?? ''; + } + + /** + * Set search value + */ + set value(val: string) { + if (this.input) { + this.input.value = val; + } + } + + /** + * Focus the search input + */ + focus(): void { + this.input?.focus(); + } + + /** + * Blur the search input + */ + blur(): void { + this.input?.blur(); + } + + /** + * Clear the search input + */ + clear(): void { + this.value = ''; + } + + private setupListeners(): void { + // Keyboard shortcuts + document.addEventListener('keydown', (e) => this.handleKeyboard(e)); + + // Input handlers + if (this.input) { + this.input.addEventListener('input', (e) => this.handleInput(e)); + + // Prevent form submission if wrapped in form + const form = this.input.closest('form'); + form?.addEventListener('submit', (e) => this.handleSubmit(e)); + } + } + + private handleKeyboard(e: KeyboardEvent): void { + // Cmd/Ctrl + K to focus search + if ((e.metaKey || e.ctrlKey) && e.key === 'k') { + e.preventDefault(); + this.focus(); + return; + } + + // Escape to blur search when focused + if (e.key === 'Escape' && document.activeElement === this.input) { + this.blur(); + } + } + + private handleInput(e: Event): void { + const target = e.target as HTMLInputElement; + const query = target.value.trim(); + + // Emit custom event for search + document.dispatchEvent(new CustomEvent('app:search', { + detail: { query }, + bubbles: true + })); + } + + private handleSubmit(e: Event): void { + e.preventDefault(); + + const query = this.value.trim(); + if (!query) return; + + // Emit custom event for search submit + document.dispatchEvent(new CustomEvent('app:search-submit', { + detail: { query }, + bubbles: true + })); + } +} diff --git a/PlanTempus.Application/wwwroot/ts/modules/sidebar.ts b/PlanTempus.Application/wwwroot/ts/modules/sidebar.ts new file mode 100644 index 0000000..ddb0a2b --- /dev/null +++ b/PlanTempus.Application/wwwroot/ts/modules/sidebar.ts @@ -0,0 +1,96 @@ +/** + * Sidebar Controller + * + * Handles sidebar collapse/expand and tooltip functionality + */ + +export class SidebarController { + private menuToggle: HTMLElement | null = null; + private appLayout: HTMLElement | null = null; + private menuTooltip: HTMLElement | null = null; + + constructor() { + this.menuToggle = document.getElementById('menuToggle'); + this.appLayout = document.querySelector('swp-app-layout'); + this.menuTooltip = document.getElementById('menuTooltip'); + + this.setupListeners(); + this.setupTooltips(); + this.restoreState(); + } + + /** + * Check if sidebar is collapsed + */ + get isCollapsed(): boolean { + return this.appLayout?.classList.contains('menu-collapsed') ?? false; + } + + /** + * Toggle sidebar collapsed state + */ + toggle(): void { + if (!this.appLayout) return; + + this.appLayout.classList.toggle('menu-collapsed'); + localStorage.setItem('sidebar-collapsed', String(this.isCollapsed)); + } + + /** + * Collapse the sidebar + */ + collapse(): void { + this.appLayout?.classList.add('menu-collapsed'); + localStorage.setItem('sidebar-collapsed', 'true'); + } + + /** + * Expand the sidebar + */ + expand(): void { + this.appLayout?.classList.remove('menu-collapsed'); + localStorage.setItem('sidebar-collapsed', 'false'); + } + + private setupListeners(): void { + this.menuToggle?.addEventListener('click', () => this.toggle()); + } + + private setupTooltips(): void { + if (!this.menuTooltip) return; + + const menuItems = document.querySelectorAll('swp-side-menu-item[data-tooltip]'); + + menuItems.forEach(item => { + item.addEventListener('mouseenter', () => this.showTooltip(item)); + item.addEventListener('mouseleave', () => this.hideTooltip()); + }); + } + + private showTooltip(item: HTMLElement): void { + if (!this.isCollapsed || !this.menuTooltip) return; + + const rect = item.getBoundingClientRect(); + const tooltipText = item.dataset.tooltip; + + if (!tooltipText) return; + + this.menuTooltip.textContent = tooltipText; + this.menuTooltip.style.left = `${rect.right + 8}px`; + this.menuTooltip.style.top = `${rect.top + rect.height / 2}px`; + this.menuTooltip.style.transform = 'translateY(-50%)'; + this.menuTooltip.showPopover(); + } + + private hideTooltip(): void { + this.menuTooltip?.hidePopover(); + } + + private restoreState(): void { + if (!this.appLayout) return; + + if (localStorage.getItem('sidebar-collapsed') === 'true') { + this.appLayout.classList.add('menu-collapsed'); + } + } +} diff --git a/PlanTempus.Application/wwwroot/ts/modules/theme.ts b/PlanTempus.Application/wwwroot/ts/modules/theme.ts new file mode 100644 index 0000000..b14e4b5 --- /dev/null +++ b/PlanTempus.Application/wwwroot/ts/modules/theme.ts @@ -0,0 +1,120 @@ +/** + * Theme Controller + * + * Handles dark/light mode switching and system preference detection + */ + +export type Theme = 'light' | 'dark' | 'system'; + +export class ThemeController { + private static readonly STORAGE_KEY = 'theme-preference'; + private static readonly DARK_CLASS = 'dark-mode'; + private static readonly LIGHT_CLASS = 'light-mode'; + + private root: HTMLElement; + private themeOptions: NodeListOf; + + constructor() { + this.root = document.documentElement; + this.themeOptions = document.querySelectorAll('swp-theme-option'); + + this.applyTheme(this.current); + this.updateUI(); + this.setupListeners(); + } + + /** + * Get the current theme setting + */ + get current(): Theme { + const stored = localStorage.getItem(ThemeController.STORAGE_KEY) as Theme | null; + if (stored === 'dark' || stored === 'light' || stored === 'system') { + return stored; + } + return 'system'; + } + + /** + * Check if dark mode is currently active + */ + get isDark(): boolean { + return this.root.classList.contains(ThemeController.DARK_CLASS) || + (this.systemPrefersDark && !this.root.classList.contains(ThemeController.LIGHT_CLASS)); + } + + /** + * Check if system prefers dark mode + */ + get systemPrefersDark(): boolean { + return window.matchMedia('(prefers-color-scheme: dark)').matches; + } + + /** + * Set theme and persist preference + */ + set(theme: Theme): void { + localStorage.setItem(ThemeController.STORAGE_KEY, theme); + this.applyTheme(theme); + this.updateUI(); + } + + /** + * Toggle between light and dark themes + */ + toggle(): void { + this.set(this.isDark ? 'light' : 'dark'); + } + + private applyTheme(theme: Theme): void { + this.root.classList.remove(ThemeController.DARK_CLASS, ThemeController.LIGHT_CLASS); + + if (theme === 'dark') { + this.root.classList.add(ThemeController.DARK_CLASS); + } else if (theme === 'light') { + this.root.classList.add(ThemeController.LIGHT_CLASS); + } + // 'system' leaves both classes off, letting CSS media query handle it + } + + private updateUI(): void { + if (!this.themeOptions) return; + + const darkActive = this.isDark; + + this.themeOptions.forEach(option => { + const theme = option.dataset.theme as Theme; + const isActive = (theme === 'dark' && darkActive) || (theme === 'light' && !darkActive); + option.classList.toggle('active', isActive); + }); + } + + private setupListeners(): void { + // Theme option clicks + this.themeOptions.forEach(option => { + option.addEventListener('click', (e) => this.handleOptionClick(e)); + }); + + // System theme changes + window.matchMedia('(prefers-color-scheme: dark)') + .addEventListener('change', () => this.handleSystemChange()); + } + + private handleOptionClick(e: Event): void { + const target = e.target as HTMLElement; + const option = target.closest('swp-theme-option'); + + if (option) { + const theme = option.dataset.theme as Theme; + if (theme) { + this.set(theme); + } + } + } + + private handleSystemChange(): void { + // Only react to system changes if we're using system preference + if (this.current === 'system') { + this.updateUI(); + } + } +} diff --git a/PlanTempus.Application/wwwroot/ts/tsconfig.json b/PlanTempus.Application/wwwroot/ts/tsconfig.json new file mode 100644 index 0000000..87d8c61 --- /dev/null +++ b/PlanTempus.Application/wwwroot/ts/tsconfig.json @@ -0,0 +1,22 @@ +{ + "compilerOptions": { + "target": "ES2020", + "module": "ESNext", + "moduleResolution": "bundler", + "strict": true, + "esModuleInterop": true, + "skipLibCheck": true, + "forceConsistentCasingInFileNames": true, + "declaration": false, + "outDir": "../js/app", + "rootDir": ".", + "sourceMap": true, + "lib": ["ES2020", "DOM", "DOM.Iterable"] + }, + "include": [ + "./**/*.ts" + ], + "exclude": [ + "node_modules" + ] +} diff --git a/PlanTempus.Components/Outbox/OutboxListener.cs b/PlanTempus.Components/Outbox/OutboxListener.cs index 80578dd..19d43cc 100644 --- a/PlanTempus.Components/Outbox/OutboxListener.cs +++ b/PlanTempus.Components/Outbox/OutboxListener.cs @@ -2,6 +2,7 @@ using Microsoft.ApplicationInsights; using Microsoft.Extensions.Hosting; using Npgsql; using PlanTempus.Core.Database.ConnectionFactory; +using PlanTempus.Core.Telemetry; namespace PlanTempus.Components.Outbox; @@ -10,19 +11,23 @@ public class OutboxListener : BackgroundService private readonly IDbConnectionFactory _connectionFactory; private readonly ICommandHandler _commandHandler; private readonly TelemetryClient _telemetryClient; + private readonly IMessageChannel _notificationChannel; public OutboxListener( IDbConnectionFactory connectionFactory, ICommandHandler commandHandler, - TelemetryClient telemetryClient) + TelemetryClient telemetryClient, + IMessageChannel notificationChannel) { _connectionFactory = connectionFactory; _commandHandler = commandHandler; _telemetryClient = telemetryClient; + _notificationChannel = notificationChannel; } protected override async Task ExecuteAsync(CancellationToken stoppingToken) { + Console.WriteLine("OutboxListener starting - listening for outbox_messages"); _telemetryClient.TrackTrace("OutboxListener starting - listening for outbox_messages"); while (!stoppingToken.IsCancellationRequested) @@ -33,6 +38,7 @@ public class OutboxListener : BackgroundService } catch (Exception ex) when (ex is not OperationCanceledException) { + Console.WriteLine($"OutboxListener error: {ex.Message}"); _telemetryClient.TrackException(ex); await Task.Delay(5000, stoppingToken); } @@ -44,18 +50,10 @@ public class OutboxListener : BackgroundService await using var conn = (NpgsqlConnection)_connectionFactory.Create(); await conn.OpenAsync(stoppingToken); - conn.Notification += async (_, e) => + conn.Notification += (_, e) => { - _telemetryClient.TrackTrace($"Outbox notification received: {e.Payload}"); - - try - { - await _commandHandler.Handle(new ProcessOutboxCommand()); - } - catch (Exception ex) - { - _telemetryClient.TrackException(ex); - } + Console.WriteLine($"Notification event received: {e.Payload}"); + _notificationChannel.Writer.TryWrite(e.Payload); }; await using (var cmd = new NpgsqlCommand("LISTEN outbox_messages;", conn)) @@ -63,14 +61,32 @@ public class OutboxListener : BackgroundService await cmd.ExecuteNonQueryAsync(stoppingToken); } + Console.WriteLine("OutboxListener now listening on outbox_messages channel"); _telemetryClient.TrackTrace("OutboxListener now listening on outbox_messages channel"); // Process any pending messages on startup + Console.WriteLine("Processing pending messages on startup..."); await _commandHandler.Handle(new ProcessOutboxCommand()); while (!stoppingToken.IsCancellationRequested) { await conn.WaitAsync(stoppingToken); + + while (_notificationChannel.Reader.TryRead(out var payload)) + { + Console.WriteLine($"Outbox notification received from channel: {payload}"); + _telemetryClient.TrackTrace($"Outbox notification received: {payload}"); + + try + { + await _commandHandler.Handle(new ProcessOutboxCommand()); + } + catch (Exception ex) + { + Console.WriteLine($"Error processing outbox: {ex.Message}"); + _telemetryClient.TrackException(ex); + } + } } } diff --git a/PlanTempus.Components/Outbox/ProcessOutboxHandler.cs b/PlanTempus.Components/Outbox/ProcessOutboxHandler.cs index cb62edd..6315869 100644 --- a/PlanTempus.Components/Outbox/ProcessOutboxHandler.cs +++ b/PlanTempus.Components/Outbox/ProcessOutboxHandler.cs @@ -13,23 +13,28 @@ public class ProcessOutboxHandler( { public async Task Handle(ProcessOutboxCommand command) { + Console.WriteLine($"ProcessOutboxHandler started"); telemetryClient.TrackTrace($"ProcessOutboxHandler started"); var messages = await outboxService.GetPendingAsync(command.BatchSize); + Console.WriteLine($"ProcessOutboxHandler found {messages.Count} pending messages"); telemetryClient.TrackTrace($"ProcessOutboxHandler found {messages.Count} pending messages"); foreach (var message in messages) { try { + Console.WriteLine($"Processing message {message.Id} of type {message.Type}"); telemetryClient.TrackTrace($"Processing message {message.Id} of type {message.Type}"); await ProcessMessageAsync(message); await outboxService.MarkAsSentAsync(message.Id); + Console.WriteLine($"Message {message.Id} marked as sent"); telemetryClient.TrackTrace($"Message {message.Id} marked as sent"); } catch (Exception ex) { + Console.WriteLine($"Message {message.Id} failed: {ex.Message}"); telemetryClient.TrackTrace($"Message {message.Id} failed: {ex.Message}"); await outboxService.MarkAsFailedAsync(message.Id, ex.Message); } diff --git a/PlanTempus.Components/PlanTempus.Components.csproj b/PlanTempus.Components/PlanTempus.Components.csproj index 2ea9caa..4dd8281 100644 --- a/PlanTempus.Components/PlanTempus.Components.csproj +++ b/PlanTempus.Components/PlanTempus.Components.csproj @@ -5,10 +5,6 @@ enable - - - - @@ -21,4 +17,8 @@ + + + + diff --git a/Core/CommandQueries/Command.cs b/PlanTempus.Core/CommandQueries/Command.cs similarity index 100% rename from Core/CommandQueries/Command.cs rename to PlanTempus.Core/CommandQueries/Command.cs diff --git a/Core/CommandQueries/CommandResponse.cs b/PlanTempus.Core/CommandQueries/CommandResponse.cs similarity index 100% rename from Core/CommandQueries/CommandResponse.cs rename to PlanTempus.Core/CommandQueries/CommandResponse.cs diff --git a/Core/CommandQueries/ICommand.cs b/PlanTempus.Core/CommandQueries/ICommand.cs similarity index 100% rename from Core/CommandQueries/ICommand.cs rename to PlanTempus.Core/CommandQueries/ICommand.cs diff --git a/Core/CommandQueries/ProblemDetails.cs b/PlanTempus.Core/CommandQueries/ProblemDetails.cs similarity index 100% rename from Core/CommandQueries/ProblemDetails.cs rename to PlanTempus.Core/CommandQueries/ProblemDetails.cs diff --git a/Core/Configurations/Common/KeyValueToJson.cs b/PlanTempus.Core/Configurations/Common/KeyValueToJson.cs similarity index 100% rename from Core/Configurations/Common/KeyValueToJson.cs rename to PlanTempus.Core/Configurations/Common/KeyValueToJson.cs diff --git a/Core/Configurations/ConfigurationBuilder.cs b/PlanTempus.Core/Configurations/ConfigurationBuilder.cs similarity index 100% rename from Core/Configurations/ConfigurationBuilder.cs rename to PlanTempus.Core/Configurations/ConfigurationBuilder.cs diff --git a/Core/Configurations/IAppConfiguration.cs b/PlanTempus.Core/Configurations/IAppConfiguration.cs similarity index 100% rename from Core/Configurations/IAppConfiguration.cs rename to PlanTempus.Core/Configurations/IAppConfiguration.cs diff --git a/Core/Configurations/IConfigurationRoot.cs b/PlanTempus.Core/Configurations/IConfigurationRoot.cs similarity index 100% rename from Core/Configurations/IConfigurationRoot.cs rename to PlanTempus.Core/Configurations/IConfigurationRoot.cs diff --git a/Core/Configurations/JsonConfigProvider/JsonConfigExtension.cs b/PlanTempus.Core/Configurations/JsonConfigProvider/JsonConfigExtension.cs similarity index 100% rename from Core/Configurations/JsonConfigProvider/JsonConfigExtension.cs rename to PlanTempus.Core/Configurations/JsonConfigProvider/JsonConfigExtension.cs diff --git a/Core/Configurations/SmartConfigProvider/AppConfiguration.cs b/PlanTempus.Core/Configurations/SmartConfigProvider/AppConfiguration.cs similarity index 100% rename from Core/Configurations/SmartConfigProvider/AppConfiguration.cs rename to PlanTempus.Core/Configurations/SmartConfigProvider/AppConfiguration.cs diff --git a/Core/Configurations/SmartConfigProvider/IConfigurationRepository.cs b/PlanTempus.Core/Configurations/SmartConfigProvider/IConfigurationRepository.cs similarity index 100% rename from Core/Configurations/SmartConfigProvider/IConfigurationRepository.cs rename to PlanTempus.Core/Configurations/SmartConfigProvider/IConfigurationRepository.cs diff --git a/Core/Configurations/SmartConfigProvider/Repositories/PostgresConfigurationRepository.cs b/PlanTempus.Core/Configurations/SmartConfigProvider/Repositories/PostgresConfigurationRepository.cs similarity index 100% rename from Core/Configurations/SmartConfigProvider/Repositories/PostgresConfigurationRepository.cs rename to PlanTempus.Core/Configurations/SmartConfigProvider/Repositories/PostgresConfigurationRepository.cs diff --git a/Core/Configurations/SmartConfigProvider/SmartConfigExtension.cs b/PlanTempus.Core/Configurations/SmartConfigProvider/SmartConfigExtension.cs similarity index 100% rename from Core/Configurations/SmartConfigProvider/SmartConfigExtension.cs rename to PlanTempus.Core/Configurations/SmartConfigProvider/SmartConfigExtension.cs diff --git a/Core/Configurations/SmartConfigProvider/SmartConfigOptions.cs b/PlanTempus.Core/Configurations/SmartConfigProvider/SmartConfigOptions.cs similarity index 100% rename from Core/Configurations/SmartConfigProvider/SmartConfigOptions.cs rename to PlanTempus.Core/Configurations/SmartConfigProvider/SmartConfigOptions.cs diff --git a/Core/Configurations/SmartConfigProvider/SmartConfigProvider.cs b/PlanTempus.Core/Configurations/SmartConfigProvider/SmartConfigProvider.cs similarity index 100% rename from Core/Configurations/SmartConfigProvider/SmartConfigProvider.cs rename to PlanTempus.Core/Configurations/SmartConfigProvider/SmartConfigProvider.cs diff --git a/Core/Database/ConnectionFactory/IDbConnectionFactory.cs b/PlanTempus.Core/Database/ConnectionFactory/IDbConnectionFactory.cs similarity index 100% rename from Core/Database/ConnectionFactory/IDbConnectionFactory.cs rename to PlanTempus.Core/Database/ConnectionFactory/IDbConnectionFactory.cs diff --git a/Core/Database/ConnectionFactory/PostgresConnectionFactory.cs b/PlanTempus.Core/Database/ConnectionFactory/PostgresConnectionFactory.cs similarity index 100% rename from Core/Database/ConnectionFactory/PostgresConnectionFactory.cs rename to PlanTempus.Core/Database/ConnectionFactory/PostgresConnectionFactory.cs diff --git a/Core/Database/DatabaseScope.cs b/PlanTempus.Core/Database/DatabaseScope.cs similarity index 100% rename from Core/Database/DatabaseScope.cs rename to PlanTempus.Core/Database/DatabaseScope.cs diff --git a/Core/Database/IDatabaseOperations.cs b/PlanTempus.Core/Database/IDatabaseOperations.cs similarity index 100% rename from Core/Database/IDatabaseOperations.cs rename to PlanTempus.Core/Database/IDatabaseOperations.cs diff --git a/Core/Database/SqlOperations.cs b/PlanTempus.Core/Database/SqlOperations.cs similarity index 100% rename from Core/Database/SqlOperations.cs rename to PlanTempus.Core/Database/SqlOperations.cs diff --git a/Core/Email/EmailModule.cs b/PlanTempus.Core/Email/EmailModule.cs similarity index 100% rename from Core/Email/EmailModule.cs rename to PlanTempus.Core/Email/EmailModule.cs diff --git a/Core/Email/IEmailService.cs b/PlanTempus.Core/Email/IEmailService.cs similarity index 100% rename from Core/Email/IEmailService.cs rename to PlanTempus.Core/Email/IEmailService.cs diff --git a/Core/Email/PostmarkConfiguration.cs b/PlanTempus.Core/Email/PostmarkConfiguration.cs similarity index 100% rename from Core/Email/PostmarkConfiguration.cs rename to PlanTempus.Core/Email/PostmarkConfiguration.cs diff --git a/Core/Email/PostmarkEmailService.cs b/PlanTempus.Core/Email/PostmarkEmailService.cs similarity index 100% rename from Core/Email/PostmarkEmailService.cs rename to PlanTempus.Core/Email/PostmarkEmailService.cs diff --git a/Core/Entities/Accounts/Account.cs b/PlanTempus.Core/Entities/Accounts/Account.cs similarity index 100% rename from Core/Entities/Accounts/Account.cs rename to PlanTempus.Core/Entities/Accounts/Account.cs diff --git a/Core/Exceptions/ConfigurationException.cs b/PlanTempus.Core/Exceptions/ConfigurationException.cs similarity index 100% rename from Core/Exceptions/ConfigurationException.cs rename to PlanTempus.Core/Exceptions/ConfigurationException.cs diff --git a/Core/ISecureTokenizer.cs b/PlanTempus.Core/ISecureTokenizer.cs similarity index 100% rename from Core/ISecureTokenizer.cs rename to PlanTempus.Core/ISecureTokenizer.cs diff --git a/Core/ModuleRegistry/SecurityModule.cs b/PlanTempus.Core/ModuleRegistry/SecurityModule.cs similarity index 100% rename from Core/ModuleRegistry/SecurityModule.cs rename to PlanTempus.Core/ModuleRegistry/SecurityModule.cs diff --git a/Core/ModuleRegistry/SeqLoggingModule.cs b/PlanTempus.Core/ModuleRegistry/SeqLoggingModule.cs similarity index 100% rename from Core/ModuleRegistry/SeqLoggingModule.cs rename to PlanTempus.Core/ModuleRegistry/SeqLoggingModule.cs diff --git a/Core/ModuleRegistry/TelemetryModule.cs b/PlanTempus.Core/ModuleRegistry/TelemetryModule.cs similarity index 100% rename from Core/ModuleRegistry/TelemetryModule.cs rename to PlanTempus.Core/ModuleRegistry/TelemetryModule.cs diff --git a/Core/MultiKeyEncryption/MasterKey.cs b/PlanTempus.Core/MultiKeyEncryption/MasterKey.cs similarity index 100% rename from Core/MultiKeyEncryption/MasterKey.cs rename to PlanTempus.Core/MultiKeyEncryption/MasterKey.cs diff --git a/Core/MultiKeyEncryption/SecureConnectionString.cs b/PlanTempus.Core/MultiKeyEncryption/SecureConnectionString.cs similarity index 100% rename from Core/MultiKeyEncryption/SecureConnectionString.cs rename to PlanTempus.Core/MultiKeyEncryption/SecureConnectionString.cs diff --git a/Core/Outbox/IOutboxService.cs b/PlanTempus.Core/Outbox/IOutboxService.cs similarity index 100% rename from Core/Outbox/IOutboxService.cs rename to PlanTempus.Core/Outbox/IOutboxService.cs diff --git a/Core/Outbox/OutboxMessage.cs b/PlanTempus.Core/Outbox/OutboxMessage.cs similarity index 100% rename from Core/Outbox/OutboxMessage.cs rename to PlanTempus.Core/Outbox/OutboxMessage.cs diff --git a/Core/Outbox/OutboxModule.cs b/PlanTempus.Core/Outbox/OutboxModule.cs similarity index 100% rename from Core/Outbox/OutboxModule.cs rename to PlanTempus.Core/Outbox/OutboxModule.cs diff --git a/Core/Outbox/OutboxService.cs b/PlanTempus.Core/Outbox/OutboxService.cs similarity index 100% rename from Core/Outbox/OutboxService.cs rename to PlanTempus.Core/Outbox/OutboxService.cs diff --git a/Core/PlanTempus.Core.csproj b/PlanTempus.Core/PlanTempus.Core.csproj similarity index 100% rename from Core/PlanTempus.Core.csproj rename to PlanTempus.Core/PlanTempus.Core.csproj diff --git a/Core/SecureTokenizer.cs b/PlanTempus.Core/SecureTokenizer.cs similarity index 100% rename from Core/SecureTokenizer.cs rename to PlanTempus.Core/SecureTokenizer.cs diff --git a/PlanTempus.Core/SeqLogging/SeqBackgroundService.cs b/PlanTempus.Core/SeqLogging/SeqBackgroundService.cs new file mode 100644 index 0000000..ab0e9a9 --- /dev/null +++ b/PlanTempus.Core/SeqLogging/SeqBackgroundService.cs @@ -0,0 +1,87 @@ +using Microsoft.ApplicationInsights; +using Microsoft.ApplicationInsights.Channel; +using Microsoft.ApplicationInsights.DataContracts; +using Microsoft.Extensions.Hosting; +using PlanTempus.Core.Telemetry; + +namespace PlanTempus.Core.SeqLogging +{ + public class SeqBackgroundService : BackgroundService + { + private readonly IMessageChannel _messageChannel; + private readonly TelemetryClient _telemetryClient; + private readonly SeqLogger _seqLogger; + private readonly TaskCompletionSource _shutdownComplete = new(); + + public SeqBackgroundService(TelemetryClient telemetryClient, + IMessageChannel messageChannel, + SeqLogger seqlogger) + { + _telemetryClient = telemetryClient; + _messageChannel = messageChannel; + _seqLogger = seqlogger; + + _telemetryClient.TrackTrace("SeqBackgroundService started"); + + } + + protected override async Task ExecuteAsync(CancellationToken stoppingToken) + { + await foreach (var telemetry in _messageChannel.Reader.ReadAllAsync()) + { + try + { + switch (telemetry) + { + case StopTelemetry: + StopGracefully(); + return; + + case ExceptionTelemetry et: + await _seqLogger.LogAsync(et); + break; + + case TraceTelemetry et: + await _seqLogger.LogAsync(et); + break; + + case DependencyTelemetry et: + await _seqLogger.LogAsync(et); + break; + + case RequestTelemetry et: + await _seqLogger.LogAsync(et); + break; + + case EventTelemetry et: + await _seqLogger.LogAsync(et); + break; + + default: + throw new NotSupportedException(telemetry.GetType().Name); + } + } + catch (Exception) + { + // Ignore errors processing telemetry + } + } + } + + public override async Task StopAsync(CancellationToken cancellationToken) + { + _messageChannel.Writer.TryWrite(new StopTelemetry()); + + // Vent max 10 sekunder på graceful shutdown + await Task.WhenAny(_shutdownComplete.Task, Task.Delay(10000)); + + await base.StopAsync(cancellationToken); + } + + private void StopGracefully() + { + _messageChannel.Dispose(); + _shutdownComplete.SetResult(); + } + } +} diff --git a/Core/SeqLogging/SeqConfiguration.cs b/PlanTempus.Core/SeqLogging/SeqConfiguration.cs similarity index 100% rename from Core/SeqLogging/SeqConfiguration.cs rename to PlanTempus.Core/SeqLogging/SeqConfiguration.cs diff --git a/Core/SeqLogging/SeqHttpClient.cs b/PlanTempus.Core/SeqLogging/SeqHttpClient.cs similarity index 100% rename from Core/SeqLogging/SeqHttpClient.cs rename to PlanTempus.Core/SeqLogging/SeqHttpClient.cs diff --git a/Core/SeqLogging/SeqLogger.cs b/PlanTempus.Core/SeqLogging/SeqLogger.cs similarity index 100% rename from Core/SeqLogging/SeqLogger.cs rename to PlanTempus.Core/SeqLogging/SeqLogger.cs diff --git a/Core/Telemetry/Enrichers/EnrichWithMetaTelemetry.cs b/PlanTempus.Core/Telemetry/Enrichers/EnrichWithMetaTelemetry.cs similarity index 100% rename from Core/Telemetry/Enrichers/EnrichWithMetaTelemetry.cs rename to PlanTempus.Core/Telemetry/Enrichers/EnrichWithMetaTelemetry.cs diff --git a/Core/Telemetry/IMessageChannel.cs b/PlanTempus.Core/Telemetry/IMessageChannel.cs similarity index 100% rename from Core/Telemetry/IMessageChannel.cs rename to PlanTempus.Core/Telemetry/IMessageChannel.cs diff --git a/Core/Telemetry/MessageChannel.cs b/PlanTempus.Core/Telemetry/MessageChannel.cs similarity index 100% rename from Core/Telemetry/MessageChannel.cs rename to PlanTempus.Core/Telemetry/MessageChannel.cs diff --git a/PlanTempus.Core/Telemetry/NotificationChannel.cs b/PlanTempus.Core/Telemetry/NotificationChannel.cs new file mode 100644 index 0000000..077e804 --- /dev/null +++ b/PlanTempus.Core/Telemetry/NotificationChannel.cs @@ -0,0 +1,13 @@ +using System.Threading.Channels; + +namespace PlanTempus.Core.Telemetry; + +public class NotificationChannel : IMessageChannel +{ + private readonly Channel _channel = Channel.CreateUnbounded(); + + public ChannelWriter Writer => _channel.Writer; + public ChannelReader Reader => _channel.Reader; + + public void Dispose() => _channel.Writer.Complete(); +} diff --git a/Core/Telemetry/SeqTelemetryChannel.cs b/PlanTempus.Core/Telemetry/SeqTelemetryChannel.cs similarity index 100% rename from Core/Telemetry/SeqTelemetryChannel.cs rename to PlanTempus.Core/Telemetry/SeqTelemetryChannel.cs diff --git a/PlanTempus.Core/Telemetry/StopTelemetry.cs b/PlanTempus.Core/Telemetry/StopTelemetry.cs new file mode 100644 index 0000000..966ef28 --- /dev/null +++ b/PlanTempus.Core/Telemetry/StopTelemetry.cs @@ -0,0 +1,25 @@ +using Microsoft.ApplicationInsights.Channel; +using Microsoft.ApplicationInsights.DataContracts; +using Microsoft.ApplicationInsights.Extensibility; + +namespace PlanTempus.Core.Telemetry; + +/// +/// Signal telemetry der bruges til at stoppe SeqBackgroundService gracefully. +/// Når denne læses fra channel, stopper servicen efter at have processeret alle tidligere beskeder. +/// +public class StopTelemetry : ITelemetry +{ + public DateTimeOffset Timestamp { get; set; } + public string Sequence { get; set; } + public TelemetryContext Context { get; } = new TelemetryContext(); + public IExtension Extension { get; set; } + + public ITelemetry DeepClone() => new StopTelemetry(); + + public void Sanitize() { } + + public void SerializeData(ISerializationWriter serializationWriter) { } + + +} diff --git a/Core/Telemetry/TelemetryExtensions.cs b/PlanTempus.Core/Telemetry/TelemetryExtensions.cs similarity index 100% rename from Core/Telemetry/TelemetryExtensions.cs rename to PlanTempus.Core/Telemetry/TelemetryExtensions.cs diff --git a/Database/Common/Validations.cs b/PlanTempus.Database/Common/Validations.cs similarity index 100% rename from Database/Common/Validations.cs rename to PlanTempus.Database/Common/Validations.cs diff --git a/Database/ConfigurationManagementSystem/SetupConfiguration.cs b/PlanTempus.Database/ConfigurationManagementSystem/SetupConfiguration.cs similarity index 100% rename from Database/ConfigurationManagementSystem/SetupConfiguration.cs rename to PlanTempus.Database/ConfigurationManagementSystem/SetupConfiguration.cs diff --git a/Database/Core/AccountService.cs b/PlanTempus.Database/Core/AccountService.cs similarity index 100% rename from Database/Core/AccountService.cs rename to PlanTempus.Database/Core/AccountService.cs diff --git a/Database/Core/DCL/SetupApplicationUser.cs b/PlanTempus.Database/Core/DCL/SetupApplicationUser.cs similarity index 100% rename from Database/Core/DCL/SetupApplicationUser.cs rename to PlanTempus.Database/Core/DCL/SetupApplicationUser.cs diff --git a/Database/Core/DCL/SetupDbAdmin.cs b/PlanTempus.Database/Core/DCL/SetupDbAdmin.cs similarity index 100% rename from Database/Core/DCL/SetupDbAdmin.cs rename to PlanTempus.Database/Core/DCL/SetupDbAdmin.cs diff --git a/Database/Core/DCL/SetupOrganizationUser.cs b/PlanTempus.Database/Core/DCL/SetupOrganizationUser.cs similarity index 100% rename from Database/Core/DCL/SetupOrganizationUser.cs rename to PlanTempus.Database/Core/DCL/SetupOrganizationUser.cs diff --git a/Database/Core/DDL/SetupIdentitySystem.cs b/PlanTempus.Database/Core/DDL/SetupIdentitySystem.cs similarity index 100% rename from Database/Core/DDL/SetupIdentitySystem.cs rename to PlanTempus.Database/Core/DDL/SetupIdentitySystem.cs diff --git a/Database/Core/DDL/SetupOutbox.cs b/PlanTempus.Database/Core/DDL/SetupOutbox.cs similarity index 100% rename from Database/Core/DDL/SetupOutbox.cs rename to PlanTempus.Database/Core/DDL/SetupOutbox.cs diff --git a/Database/Core/IDbConfigure.cs b/PlanTempus.Database/Core/IDbConfigure.cs similarity index 100% rename from Database/Core/IDbConfigure.cs rename to PlanTempus.Database/Core/IDbConfigure.cs diff --git a/Database/ModuleRegistry/DbPostgreSqlModule.cs b/PlanTempus.Database/ModuleRegistry/DbPostgreSqlModule.cs similarity index 100% rename from Database/ModuleRegistry/DbPostgreSqlModule.cs rename to PlanTempus.Database/ModuleRegistry/DbPostgreSqlModule.cs diff --git a/Database/NavigationSystem/Setup.cs b/PlanTempus.Database/NavigationSystem/Setup.cs similarity index 100% rename from Database/NavigationSystem/Setup.cs rename to PlanTempus.Database/NavigationSystem/Setup.cs diff --git a/Database/PlanTempus.Database.csproj b/PlanTempus.Database/PlanTempus.Database.csproj similarity index 77% rename from Database/PlanTempus.Database.csproj rename to PlanTempus.Database/PlanTempus.Database.csproj index 76de5e5..b76d642 100644 --- a/Database/PlanTempus.Database.csproj +++ b/PlanTempus.Database/PlanTempus.Database.csproj @@ -5,12 +5,12 @@ enable - - - - + + + + diff --git a/Database/RolesPermissionSystem/Setup.cs b/PlanTempus.Database/RolesPermissionSystem/Setup.cs similarity index 100% rename from Database/RolesPermissionSystem/Setup.cs rename to PlanTempus.Database/RolesPermissionSystem/Setup.cs diff --git a/Database/Tenants/InitializeTenantData.cs b/PlanTempus.Database/Tenants/InitializeTenantData.cs similarity index 100% rename from Database/Tenants/InitializeTenantData.cs rename to PlanTempus.Database/Tenants/InitializeTenantData.cs diff --git a/Application/Common/ComponentsViewLocationExpander.cs b/PlanTempus.DeprecatedApplication/Common/ComponentsViewLocationExpander.cs similarity index 100% rename from Application/Common/ComponentsViewLocationExpander.cs rename to PlanTempus.DeprecatedApplication/Common/ComponentsViewLocationExpander.cs diff --git a/Application/Components/ApiViewComponentBase.cs b/PlanTempus.DeprecatedApplication/Components/ApiViewComponentBase.cs similarity index 100% rename from Application/Components/ApiViewComponentBase.cs rename to PlanTempus.DeprecatedApplication/Components/ApiViewComponentBase.cs diff --git a/Application/Components/Navigation/Default.cshtml b/PlanTempus.DeprecatedApplication/Components/Navigation/Default.cshtml similarity index 100% rename from Application/Components/Navigation/Default.cshtml rename to PlanTempus.DeprecatedApplication/Components/Navigation/Default.cshtml diff --git a/Application/Components/Navigation/NavigationViewComponent.cs b/PlanTempus.DeprecatedApplication/Components/Navigation/NavigationViewComponent.cs similarity index 100% rename from Application/Components/Navigation/NavigationViewComponent.cs rename to PlanTempus.DeprecatedApplication/Components/Navigation/NavigationViewComponent.cs diff --git a/Application/Components/OrganizationViewComponent.cs b/PlanTempus.DeprecatedApplication/Components/OrganizationViewComponent.cs similarity index 100% rename from Application/Components/OrganizationViewComponent.cs rename to PlanTempus.DeprecatedApplication/Components/OrganizationViewComponent.cs diff --git a/Application/Pages/Index.cshtml b/PlanTempus.DeprecatedApplication/Pages/Index.cshtml similarity index 100% rename from Application/Pages/Index.cshtml rename to PlanTempus.DeprecatedApplication/Pages/Index.cshtml diff --git a/Application/Pages/Index.cshtml.cs b/PlanTempus.DeprecatedApplication/Pages/Index.cshtml.cs similarity index 100% rename from Application/Pages/Index.cshtml.cs rename to PlanTempus.DeprecatedApplication/Pages/Index.cshtml.cs diff --git a/Application/Pages/Shared/_Layout.cshtml b/PlanTempus.DeprecatedApplication/Pages/Shared/_Layout.cshtml similarity index 100% rename from Application/Pages/Shared/_Layout.cshtml rename to PlanTempus.DeprecatedApplication/Pages/Shared/_Layout.cshtml diff --git a/Application/Pages/Shared/_Layout.cshtml.css b/PlanTempus.DeprecatedApplication/Pages/Shared/_Layout.cshtml.css similarity index 100% rename from Application/Pages/Shared/_Layout.cshtml.css rename to PlanTempus.DeprecatedApplication/Pages/Shared/_Layout.cshtml.css diff --git a/Application/Pages/_ViewImports.cshtml b/PlanTempus.DeprecatedApplication/Pages/_ViewImports.cshtml similarity index 100% rename from Application/Pages/_ViewImports.cshtml rename to PlanTempus.DeprecatedApplication/Pages/_ViewImports.cshtml diff --git a/Application/Pages/_ViewStart.cshtml b/PlanTempus.DeprecatedApplication/Pages/_ViewStart.cshtml similarity index 100% rename from Application/Pages/_ViewStart.cshtml rename to PlanTempus.DeprecatedApplication/Pages/_ViewStart.cshtml diff --git a/Application/PlanTempus.Application.csproj b/PlanTempus.DeprecatedApplication/PlanTempus.DeprecatedApplication.csproj similarity index 100% rename from Application/PlanTempus.Application.csproj rename to PlanTempus.DeprecatedApplication/PlanTempus.DeprecatedApplication.csproj diff --git a/Application/Program.cs b/PlanTempus.DeprecatedApplication/Program.cs similarity index 100% rename from Application/Program.cs rename to PlanTempus.DeprecatedApplication/Program.cs diff --git a/Application/Properties/launchSettings.json b/PlanTempus.DeprecatedApplication/Properties/launchSettings.json similarity index 100% rename from Application/Properties/launchSettings.json rename to PlanTempus.DeprecatedApplication/Properties/launchSettings.json diff --git a/Application/Startup.cs b/PlanTempus.DeprecatedApplication/Startup.cs similarity index 100% rename from Application/Startup.cs rename to PlanTempus.DeprecatedApplication/Startup.cs diff --git a/Application/TypeScript/App.ts b/PlanTempus.DeprecatedApplication/TypeScript/App.ts similarity index 100% rename from Application/TypeScript/App.ts rename to PlanTempus.DeprecatedApplication/TypeScript/App.ts diff --git a/Application/TypeScript/Modules/esbuild_ReadMe.txt b/PlanTempus.DeprecatedApplication/TypeScript/Modules/esbuild_ReadMe.txt similarity index 100% rename from Application/TypeScript/Modules/esbuild_ReadMe.txt rename to PlanTempus.DeprecatedApplication/TypeScript/Modules/esbuild_ReadMe.txt diff --git a/Application/appconfiguration.Development.json b/PlanTempus.DeprecatedApplication/appconfiguration.Development.json similarity index 100% rename from Application/appconfiguration.Development.json rename to PlanTempus.DeprecatedApplication/appconfiguration.Development.json diff --git a/Application/appconfiguration.json b/PlanTempus.DeprecatedApplication/appconfiguration.json similarity index 100% rename from Application/appconfiguration.json rename to PlanTempus.DeprecatedApplication/appconfiguration.json diff --git a/Application/package-lock.json b/PlanTempus.DeprecatedApplication/package-lock.json similarity index 100% rename from Application/package-lock.json rename to PlanTempus.DeprecatedApplication/package-lock.json diff --git a/Application/package.json b/PlanTempus.DeprecatedApplication/package.json similarity index 100% rename from Application/package.json rename to PlanTempus.DeprecatedApplication/package.json diff --git a/Application/wwwroot/css/site.css b/PlanTempus.DeprecatedApplication/wwwroot/css/site.css similarity index 100% rename from Application/wwwroot/css/site.css rename to PlanTempus.DeprecatedApplication/wwwroot/css/site.css diff --git a/Application/wwwroot/favicon.ico b/PlanTempus.DeprecatedApplication/wwwroot/favicon.ico similarity index 100% rename from Application/wwwroot/favicon.ico rename to PlanTempus.DeprecatedApplication/wwwroot/favicon.ico diff --git a/Application/wwwroot/fonts/Rubik-Regular.woff2 b/PlanTempus.DeprecatedApplication/wwwroot/fonts/Rubik-Regular.woff2 similarity index 100% rename from Application/wwwroot/fonts/Rubik-Regular.woff2 rename to PlanTempus.DeprecatedApplication/wwwroot/fonts/Rubik-Regular.woff2 diff --git a/Application/wwwroot/fonts/Rubik-VariableFont_wght.woff2 b/PlanTempus.DeprecatedApplication/wwwroot/fonts/Rubik-VariableFont_wght.woff2 similarity index 100% rename from Application/wwwroot/fonts/Rubik-VariableFont_wght.woff2 rename to PlanTempus.DeprecatedApplication/wwwroot/fonts/Rubik-VariableFont_wght.woff2 diff --git a/Application/wwwroot/images/logo-1.png b/PlanTempus.DeprecatedApplication/wwwroot/images/logo-1.png similarity index 100% rename from Application/wwwroot/images/logo-1.png rename to PlanTempus.DeprecatedApplication/wwwroot/images/logo-1.png diff --git a/Application/wwwroot/images/pt-logo1.png b/PlanTempus.DeprecatedApplication/wwwroot/images/pt-logo1.png similarity index 100% rename from Application/wwwroot/images/pt-logo1.png rename to PlanTempus.DeprecatedApplication/wwwroot/images/pt-logo1.png diff --git a/Application/wwwroot/images/pt-logo2.png b/PlanTempus.DeprecatedApplication/wwwroot/images/pt-logo2.png similarity index 100% rename from Application/wwwroot/images/pt-logo2.png rename to PlanTempus.DeprecatedApplication/wwwroot/images/pt-logo2.png diff --git a/Application/wwwroot/js/app.js b/PlanTempus.DeprecatedApplication/wwwroot/js/app.js similarity index 100% rename from Application/wwwroot/js/app.js rename to PlanTempus.DeprecatedApplication/wwwroot/js/app.js diff --git a/Application/wwwroot/js/app.js.map b/PlanTempus.DeprecatedApplication/wwwroot/js/app.js.map similarity index 100% rename from Application/wwwroot/js/app.js.map rename to PlanTempus.DeprecatedApplication/wwwroot/js/app.js.map diff --git a/SetupInfrastructure/CreateRole.txt b/PlanTempus.SetupInfrastructure/CreateRole.txt similarity index 100% rename from SetupInfrastructure/CreateRole.txt rename to PlanTempus.SetupInfrastructure/CreateRole.txt diff --git a/SetupInfrastructure/PlanTempus.SetupInfrastructure.csproj b/PlanTempus.SetupInfrastructure/PlanTempus.SetupInfrastructure.csproj similarity index 84% rename from SetupInfrastructure/PlanTempus.SetupInfrastructure.csproj rename to PlanTempus.SetupInfrastructure/PlanTempus.SetupInfrastructure.csproj index 5c9236b..d1f423e 100644 --- a/SetupInfrastructure/PlanTempus.SetupInfrastructure.csproj +++ b/PlanTempus.SetupInfrastructure/PlanTempus.SetupInfrastructure.csproj @@ -10,8 +10,8 @@ - - + + diff --git a/SetupInfrastructure/Program.cs b/PlanTempus.SetupInfrastructure/Program.cs similarity index 100% rename from SetupInfrastructure/Program.cs rename to PlanTempus.SetupInfrastructure/Program.cs diff --git a/SetupInfrastructure/Startup.cs b/PlanTempus.SetupInfrastructure/Startup.cs similarity index 100% rename from SetupInfrastructure/Startup.cs rename to PlanTempus.SetupInfrastructure/Startup.cs diff --git a/SetupInfrastructure/appconfiguration.json b/PlanTempus.SetupInfrastructure/appconfiguration.json similarity index 100% rename from SetupInfrastructure/appconfiguration.json rename to PlanTempus.SetupInfrastructure/appconfiguration.json diff --git a/SqlManagement/.dbeaver/.credentials-config.json.bak b/PlanTempus.SqlManagement/.dbeaver/.credentials-config.json.bak similarity index 100% rename from SqlManagement/.dbeaver/.credentials-config.json.bak rename to PlanTempus.SqlManagement/.dbeaver/.credentials-config.json.bak diff --git a/SqlManagement/.dbeaver/.data-sources.json.bak b/PlanTempus.SqlManagement/.dbeaver/.data-sources.json.bak similarity index 100% rename from SqlManagement/.dbeaver/.data-sources.json.bak rename to PlanTempus.SqlManagement/.dbeaver/.data-sources.json.bak diff --git a/SqlManagement/.dbeaver/.project-metadata.json.bak b/PlanTempus.SqlManagement/.dbeaver/.project-metadata.json.bak similarity index 100% rename from SqlManagement/.dbeaver/.project-metadata.json.bak rename to PlanTempus.SqlManagement/.dbeaver/.project-metadata.json.bak diff --git a/SqlManagement/.dbeaver/credentials-config.json b/PlanTempus.SqlManagement/.dbeaver/credentials-config.json similarity index 100% rename from SqlManagement/.dbeaver/credentials-config.json rename to PlanTempus.SqlManagement/.dbeaver/credentials-config.json diff --git a/SqlManagement/.dbeaver/data-sources.json b/PlanTempus.SqlManagement/.dbeaver/data-sources.json similarity index 100% rename from SqlManagement/.dbeaver/data-sources.json rename to PlanTempus.SqlManagement/.dbeaver/data-sources.json diff --git a/SqlManagement/.dbeaver/project-metadata.json b/PlanTempus.SqlManagement/.dbeaver/project-metadata.json similarity index 100% rename from SqlManagement/.dbeaver/project-metadata.json rename to PlanTempus.SqlManagement/.dbeaver/project-metadata.json diff --git a/SqlManagement/.dbeaver/project-settings.json b/PlanTempus.SqlManagement/.dbeaver/project-settings.json similarity index 100% rename from SqlManagement/.dbeaver/project-settings.json rename to PlanTempus.SqlManagement/.dbeaver/project-settings.json diff --git a/SqlManagement/.project b/PlanTempus.SqlManagement/.project similarity index 100% rename from SqlManagement/.project rename to PlanTempus.SqlManagement/.project diff --git a/SqlManagement/Dashboards/Test.dashboard b/PlanTempus.SqlManagement/Dashboards/Test.dashboard similarity index 100% rename from SqlManagement/Dashboards/Test.dashboard rename to PlanTempus.SqlManagement/Dashboards/Test.dashboard diff --git a/SqlManagement/Diagrams/global.erd b/PlanTempus.SqlManagement/Diagrams/global.erd similarity index 100% rename from SqlManagement/Diagrams/global.erd rename to PlanTempus.SqlManagement/Diagrams/global.erd diff --git a/SqlManagement/Scripts/Script.sql b/PlanTempus.SqlManagement/Scripts/Script.sql similarity index 100% rename from SqlManagement/Scripts/Script.sql rename to PlanTempus.SqlManagement/Scripts/Script.sql diff --git a/SqlManagement/Scripts/SmartConfigSystem.sql b/PlanTempus.SqlManagement/Scripts/SmartConfigSystem.sql similarity index 100% rename from SqlManagement/Scripts/SmartConfigSystem.sql rename to PlanTempus.SqlManagement/Scripts/SmartConfigSystem.sql diff --git a/SqlManagement/Scripts/grant-privileges.sql b/PlanTempus.SqlManagement/Scripts/grant-privileges.sql similarity index 100% rename from SqlManagement/Scripts/grant-privileges.sql rename to PlanTempus.SqlManagement/Scripts/grant-privileges.sql diff --git a/Tests/CodeSnippets/TestPostgresLISTENNOTIFY.cs b/PlanTempus.Tests/CodeSnippets/TestPostgresLISTENNOTIFY.cs similarity index 100% rename from Tests/CodeSnippets/TestPostgresLISTENNOTIFY.cs rename to PlanTempus.Tests/CodeSnippets/TestPostgresLISTENNOTIFY.cs diff --git a/Tests/CodeSnippets/sandbox.sql b/PlanTempus.Tests/CodeSnippets/sandbox.sql similarity index 100% rename from Tests/CodeSnippets/sandbox.sql rename to PlanTempus.Tests/CodeSnippets/sandbox.sql diff --git a/Tests/CommandQueryHandlerTests/HandlerTest.cs b/PlanTempus.Tests/CommandQueryHandlerTests/HandlerTest.cs similarity index 100% rename from Tests/CommandQueryHandlerTests/HandlerTest.cs rename to PlanTempus.Tests/CommandQueryHandlerTests/HandlerTest.cs diff --git a/Tests/CommandQueryHandlerTests/ResponseTests.cs b/PlanTempus.Tests/CommandQueryHandlerTests/ResponseTests.cs similarity index 100% rename from Tests/CommandQueryHandlerTests/ResponseTests.cs rename to PlanTempus.Tests/CommandQueryHandlerTests/ResponseTests.cs diff --git a/Tests/ConfigurationSystem/SetupConfigurationTests.cs b/PlanTempus.Tests/ConfigurationSystem/SetupConfigurationTests.cs similarity index 100% rename from Tests/ConfigurationSystem/SetupConfigurationTests.cs rename to PlanTempus.Tests/ConfigurationSystem/SetupConfigurationTests.cs diff --git a/Tests/ConfigurationTests/JsonConfigurationProviderTests.cs b/PlanTempus.Tests/ConfigurationTests/JsonConfigurationProviderTests.cs similarity index 100% rename from Tests/ConfigurationTests/JsonConfigurationProviderTests.cs rename to PlanTempus.Tests/ConfigurationTests/JsonConfigurationProviderTests.cs diff --git a/Tests/ConfigurationTests/KeyValueJsonHandlingTests.cs b/PlanTempus.Tests/ConfigurationTests/KeyValueJsonHandlingTests.cs similarity index 100% rename from Tests/ConfigurationTests/KeyValueJsonHandlingTests.cs rename to PlanTempus.Tests/ConfigurationTests/KeyValueJsonHandlingTests.cs diff --git a/Tests/ConfigurationTests/SmartConfigProviderTests.cs b/PlanTempus.Tests/ConfigurationTests/SmartConfigProviderTests.cs similarity index 100% rename from Tests/ConfigurationTests/SmartConfigProviderTests.cs rename to PlanTempus.Tests/ConfigurationTests/SmartConfigProviderTests.cs diff --git a/Tests/ConfigurationTests/appconfiguration.dev.json b/PlanTempus.Tests/ConfigurationTests/appconfiguration.dev.json similarity index 100% rename from Tests/ConfigurationTests/appconfiguration.dev.json rename to PlanTempus.Tests/ConfigurationTests/appconfiguration.dev.json diff --git a/Tests/Logging/SeqBackgroundServiceTest.cs b/PlanTempus.Tests/Logging/SeqBackgroundServiceTest.cs similarity index 100% rename from Tests/Logging/SeqBackgroundServiceTest.cs rename to PlanTempus.Tests/Logging/SeqBackgroundServiceTest.cs diff --git a/Tests/Logging/SeqLoggerTests.cs b/PlanTempus.Tests/Logging/SeqLoggerTests.cs similarity index 100% rename from Tests/Logging/SeqLoggerTests.cs rename to PlanTempus.Tests/Logging/SeqLoggerTests.cs diff --git a/Tests/Logging/SeqTelemetryChannelTest.cs b/PlanTempus.Tests/Logging/SeqTelemetryChannelTest.cs similarity index 100% rename from Tests/Logging/SeqTelemetryChannelTest.cs rename to PlanTempus.Tests/Logging/SeqTelemetryChannelTest.cs diff --git a/Tests/PasswordHasherTest.cs b/PlanTempus.Tests/PasswordHasherTest.cs similarity index 100% rename from Tests/PasswordHasherTest.cs rename to PlanTempus.Tests/PasswordHasherTest.cs diff --git a/Tests/PlanTempus.X.TDD.csproj b/PlanTempus.Tests/PlanTempus.X.TDD.csproj similarity index 77% rename from Tests/PlanTempus.X.TDD.csproj rename to PlanTempus.Tests/PlanTempus.X.TDD.csproj index f411132..0b14a69 100644 --- a/Tests/PlanTempus.X.TDD.csproj +++ b/PlanTempus.Tests/PlanTempus.X.TDD.csproj @@ -9,23 +9,23 @@ - - - - - - - + + + + + + + - - + + - + diff --git a/Tests/PostgresTests.cs b/PlanTempus.Tests/PostgresTests.cs similarity index 100% rename from Tests/PostgresTests.cs rename to PlanTempus.Tests/PostgresTests.cs diff --git a/Tests/SecureConnectionStringTests.cs b/PlanTempus.Tests/SecureConnectionStringTests.cs similarity index 100% rename from Tests/SecureConnectionStringTests.cs rename to PlanTempus.Tests/SecureConnectionStringTests.cs diff --git a/Tests/TestFixture.cs b/PlanTempus.Tests/TestFixture.cs similarity index 100% rename from Tests/TestFixture.cs rename to PlanTempus.Tests/TestFixture.cs diff --git a/Tests/appconfiguration.dev.json b/PlanTempus.Tests/appconfiguration.dev.json similarity index 100% rename from Tests/appconfiguration.dev.json rename to PlanTempus.Tests/appconfiguration.dev.json diff --git a/PlanTempus.X.BDD/BddTestFixture.cs b/PlanTempus.X.BDD/BddTestFixture.cs index f8f7b3a..ed39b91 100644 --- a/PlanTempus.X.BDD/BddTestFixture.cs +++ b/PlanTempus.X.BDD/BddTestFixture.cs @@ -11,6 +11,7 @@ using PlanTempus.Core.Email; using PlanTempus.Core.ModuleRegistry; using PlanTempus.Core.Outbox; using PlanTempus.Core.SeqLogging; +using PlanTempus.Core.Telemetry; using PlanTempus.Database.ModuleRegistry; using CrypticWizard.RandomWordGenerator; @@ -84,6 +85,7 @@ public abstract class BddTestFixture : FeatureFixture PostmarkConfiguration = configuration.GetSection("Postmark").ToObject() }); + builder.RegisterType().As>().SingleInstance(); builder.RegisterType().SingleInstance(); builder.RegisterType().SingleInstance(); @@ -104,8 +106,11 @@ public abstract class BddTestFixture : FeatureFixture [TestCleanup] public void CleanupContainer() { + // 1. Stop OutboxListener først (så der ikke genereres mere telemetry) _cts?.Cancel(); _outboxListener?.StopAsync(CancellationToken.None).Wait(); + + // 2. Stop SeqBackgroundService - den sender stop signal, dræner channel og lukker ned _seqBackgroundService?.StopAsync(CancellationToken.None).Wait(); Trace.Flush(); @@ -121,7 +126,7 @@ public abstract class BddTestFixture : FeatureFixture } catch (System.Threading.Channels.ChannelClosedException) { - // Channel already closed by SeqBackgroundService.StopAsync + // Channel already closed } Container = null; diff --git a/PlanTempus.X.BDD/PlanTempus.X.BDD.csproj b/PlanTempus.X.BDD/PlanTempus.X.BDD.csproj index a3190fc..0b4b97d 100644 --- a/PlanTempus.X.BDD/PlanTempus.X.BDD.csproj +++ b/PlanTempus.X.BDD/PlanTempus.X.BDD.csproj @@ -22,10 +22,10 @@ - - - + + + diff --git a/PlanTempus.sln b/PlanTempus.sln index e82f7ea..6a0a8f2 100644 --- a/PlanTempus.sln +++ b/PlanTempus.sln @@ -2,15 +2,13 @@ Microsoft Visual Studio Solution File, Format Version 12.00 # Visual Studio Version 17 VisualStudioVersion = 17.10.35013.160 MinimumVisualStudioVersion = 10.0.40219.1 -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.Core", "Core\PlanTempus.Core.csproj", "{7B554252-1CE4-44BD-B108-B0BDCCB24742}" +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.Core", "PlanTempus.Core\PlanTempus.Core.csproj", "{7B554252-1CE4-44BD-B108-B0BDCCB24742}" EndProject -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.X.TDD", "Tests\PlanTempus.X.TDD.csproj", "{85614050-CFB0-4E39-81D3-7D99946449D9}" +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.X.TDD", "PlanTempus.Tests\PlanTempus.X.TDD.csproj", "{85614050-CFB0-4E39-81D3-7D99946449D9}" EndProject -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.Application", "Application\PlanTempus.Application.csproj", "{111CE8AE-E637-4376-A5A3-88D33E77EA88}" +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.Database", "PlanTempus.Database\PlanTempus.Database.csproj", "{D5096A7F-E6D4-4C87-874E-2D9A6BEAD57F}" EndProject -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.Database", "Database\PlanTempus.Database.csproj", "{D5096A7F-E6D4-4C87-874E-2D9A6BEAD57F}" -EndProject -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.SetupInfrastructure", "SetupInfrastructure\PlanTempus.SetupInfrastructure.csproj", "{48300227-BCBB-45A3-8359-9064DA85B1F9}" +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PlanTempus.SetupInfrastructure", "PlanTempus.SetupInfrastructure\PlanTempus.SetupInfrastructure.csproj", "{48300227-BCBB-45A3-8359-9064DA85B1F9}" EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PlanTempus.X.BDD", "PlanTempus.X.BDD\PlanTempus.X.BDD.csproj", "{8CA2246B-7D8C-40DA-9042-CA17A7A7672B}" EndProject @@ -18,6 +16,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PlanTempus.Components", "Pl EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "TestPostgresql", "TestPostgresLISTEN\TestPostgresql.csproj", "{67C167C4-8086-0556-39DA-5F9DF6CEE51F}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PlanTempus.Application", "PlanTempus.Application\PlanTempus.Application.csproj", "{7B846B69-FB5B-799B-02A7-4F1AADC6F7C9}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -32,10 +32,6 @@ Global {85614050-CFB0-4E39-81D3-7D99946449D9}.Debug|Any CPU.Build.0 = Debug|Any CPU {85614050-CFB0-4E39-81D3-7D99946449D9}.Release|Any CPU.ActiveCfg = Release|Any CPU {85614050-CFB0-4E39-81D3-7D99946449D9}.Release|Any CPU.Build.0 = Release|Any CPU - {111CE8AE-E637-4376-A5A3-88D33E77EA88}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {111CE8AE-E637-4376-A5A3-88D33E77EA88}.Debug|Any CPU.Build.0 = Debug|Any CPU - {111CE8AE-E637-4376-A5A3-88D33E77EA88}.Release|Any CPU.ActiveCfg = Release|Any CPU - {111CE8AE-E637-4376-A5A3-88D33E77EA88}.Release|Any CPU.Build.0 = Release|Any CPU {D5096A7F-E6D4-4C87-874E-2D9A6BEAD57F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU {D5096A7F-E6D4-4C87-874E-2D9A6BEAD57F}.Debug|Any CPU.Build.0 = Debug|Any CPU {D5096A7F-E6D4-4C87-874E-2D9A6BEAD57F}.Release|Any CPU.ActiveCfg = Release|Any CPU @@ -56,6 +52,10 @@ Global {67C167C4-8086-0556-39DA-5F9DF6CEE51F}.Debug|Any CPU.Build.0 = Debug|Any CPU {67C167C4-8086-0556-39DA-5F9DF6CEE51F}.Release|Any CPU.ActiveCfg = Release|Any CPU {67C167C4-8086-0556-39DA-5F9DF6CEE51F}.Release|Any CPU.Build.0 = Release|Any CPU + {7B846B69-FB5B-799B-02A7-4F1AADC6F7C9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {7B846B69-FB5B-799B-02A7-4F1AADC6F7C9}.Debug|Any CPU.Build.0 = Debug|Any CPU + {7B846B69-FB5B-799B-02A7-4F1AADC6F7C9}.Release|Any CPU.ActiveCfg = Release|Any CPU + {7B846B69-FB5B-799B-02A7-4F1AADC6F7C9}.Release|Any CPU.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE