From eeaa9de3c5c0b46435c43e5831ef3dc55aef822f Mon Sep 17 00:00:00 2001 From: local Date: Fri, 27 Mar 2026 22:51:55 +0000 Subject: [PATCH] feat(01-01): create three-project .NET 9 solution scaffold - ChatAgent.sln at repo root with Client, Api, Shared projects - Blazor WASM client, ASP.NET Core Web API, shared class library - Both Client and Api reference Shared project - Removed template boilerplate (WeatherForecast, Class1) - Added .NET ignores to .gitignore - Pinned .NET 9 SDK via global.json Co-Authored-By: Claude Opus 4.6 (1M context) --- .gitignore | 11 ++ ChatAgent.sln | 69 +++++++++++ global.json | 6 + src/ChatAgent.Api/ChatAgent.Api.csproj | 17 +++ src/ChatAgent.Api/Program.cs | 23 ++++ .../Properties/launchSettings.json | 23 ++++ .../appsettings.Development.json | 8 ++ src/ChatAgent.Api/appsettings.json | 9 ++ src/ChatAgent.Client/App.razor | 12 ++ src/ChatAgent.Client/ChatAgent.Client.csproj | 18 +++ src/ChatAgent.Client/Layout/MainLayout.razor | 16 +++ .../Layout/MainLayout.razor.css | 77 ++++++++++++ src/ChatAgent.Client/Layout/NavMenu.razor | 39 ++++++ src/ChatAgent.Client/Layout/NavMenu.razor.css | 83 +++++++++++++ src/ChatAgent.Client/Pages/Counter.razor | 18 +++ src/ChatAgent.Client/Pages/Home.razor | 7 ++ src/ChatAgent.Client/Pages/Weather.razor | 57 +++++++++ src/ChatAgent.Client/Program.cs | 11 ++ .../Properties/launchSettings.json | 25 ++++ src/ChatAgent.Client/_Imports.razor | 10 ++ src/ChatAgent.Client/wwwroot/css/app.css | 114 ++++++++++++++++++ src/ChatAgent.Client/wwwroot/favicon.png | Bin 0 -> 1148 bytes src/ChatAgent.Client/wwwroot/icon-192.png | Bin 0 -> 2626 bytes src/ChatAgent.Client/wwwroot/index.html | 32 +++++ .../wwwroot/sample-data/weather.json | 27 +++++ src/ChatAgent.Shared/ChatAgent.Shared.csproj | 9 ++ 26 files changed, 721 insertions(+) create mode 100644 ChatAgent.sln create mode 100644 global.json create mode 100644 src/ChatAgent.Api/ChatAgent.Api.csproj create mode 100644 src/ChatAgent.Api/Program.cs create mode 100644 src/ChatAgent.Api/Properties/launchSettings.json create mode 100644 src/ChatAgent.Api/appsettings.Development.json create mode 100644 src/ChatAgent.Api/appsettings.json create mode 100644 src/ChatAgent.Client/App.razor create mode 100644 src/ChatAgent.Client/ChatAgent.Client.csproj create mode 100644 src/ChatAgent.Client/Layout/MainLayout.razor create mode 100644 src/ChatAgent.Client/Layout/MainLayout.razor.css create mode 100644 src/ChatAgent.Client/Layout/NavMenu.razor create mode 100644 src/ChatAgent.Client/Layout/NavMenu.razor.css create mode 100644 src/ChatAgent.Client/Pages/Counter.razor create mode 100644 src/ChatAgent.Client/Pages/Home.razor create mode 100644 src/ChatAgent.Client/Pages/Weather.razor create mode 100644 src/ChatAgent.Client/Program.cs create mode 100644 src/ChatAgent.Client/Properties/launchSettings.json create mode 100644 src/ChatAgent.Client/_Imports.razor create mode 100644 src/ChatAgent.Client/wwwroot/css/app.css create mode 100644 src/ChatAgent.Client/wwwroot/favicon.png create mode 100644 src/ChatAgent.Client/wwwroot/icon-192.png create mode 100644 src/ChatAgent.Client/wwwroot/index.html create mode 100644 src/ChatAgent.Client/wwwroot/sample-data/weather.json create mode 100644 src/ChatAgent.Shared/ChatAgent.Shared.csproj diff --git a/.gitignore b/.gitignore index 63775f1..17f6a74 100644 --- a/.gitignore +++ b/.gitignore @@ -216,3 +216,14 @@ pyvenv.cfg .venv pip-selfcheck.json +# ---> .NET / C# +bin/ +obj/ +*.user +*.suo +.vs/ +.idea/ +*.swp +**/wwwroot/_framework/ +*.DotSettings.user + diff --git a/ChatAgent.sln b/ChatAgent.sln new file mode 100644 index 0000000..356303c --- /dev/null +++ b/ChatAgent.sln @@ -0,0 +1,69 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 17 +VisualStudioVersion = 17.0.31903.59 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "src", "src", "{827E0CD3-B72D-47B6-A68D-7590B98EB39B}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ChatAgent.Client", "src\ChatAgent.Client\ChatAgent.Client.csproj", "{600EA0C4-7CDE-4807-BE3C-30A6D2242392}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ChatAgent.Api", "src\ChatAgent.Api\ChatAgent.Api.csproj", "{467D4550-6F9A-456E-B99C-0ABE94070ECF}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ChatAgent.Shared", "src\ChatAgent.Shared\ChatAgent.Shared.csproj", "{06182E3F-BC78-449B-ADF6-D9EE49E48945}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Debug|x64 = Debug|x64 + Debug|x86 = Debug|x86 + Release|Any CPU = Release|Any CPU + Release|x64 = Release|x64 + Release|x86 = Release|x86 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Debug|Any CPU.Build.0 = Debug|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Debug|x64.ActiveCfg = Debug|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Debug|x64.Build.0 = Debug|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Debug|x86.ActiveCfg = Debug|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Debug|x86.Build.0 = Debug|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Release|Any CPU.ActiveCfg = Release|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Release|Any CPU.Build.0 = Release|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Release|x64.ActiveCfg = Release|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Release|x64.Build.0 = Release|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Release|x86.ActiveCfg = Release|Any CPU + {600EA0C4-7CDE-4807-BE3C-30A6D2242392}.Release|x86.Build.0 = Release|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Debug|Any CPU.Build.0 = Debug|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Debug|x64.ActiveCfg = Debug|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Debug|x64.Build.0 = Debug|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Debug|x86.ActiveCfg = Debug|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Debug|x86.Build.0 = Debug|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Release|Any CPU.ActiveCfg = Release|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Release|Any CPU.Build.0 = Release|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Release|x64.ActiveCfg = Release|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Release|x64.Build.0 = Release|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Release|x86.ActiveCfg = Release|Any CPU + {467D4550-6F9A-456E-B99C-0ABE94070ECF}.Release|x86.Build.0 = Release|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Debug|Any CPU.Build.0 = Debug|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Debug|x64.ActiveCfg = Debug|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Debug|x64.Build.0 = Debug|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Debug|x86.ActiveCfg = Debug|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Debug|x86.Build.0 = Debug|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Release|Any CPU.ActiveCfg = Release|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Release|Any CPU.Build.0 = Release|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Release|x64.ActiveCfg = Release|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Release|x64.Build.0 = Release|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Release|x86.ActiveCfg = Release|Any CPU + {06182E3F-BC78-449B-ADF6-D9EE49E48945}.Release|x86.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(NestedProjects) = preSolution + {600EA0C4-7CDE-4807-BE3C-30A6D2242392} = {827E0CD3-B72D-47B6-A68D-7590B98EB39B} + {467D4550-6F9A-456E-B99C-0ABE94070ECF} = {827E0CD3-B72D-47B6-A68D-7590B98EB39B} + {06182E3F-BC78-449B-ADF6-D9EE49E48945} = {827E0CD3-B72D-47B6-A68D-7590B98EB39B} + EndGlobalSection +EndGlobal diff --git a/global.json b/global.json new file mode 100644 index 0000000..ce8c0c4 --- /dev/null +++ b/global.json @@ -0,0 +1,6 @@ +{ + "sdk": { + "version": "9.0.312", + "rollForward": "latestPatch" + } +} diff --git a/src/ChatAgent.Api/ChatAgent.Api.csproj b/src/ChatAgent.Api/ChatAgent.Api.csproj new file mode 100644 index 0000000..86a2035 --- /dev/null +++ b/src/ChatAgent.Api/ChatAgent.Api.csproj @@ -0,0 +1,17 @@ + + + + net9.0 + enable + enable + + + + + + + + + + + diff --git a/src/ChatAgent.Api/Program.cs b/src/ChatAgent.Api/Program.cs new file mode 100644 index 0000000..6189b36 --- /dev/null +++ b/src/ChatAgent.Api/Program.cs @@ -0,0 +1,23 @@ +var builder = WebApplication.CreateBuilder(args); + +// Add services to the container. + +builder.Services.AddControllers(); +// Learn more about configuring OpenAPI at https://aka.ms/aspnet/openapi +builder.Services.AddOpenApi(); + +var app = builder.Build(); + +// Configure the HTTP request pipeline. +if (app.Environment.IsDevelopment()) +{ + app.MapOpenApi(); +} + +app.UseHttpsRedirection(); + +app.UseAuthorization(); + +app.MapControllers(); + +app.Run(); diff --git a/src/ChatAgent.Api/Properties/launchSettings.json b/src/ChatAgent.Api/Properties/launchSettings.json new file mode 100644 index 0000000..c826061 --- /dev/null +++ b/src/ChatAgent.Api/Properties/launchSettings.json @@ -0,0 +1,23 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": false, + "applicationUrl": "http://localhost:5032", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "https": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": false, + "applicationUrl": "https://localhost:7241;http://localhost:5032", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } +} diff --git a/src/ChatAgent.Api/appsettings.Development.json b/src/ChatAgent.Api/appsettings.Development.json new file mode 100644 index 0000000..ff66ba6 --- /dev/null +++ b/src/ChatAgent.Api/appsettings.Development.json @@ -0,0 +1,8 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} diff --git a/src/ChatAgent.Api/appsettings.json b/src/ChatAgent.Api/appsettings.json new file mode 100644 index 0000000..4d56694 --- /dev/null +++ b/src/ChatAgent.Api/appsettings.json @@ -0,0 +1,9 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + }, + "AllowedHosts": "*" +} diff --git a/src/ChatAgent.Client/App.razor b/src/ChatAgent.Client/App.razor new file mode 100644 index 0000000..623580d --- /dev/null +++ b/src/ChatAgent.Client/App.razor @@ -0,0 +1,12 @@ + + + + + + + Not found + +

Sorry, there's nothing at this address.

+
+
+
diff --git a/src/ChatAgent.Client/ChatAgent.Client.csproj b/src/ChatAgent.Client/ChatAgent.Client.csproj new file mode 100644 index 0000000..6e3a9b4 --- /dev/null +++ b/src/ChatAgent.Client/ChatAgent.Client.csproj @@ -0,0 +1,18 @@ + + + + net9.0 + enable + enable + + + + + + + + + + + + diff --git a/src/ChatAgent.Client/Layout/MainLayout.razor b/src/ChatAgent.Client/Layout/MainLayout.razor new file mode 100644 index 0000000..e465845 --- /dev/null +++ b/src/ChatAgent.Client/Layout/MainLayout.razor @@ -0,0 +1,16 @@ +@inherits LayoutComponentBase +
+ + +
+
+ About +
+ +
+ @Body +
+
+
diff --git a/src/ChatAgent.Client/Layout/MainLayout.razor.css b/src/ChatAgent.Client/Layout/MainLayout.razor.css new file mode 100644 index 0000000..baef3ee --- /dev/null +++ b/src/ChatAgent.Client/Layout/MainLayout.razor.css @@ -0,0 +1,77 @@ +.page { + position: relative; + display: flex; + flex-direction: column; +} + +main { + flex: 1; +} + +.sidebar { + background-image: linear-gradient(180deg, rgb(5, 39, 103) 0%, #3a0647 70%); +} + +.top-row { + background-color: #f7f7f7; + border-bottom: 1px solid #d6d5d5; + justify-content: flex-end; + height: 3.5rem; + display: flex; + align-items: center; +} + + .top-row ::deep a, .top-row ::deep .btn-link { + white-space: nowrap; + margin-left: 1.5rem; + text-decoration: none; + } + + .top-row ::deep a:hover, .top-row ::deep .btn-link:hover { + text-decoration: underline; + } + + .top-row ::deep a:first-child { + overflow: hidden; + text-overflow: ellipsis; + } + +@media (max-width: 640.98px) { + .top-row { + justify-content: space-between; + } + + .top-row ::deep a, .top-row ::deep .btn-link { + margin-left: 0; + } +} + +@media (min-width: 641px) { + .page { + flex-direction: row; + } + + .sidebar { + width: 250px; + height: 100vh; + position: sticky; + top: 0; + } + + .top-row { + position: sticky; + top: 0; + z-index: 1; + } + + .top-row.auth ::deep a:first-child { + flex: 1; + text-align: right; + width: 0; + } + + .top-row, article { + padding-left: 2rem !important; + padding-right: 1.5rem !important; + } +} diff --git a/src/ChatAgent.Client/Layout/NavMenu.razor b/src/ChatAgent.Client/Layout/NavMenu.razor new file mode 100644 index 0000000..18398fa --- /dev/null +++ b/src/ChatAgent.Client/Layout/NavMenu.razor @@ -0,0 +1,39 @@ + + + + +@code { + private bool collapseNavMenu = true; + + private string? NavMenuCssClass => collapseNavMenu ? "collapse" : null; + + private void ToggleNavMenu() + { + collapseNavMenu = !collapseNavMenu; + } +} diff --git a/src/ChatAgent.Client/Layout/NavMenu.razor.css b/src/ChatAgent.Client/Layout/NavMenu.razor.css new file mode 100644 index 0000000..6724fd1 --- /dev/null +++ b/src/ChatAgent.Client/Layout/NavMenu.razor.css @@ -0,0 +1,83 @@ +.navbar-toggler { + background-color: rgba(255, 255, 255, 0.1); +} + +.top-row { + min-height: 3.5rem; + background-color: rgba(0,0,0,0.4); +} + +.navbar-brand { + font-size: 1.1rem; +} + +.bi { + display: inline-block; + position: relative; + width: 1.25rem; + height: 1.25rem; + margin-right: 0.75rem; + top: -1px; + background-size: cover; +} + +.bi-house-door-fill-nav-menu { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-house-door-fill' viewBox='0 0 16 16'%3E%3Cpath d='M6.5 14.5v-3.505c0-.245.25-.495.5-.495h2c.25 0 .5.25.5.5v3.5a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5v-7a.5.5 0 0 0-.146-.354L13 5.793V2.5a.5.5 0 0 0-.5-.5h-1a.5.5 0 0 0-.5.5v1.293L8.354 1.146a.5.5 0 0 0-.708 0l-6 6A.5.5 0 0 0 1.5 7.5v7a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5Z'/%3E%3C/svg%3E"); +} + +.bi-plus-square-fill-nav-menu { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-plus-square-fill' viewBox='0 0 16 16'%3E%3Cpath d='M2 0a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V2a2 2 0 0 0-2-2H2zm6.5 4.5v3h3a.5.5 0 0 1 0 1h-3v3a.5.5 0 0 1-1 0v-3h-3a.5.5 0 0 1 0-1h3v-3a.5.5 0 0 1 1 0z'/%3E%3C/svg%3E"); +} + +.bi-list-nested-nav-menu { + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-list-nested' viewBox='0 0 16 16'%3E%3Cpath fill-rule='evenodd' d='M4.5 11.5A.5.5 0 0 1 5 11h10a.5.5 0 0 1 0 1H5a.5.5 0 0 1-.5-.5zm-2-4A.5.5 0 0 1 3 7h10a.5.5 0 0 1 0 1H3a.5.5 0 0 1-.5-.5zm-2-4A.5.5 0 0 1 1 3h10a.5.5 0 0 1 0 1H1a.5.5 0 0 1-.5-.5z'/%3E%3C/svg%3E"); +} + +.nav-item { + font-size: 0.9rem; + padding-bottom: 0.5rem; +} + + .nav-item:first-of-type { + padding-top: 1rem; + } + + .nav-item:last-of-type { + padding-bottom: 1rem; + } + + .nav-item ::deep a { + color: #d7d7d7; + border-radius: 4px; + height: 3rem; + display: flex; + align-items: center; + line-height: 3rem; + } + +.nav-item ::deep a.active { + background-color: rgba(255,255,255,0.37); + color: white; +} + +.nav-item ::deep a:hover { + background-color: rgba(255,255,255,0.1); + color: white; +} + +@media (min-width: 641px) { + .navbar-toggler { + display: none; + } + + .collapse { + /* Never collapse the sidebar for wide screens */ + display: block; + } + + .nav-scrollable { + /* Allow sidebar to scroll for tall menus */ + height: calc(100vh - 3.5rem); + overflow-y: auto; + } +} diff --git a/src/ChatAgent.Client/Pages/Counter.razor b/src/ChatAgent.Client/Pages/Counter.razor new file mode 100644 index 0000000..b21f052 --- /dev/null +++ b/src/ChatAgent.Client/Pages/Counter.razor @@ -0,0 +1,18 @@ +@page "/counter" + +Counter + +

Counter

+ +

Current count: @currentCount

+ + + +@code { + private int currentCount = 0; + + private void IncrementCount() + { + currentCount++; + } +} diff --git a/src/ChatAgent.Client/Pages/Home.razor b/src/ChatAgent.Client/Pages/Home.razor new file mode 100644 index 0000000..df05023 --- /dev/null +++ b/src/ChatAgent.Client/Pages/Home.razor @@ -0,0 +1,7 @@ +@page "/" + +Home + +

Hello, world!

+ +Welcome to your new app. diff --git a/src/ChatAgent.Client/Pages/Weather.razor b/src/ChatAgent.Client/Pages/Weather.razor new file mode 100644 index 0000000..70a1c0f --- /dev/null +++ b/src/ChatAgent.Client/Pages/Weather.razor @@ -0,0 +1,57 @@ +@page "/weather" +@inject HttpClient Http + +Weather + +

Weather

+ +

This component demonstrates fetching data from the server.

+ +@if (forecasts == null) +{ +

Loading...

+} +else +{ + + + + + + + + + + + @foreach (var forecast in forecasts) + { + + + + + + + } + +
DateTemp. (C)Temp. (F)Summary
@forecast.Date.ToShortDateString()@forecast.TemperatureC@forecast.TemperatureF@forecast.Summary
+} + +@code { + private WeatherForecast[]? forecasts; + + protected override async Task OnInitializedAsync() + { + forecasts = await Http.GetFromJsonAsync("sample-data/weather.json"); + } + + public class WeatherForecast + { + public DateOnly Date { get; set; } + + public int TemperatureC { get; set; } + + public string? Summary { get; set; } + + public int TemperatureF => 32 + (int)(TemperatureC / 0.5556); + } +} diff --git a/src/ChatAgent.Client/Program.cs b/src/ChatAgent.Client/Program.cs new file mode 100644 index 0000000..d93447e --- /dev/null +++ b/src/ChatAgent.Client/Program.cs @@ -0,0 +1,11 @@ +using Microsoft.AspNetCore.Components.Web; +using Microsoft.AspNetCore.Components.WebAssembly.Hosting; +using ChatAgent.Client; + +var builder = WebAssemblyHostBuilder.CreateDefault(args); +builder.RootComponents.Add("#app"); +builder.RootComponents.Add("head::after"); + +builder.Services.AddScoped(sp => new HttpClient { BaseAddress = new Uri(builder.HostEnvironment.BaseAddress) }); + +await builder.Build().RunAsync(); diff --git a/src/ChatAgent.Client/Properties/launchSettings.json b/src/ChatAgent.Client/Properties/launchSettings.json new file mode 100644 index 0000000..8f98434 --- /dev/null +++ b/src/ChatAgent.Client/Properties/launchSettings.json @@ -0,0 +1,25 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", + "applicationUrl": "http://localhost:5039", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "https": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", + "applicationUrl": "https://localhost:7296;http://localhost:5039", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } +} diff --git a/src/ChatAgent.Client/_Imports.razor b/src/ChatAgent.Client/_Imports.razor new file mode 100644 index 0000000..f256289 --- /dev/null +++ b/src/ChatAgent.Client/_Imports.razor @@ -0,0 +1,10 @@ +@using System.Net.Http +@using System.Net.Http.Json +@using Microsoft.AspNetCore.Components.Forms +@using Microsoft.AspNetCore.Components.Routing +@using Microsoft.AspNetCore.Components.Web +@using Microsoft.AspNetCore.Components.Web.Virtualization +@using Microsoft.AspNetCore.Components.WebAssembly.Http +@using Microsoft.JSInterop +@using ChatAgent.Client +@using ChatAgent.Client.Layout diff --git a/src/ChatAgent.Client/wwwroot/css/app.css b/src/ChatAgent.Client/wwwroot/css/app.css new file mode 100644 index 0000000..d919e04 --- /dev/null +++ b/src/ChatAgent.Client/wwwroot/css/app.css @@ -0,0 +1,114 @@ +html, body { + font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; +} + +h1:focus { + outline: none; +} + +a, .btn-link { + color: #0071c1; +} + +.btn-primary { + color: #fff; + background-color: #1b6ec2; + border-color: #1861ac; +} + +.btn:focus, .btn:active:focus, .btn-link.nav-link:focus, .form-control:focus, .form-check-input:focus { + box-shadow: 0 0 0 0.1rem white, 0 0 0 0.25rem #258cfb; +} + +.content { + padding-top: 1.1rem; +} + +.valid.modified:not([type=checkbox]) { + outline: 1px solid #26b050; +} + +.invalid { + outline: 1px solid red; +} + +.validation-message { + color: red; +} + +#blazor-error-ui { + color-scheme: light only; + background: lightyellow; + bottom: 0; + box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2); + box-sizing: border-box; + display: none; + left: 0; + padding: 0.6rem 1.25rem 0.7rem 1.25rem; + position: fixed; + width: 100%; + z-index: 1000; +} + + #blazor-error-ui .dismiss { + cursor: pointer; + position: absolute; + right: 0.75rem; + top: 0.5rem; + } + +.blazor-error-boundary { + background: url(data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNTYiIGhlaWdodD0iNDkiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIG92ZXJmbG93PSJoaWRkZW4iPjxkZWZzPjxjbGlwUGF0aCBpZD0iY2xpcDAiPjxyZWN0IHg9IjIzNSIgeT0iNTEiIHdpZHRoPSI1NiIgaGVpZ2h0PSI0OSIvPjwvY2xpcFBhdGg+PC9kZWZzPjxnIGNsaXAtcGF0aD0idXJsKCNjbGlwMCkiIHRyYW5zZm9ybT0idHJhbnNsYXRlKC0yMzUgLTUxKSI+PHBhdGggZD0iTTI2My41MDYgNTFDMjY0LjcxNyA1MSAyNjUuODEzIDUxLjQ4MzcgMjY2LjYwNiA1Mi4yNjU4TDI2Ny4wNTIgNTIuNzk4NyAyNjcuNTM5IDUzLjYyODMgMjkwLjE4NSA5Mi4xODMxIDI5MC41NDUgOTIuNzk1IDI5MC42NTYgOTIuOTk2QzI5MC44NzcgOTMuNTEzIDI5MSA5NC4wODE1IDI5MSA5NC42NzgyIDI5MSA5Ny4wNjUxIDI4OS4wMzggOTkgMjg2LjYxNyA5OUwyNDAuMzgzIDk5QzIzNy45NjMgOTkgMjM2IDk3LjA2NTEgMjM2IDk0LjY3ODIgMjM2IDk0LjM3OTkgMjM2LjAzMSA5NC4wODg2IDIzNi4wODkgOTMuODA3MkwyMzYuMzM4IDkzLjAxNjIgMjM2Ljg1OCA5Mi4xMzE0IDI1OS40NzMgNTMuNjI5NCAyNTkuOTYxIDUyLjc5ODUgMjYwLjQwNyA1Mi4yNjU4QzI2MS4yIDUxLjQ4MzcgMjYyLjI5NiA1MSAyNjMuNTA2IDUxWk0yNjMuNTg2IDY2LjAxODNDMjYwLjczNyA2Ni4wMTgzIDI1OS4zMTMgNjcuMTI0NSAyNTkuMzEzIDY5LjMzNyAyNTkuMzEzIDY5LjYxMDIgMjU5LjMzMiA2OS44NjA4IDI1OS4zNzEgNzAuMDg4N0wyNjEuNzk1IDg0LjAxNjEgMjY1LjM4IDg0LjAxNjEgMjY3LjgyMSA2OS43NDc1QzI2Ny44NiA2OS43MzA5IDI2Ny44NzkgNjkuNTg3NyAyNjcuODc5IDY5LjMxNzkgMjY3Ljg3OSA2Ny4xMTgyIDI2Ni40NDggNjYuMDE4MyAyNjMuNTg2IDY2LjAxODNaTTI2My41NzYgODYuMDU0N0MyNjEuMDQ5IDg2LjA1NDcgMjU5Ljc4NiA4Ny4zMDA1IDI1OS43ODYgODkuNzkyMSAyNTkuNzg2IDkyLjI4MzcgMjYxLjA0OSA5My41Mjk1IDI2My41NzYgOTMuNTI5NSAyNjYuMTE2IDkzLjUyOTUgMjY3LjM4NyA5Mi4yODM3IDI2Ny4zODcgODkuNzkyMSAyNjcuMzg3IDg3LjMwMDUgMjY2LjExNiA4Ni4wNTQ3IDI2My41NzYgODYuMDU0N1oiIGZpbGw9IiNGRkU1MDAiIGZpbGwtcnVsZT0iZXZlbm9kZCIvPjwvZz48L3N2Zz4=) no-repeat 1rem/1.8rem, #b32121; + padding: 1rem 1rem 1rem 3.7rem; + color: white; +} + + .blazor-error-boundary::after { + content: "An error has occurred." + } + +.loading-progress { + position: relative; + display: block; + width: 8rem; + height: 8rem; + margin: 20vh auto 1rem auto; +} + + .loading-progress circle { + fill: none; + stroke: #e0e0e0; + stroke-width: 0.6rem; + transform-origin: 50% 50%; + transform: rotate(-90deg); + } + + .loading-progress circle:last-child { + stroke: #1b6ec2; + stroke-dasharray: calc(3.141 * var(--blazor-load-percentage, 0%) * 0.8), 500%; + transition: stroke-dasharray 0.05s ease-in-out; + } + +.loading-progress-text { + position: absolute; + text-align: center; + font-weight: bold; + inset: calc(20vh + 3.25rem) 0 auto 0.2rem; +} + + .loading-progress-text:after { + content: var(--blazor-load-percentage-text, "Loading"); + } + +code { + color: #c02d76; +} + +.form-floating > .form-control-plaintext::placeholder, .form-floating > .form-control::placeholder { + color: var(--bs-secondary-color); + text-align: end; +} + +.form-floating > .form-control-plaintext:focus::placeholder, .form-floating > .form-control:focus::placeholder { + text-align: start; +} \ No newline at end of file diff --git a/src/ChatAgent.Client/wwwroot/favicon.png b/src/ChatAgent.Client/wwwroot/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..8422b59695935d180d11d5dbe99653e711097819 GIT binary patch literal 1148 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 literal 0 HcmV?d00001 diff --git a/src/ChatAgent.Client/wwwroot/index.html b/src/ChatAgent.Client/wwwroot/index.html new file mode 100644 index 0000000..20bca18 --- /dev/null +++ b/src/ChatAgent.Client/wwwroot/index.html @@ -0,0 +1,32 @@ + + + + + + + ChatAgent.Client + + + + + + + + +
+ + + + +
+
+ +
+ An unhandled error has occurred. + Reload + 🗙 +
+ + + + diff --git a/src/ChatAgent.Client/wwwroot/sample-data/weather.json b/src/ChatAgent.Client/wwwroot/sample-data/weather.json new file mode 100644 index 0000000..f0648e7 --- /dev/null +++ b/src/ChatAgent.Client/wwwroot/sample-data/weather.json @@ -0,0 +1,27 @@ +[ + { + "date": "2022-01-06", + "temperatureC": 1, + "summary": "Freezing" + }, + { + "date": "2022-01-07", + "temperatureC": 14, + "summary": "Bracing" + }, + { + "date": "2022-01-08", + "temperatureC": -13, + "summary": "Freezing" + }, + { + "date": "2022-01-09", + "temperatureC": -16, + "summary": "Balmy" + }, + { + "date": "2022-01-10", + "temperatureC": -2, + "summary": "Chilly" + } +] diff --git a/src/ChatAgent.Shared/ChatAgent.Shared.csproj b/src/ChatAgent.Shared/ChatAgent.Shared.csproj new file mode 100644 index 0000000..125f4c9 --- /dev/null +++ b/src/ChatAgent.Shared/ChatAgent.Shared.csproj @@ -0,0 +1,9 @@ + + + + net9.0 + enable + enable + + +