diff --git a/.gitignore b/.gitignore index dd31507..a06a9d7 100644 --- a/.gitignore +++ b/.gitignore @@ -3,9 +3,15 @@ ## ## Get latest from `dotnet new gitignore` +# Sqlite database files +*.db + # dotenv files .env +# Certificates +*.pem + # User-specific files *.rsuser *.suo @@ -481,4 +487,4 @@ $RECYCLE.BIN/ *.lnk # Vim temporary swap files -*.swp +*.swp \ No newline at end of file diff --git a/Directory.Packages.props b/Directory.Packages.props index fec2033..aa7f20d 100644 --- a/Directory.Packages.props +++ b/Directory.Packages.props @@ -4,15 +4,20 @@ true - - - - - - - - - - + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/NSeguin.Dev.Web.sln b/NSeguin.Dev.Web.sln index ee10fa0..ee14297 100644 --- a/NSeguin.Dev.Web.sln +++ b/NSeguin.Dev.Web.sln @@ -1,64 +1,34 @@ + Microsoft Visual Studio Solution File, Format Version 12.00 -# Visual Studio Version 16 -VisualStudioVersion = 16.0.0.0 -MinimumVisualStudioVersion = 16.0.0.0 -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "NSeguin.Dev.Web.Server", "src\Server\NSeguin.Dev.Web.Server.csproj", "{6ADDE87B-085E-402D-A843-36DCB7CAE8CF}" +# Visual Studio Version 17 +VisualStudioVersion = 17.0.31903.59 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "src", "src", "{30B827BB-3B45-4D5C-B989-5471C7DE497B}" EndProject -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "NSeguin.Dev.Web.Client", "src\Client\NSeguin.Dev.Web.Client.csproj", "{1BD74ED6-0D41-498C-B686-D6008E64A769}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "NSeguin.Dev.Web.Client", "src\NSeguin.Dev.Web.Client\NSeguin.Dev.Web.Client.csproj", "{17BD1A48-1797-469E-8D83-24A5CCA6B131}" EndProject -Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "NSeguin.Dev.Web.Shared", "src\Shared\NSeguin.Dev.Web.Shared.csproj", "{E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "NSeguin.Dev.Web", "src\NSeguin.Dev.Web\NSeguin.Dev.Web.csproj", "{D9AAF405-A6F0-4B92-96C5-825F02E46FCF}" 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 - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Debug|Any CPU.Build.0 = Debug|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Debug|x64.ActiveCfg = Debug|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Debug|x64.Build.0 = Debug|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Debug|x86.ActiveCfg = Debug|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Debug|x86.Build.0 = Debug|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Release|Any CPU.ActiveCfg = Release|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Release|Any CPU.Build.0 = Release|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Release|x64.ActiveCfg = Release|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Release|x64.Build.0 = Release|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Release|x86.ActiveCfg = Release|Any CPU - {1BD74ED6-0D41-498C-B686-D6008E64A769}.Release|x86.Build.0 = Release|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Debug|Any CPU.Build.0 = Debug|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Debug|x64.ActiveCfg = Debug|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Debug|x64.Build.0 = Debug|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Debug|x86.ActiveCfg = Debug|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Debug|x86.Build.0 = Debug|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Release|Any CPU.ActiveCfg = Release|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Release|Any CPU.Build.0 = Release|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Release|x64.ActiveCfg = Release|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Release|x64.Build.0 = Release|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Release|x86.ActiveCfg = Release|Any CPU - {6ADDE87B-085E-402D-A843-36DCB7CAE8CF}.Release|x86.Build.0 = Release|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Debug|Any CPU.Build.0 = Debug|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Debug|x64.ActiveCfg = Debug|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Debug|x64.Build.0 = Debug|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Debug|x86.ActiveCfg = Debug|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Debug|x86.Build.0 = Debug|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Release|Any CPU.ActiveCfg = Release|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Release|Any CPU.Build.0 = Release|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Release|x64.ActiveCfg = Release|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Release|x64.Build.0 = Release|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Release|x86.ActiveCfg = Release|Any CPU - {E75CCB8B-53A3-4871-B0F5-1AC1D2CB451B}.Release|x86.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE EndGlobalSection - GlobalSection(ExtensibilityGlobals) = postSolution - SolutionGuid = {9422CB08-E3A1-45FD-97F0-8020A865A54B} + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {17BD1A48-1797-469E-8D83-24A5CCA6B131}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {17BD1A48-1797-469E-8D83-24A5CCA6B131}.Debug|Any CPU.Build.0 = Debug|Any CPU + {17BD1A48-1797-469E-8D83-24A5CCA6B131}.Release|Any CPU.ActiveCfg = Release|Any CPU + {17BD1A48-1797-469E-8D83-24A5CCA6B131}.Release|Any CPU.Build.0 = Release|Any CPU + {D9AAF405-A6F0-4B92-96C5-825F02E46FCF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {D9AAF405-A6F0-4B92-96C5-825F02E46FCF}.Debug|Any CPU.Build.0 = Debug|Any CPU + {D9AAF405-A6F0-4B92-96C5-825F02E46FCF}.Release|Any CPU.ActiveCfg = Release|Any CPU + {D9AAF405-A6F0-4B92-96C5-825F02E46FCF}.Release|Any CPU.Build.0 = Release|Any CPU EndGlobalSection -EndGlobal \ No newline at end of file + GlobalSection(NestedProjects) = preSolution + {17BD1A48-1797-469E-8D83-24A5CCA6B131} = {30B827BB-3B45-4D5C-B989-5471C7DE497B} + {D9AAF405-A6F0-4B92-96C5-825F02E46FCF} = {30B827BB-3B45-4D5C-B989-5471C7DE497B} + EndGlobalSection +EndGlobal diff --git a/StyleCop.ruleset b/StyleCop.ruleset index 35b688e..5865b6a 100644 --- a/StyleCop.ruleset +++ b/StyleCop.ruleset @@ -38,7 +38,7 @@ - + @@ -129,10 +129,10 @@ - + - + diff --git a/src/Client/NSeguin.Dev.Web.Client.csproj b/src/Client/NSeguin.Dev.Web.Client.csproj deleted file mode 100644 index 54332ce..0000000 --- a/src/Client/NSeguin.Dev.Web.Client.csproj +++ /dev/null @@ -1,19 +0,0 @@ - - - - net8.0 - false - - - - - - - - - - - - - - \ No newline at end of file diff --git a/src/Client/Pages/Authentication.razor b/src/Client/Pages/Authentication.razor deleted file mode 100644 index adc026c..0000000 --- a/src/Client/Pages/Authentication.razor +++ /dev/null @@ -1,10 +0,0 @@ -@page "/authentication/{action}" -@using Microsoft.AspNetCore.Components.WebAssembly.Authentication - - -@code{ - - [Parameter] - public string? Action { get; set; } - -} \ No newline at end of file diff --git a/src/Client/Pages/FetchData.razor b/src/Client/Pages/FetchData.razor deleted file mode 100644 index 87d55fc..0000000 --- a/src/Client/Pages/FetchData.razor +++ /dev/null @@ -1,60 +0,0 @@ -@page "/fetchdata" -@using NSeguin.Dev.Web.Shared -@using Microsoft.AspNetCore.Components.WebAssembly.Authentication -@using Microsoft.AspNetCore.Authorization -@attribute [Authorize] -@inject HttpClient Http - -Weather forecast - -

Weather forecast

- -

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() - { - try - { - forecasts = await Http.GetFromJsonAsync("WeatherForecast"); - } - catch (AccessTokenNotAvailableException exception) - { - exception.Redirect(); - } - } - -} \ No newline at end of file diff --git a/src/Client/Pages/Index.razor b/src/Client/Pages/Index.razor deleted file mode 100644 index 66ce12c..0000000 --- a/src/Client/Pages/Index.razor +++ /dev/null @@ -1,9 +0,0 @@ -@page "/" - -Index - -

Hello, world!

- -Welcome to your new app. - - \ No newline at end of file diff --git a/src/Client/Program.cs b/src/Client/Program.cs deleted file mode 100644 index 9b393c3..0000000 --- a/src/Client/Program.cs +++ /dev/null @@ -1,30 +0,0 @@ -using Microsoft.AspNetCore.Components.Web; -using Microsoft.AspNetCore.Components.WebAssembly.Authentication; -using Microsoft.AspNetCore.Components.WebAssembly.Hosting; - -using NSeguin.Dev.Web.Client; - -WebAssemblyHostBuilder builder = WebAssemblyHostBuilder.CreateDefault(args); -builder.RootComponents.Add("#app"); -builder.RootComponents.Add("head::after"); -builder.Services - .AddHttpClient( - "NSeguin.Dev.Web.ServerAPI", - client => client.BaseAddress = new Uri(builder.HostEnvironment.BaseAddress)) - .AddHttpMessageHandler(); - -// Supply HttpClient instances that include access tokens when making requests to the server project -builder.Services.AddScoped( - sp => sp.GetRequiredService().CreateClient("NSeguin.Dev.Web.ServerAPI")); - -builder.Services.AddMsalAuthentication( - options => - { - builder.Configuration.Bind("AzureAdB2C", options.ProviderOptions.Authentication); - options.ProviderOptions.DefaultAccessTokenScopes.Add( - "https://nseguindev.onmicrosoft.com/db78f018-cfcd-4bb4-b81e-c139a1f76af6/access_as_user"); - - options.ProviderOptions.LoginMode = "redirect"; - }); - -await builder.Build().RunAsync(); diff --git a/src/Client/Properties/launchSettings.json b/src/Client/Properties/launchSettings.json deleted file mode 100644 index 496586e..0000000 --- a/src/Client/Properties/launchSettings.json +++ /dev/null @@ -1,30 +0,0 @@ -{ - "iisSettings": { - "windowsAuthentication": false, - "anonymousAuthentication": true, - "iisExpress": { - "applicationUrl": "http://localhost:21374", - "sslPort": 44327 - } - }, - "profiles": { - "NSeguin.Dev.Web": { - "commandName": "Project", - "dotnetRunMessages": true, - "launchBrowser": true, - "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", - "applicationUrl": "https://localhost:7091;http://localhost:5198", - "environmentVariables": { - "ASPNETCORE_ENVIRONMENT": "Development" - } - }, - "IIS Express": { - "commandName": "IISExpress", - "launchBrowser": true, - "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", - "environmentVariables": { - "ASPNETCORE_ENVIRONMENT": "Development" - } - } - } -} diff --git a/src/Client/Shared/MainLayout.razor b/src/Client/Shared/MainLayout.razor deleted file mode 100644 index 3a0c23d..0000000 --- a/src/Client/Shared/MainLayout.razor +++ /dev/null @@ -1,17 +0,0 @@ -@inherits LayoutComponentBase - -
- - -
-
- -
- -
- @Body -
-
-
\ No newline at end of file diff --git a/src/Client/Shared/NavMenu.razor b/src/Client/Shared/NavMenu.razor deleted file mode 100644 index 699cb52..0000000 --- a/src/Client/Shared/NavMenu.razor +++ /dev/null @@ -1,40 +0,0 @@ - - -
- -
- -@code { - private bool collapseNavMenu = true; - - private string? NavMenuCssClass => collapseNavMenu ? "collapse" : null; - - private void ToggleNavMenu() - { - collapseNavMenu = !collapseNavMenu; - } - -} \ No newline at end of file diff --git a/src/Client/Shared/NavMenu.razor.css b/src/Client/Shared/NavMenu.razor.css deleted file mode 100644 index 1c7f712..0000000 --- a/src/Client/Shared/NavMenu.razor.css +++ /dev/null @@ -1,62 +0,0 @@ -.navbar-toggler { - background-color: rgba(255, 255, 255, 0.1); -} - -.top-row { - height: 3.5rem; - background-color: rgba(0, 0, 0, 0.4); -} - -.navbar-brand { - font-size: 1.1rem; -} - -.oi { - width: 2rem; - font-size: 1.1rem; - vertical-align: text-top; - top: -2px; -} - -.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.25); - 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; - } -} \ No newline at end of file diff --git a/src/Client/Shared/SurveyPrompt.razor b/src/Client/Shared/SurveyPrompt.razor deleted file mode 100644 index 88467b1..0000000 --- a/src/Client/Shared/SurveyPrompt.razor +++ /dev/null @@ -1,17 +0,0 @@ -
- - @Title - - - Please take our - brief survey - - and tell us what you think. -
- -@code { - // Demonstrates how a parent component can supply parameters - [Parameter] - public string? Title { get; set; } - -} \ No newline at end of file diff --git a/src/Client/wwwroot/appsettings.json b/src/Client/wwwroot/appsettings.json deleted file mode 100644 index 6f1ed95..0000000 --- a/src/Client/wwwroot/appsettings.json +++ /dev/null @@ -1,8 +0,0 @@ -{ - "AzureAdB2C": { - "Authority": "https://nseguindev.b2clogin.com/nseguindev.onmicrosoft.com/b2c_1_signin", - "ClientId": "eebf1b00-74e5-4d9e-a05c-0ceab9b382e9", - "ValidateAuthority": false, - "RedirectUri": "https://localhost:7091/authentication/login-callback" - } -} \ No newline at end of file diff --git a/src/Client/wwwroot/css/open-iconic/FONT-LICENSE b/src/Client/wwwroot/css/open-iconic/FONT-LICENSE deleted file mode 100644 index a1dc03f..0000000 --- a/src/Client/wwwroot/css/open-iconic/FONT-LICENSE +++ /dev/null @@ -1,86 +0,0 @@ -SIL OPEN FONT LICENSE Version 1.1 - -Copyright (c) 2014 Waybury - -PREAMBLE -The goals of the Open Font License (OFL) are to stimulate worldwide -development of collaborative font projects, to support the font creation -efforts of academic and linguistic communities, and to provide a free and -open framework in which fonts may be shared and improved in partnership -with others. - -The OFL allows the licensed fonts to be used, studied, modified and -redistributed freely as long as they are not sold by themselves. The -fonts, including any derivative works, can be bundled, embedded, -redistributed and/or sold with any software provided that any reserved -names are not used by derivative works. The fonts and derivatives, -however, cannot be released under any other type of license. The -requirement for fonts to remain under this license does not apply -to any document created using the fonts or their derivatives. - -DEFINITIONS -"Font Software" refers to the set of files released by the Copyright -Holder(s) under this license and clearly marked as such. This may -include source files, build scripts and documentation. - -"Reserved Font Name" refers to any names specified as such after the -copyright statement(s). - -"Original Version" refers to the collection of Font Software components as -distributed by the Copyright Holder(s). - -"Modified Version" refers to any derivative made by adding to, deleting, -or substituting -- in part or in whole -- any of the components of the -Original Version, by changing formats or by porting the Font Software to a -new environment. - -"Author" refers to any designer, engineer, programmer, technical -writer or other person who contributed to the Font Software. - -PERMISSION & CONDITIONS -Permission is hereby granted, free of charge, to any person obtaining -a copy of the Font Software, to use, study, copy, merge, embed, modify, -redistribute, and sell modified and unmodified copies of the Font -Software, subject to the following conditions: - -1) Neither the Font Software nor any of its individual components, -in Original or Modified Versions, may be sold by itself. - -2) Original or Modified Versions of the Font Software may be bundled, -redistributed and/or sold with any software, provided that each copy -contains the above copyright notice and this license. These can be -included either as stand-alone text files, human-readable headers or -in the appropriate machine-readable metadata fields within text or -binary files as long as those fields can be easily viewed by the user. - -3) No Modified Version of the Font Software may use the Reserved Font -Name(s) unless explicit written permission is granted by the corresponding -Copyright Holder. This restriction only applies to the primary font name as -presented to the users. - -4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font -Software shall not be used to promote, endorse or advertise any -Modified Version, except to acknowledge the contribution(s) of the -Copyright Holder(s) and the Author(s) or with their explicit written -permission. - -5) The Font Software, modified or unmodified, in part or in whole, -must be distributed entirely under this license, and must not be -distributed under any other license. The requirement for fonts to -remain under this license does not apply to any document created -using the Font Software. - -TERMINATION -This license becomes null and void if any of the above conditions are -not met. - -DISCLAIMER -THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF -MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT -OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE -COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, -INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL -DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING -FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM -OTHER DEALINGS IN THE FONT SOFTWARE. diff --git a/src/Client/wwwroot/css/open-iconic/ICON-LICENSE b/src/Client/wwwroot/css/open-iconic/ICON-LICENSE deleted file mode 100644 index 2199f4a..0000000 --- a/src/Client/wwwroot/css/open-iconic/ICON-LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -The MIT License (MIT) - -Copyright (c) 2014 Waybury - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in -all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -THE SOFTWARE. \ No newline at end of file diff --git a/src/Client/wwwroot/css/open-iconic/README.md b/src/Client/wwwroot/css/open-iconic/README.md deleted file mode 100644 index 50d1fc5..0000000 --- a/src/Client/wwwroot/css/open-iconic/README.md +++ /dev/null @@ -1,111 +0,0 @@ -[Open Iconic v1.1.1](https://github.com/iconic/open-iconic) -=========== - -### Open Iconic is the open source sibling of [Iconic](https://github.com/iconic/open-iconic). It is a hyper-legible collection of 223 icons with a tiny footprint—ready to use with Bootstrap and Foundation. [View the collection](https://github.com/iconic/open-iconic) - -## What's in Open Iconic? - -* 223 icons designed to be legible down to 8 pixels -* Super-light SVG files - 61.8 for the entire set -* SVG sprite—the modern replacement for icon fonts -* Webfont (EOT, OTF, SVG, TTF, WOFF), PNG and WebP formats -* Webfont stylesheets (including versions for Bootstrap and Foundation) in CSS, LESS, SCSS and Stylus formats -* PNG and WebP raster images in 8px, 16px, 24px, 32px, 48px and 64px. - -## Getting Started - -#### For code samples and everything else you need to get started with Open Iconic, check out our [Icons](https://github.com/iconic/open-iconic) and [Reference](https://github.com/iconic/open-iconic) sections. - -### General Usage - -#### Using Open Iconic's SVGs - -We like SVGs and we think they're the way to display icons on the web. Since Open Iconic are just basic SVGs, we suggest -you display them like you would any other image (don't forget the `alt` attribute). - -``` -icon name -``` - -#### Using Open Iconic's SVG Sprite - -Open Iconic also comes in a SVG sprite which allows you to display all the icons in the set with a single request. It's -like an icon font, without being a hack. - -Adding an icon from an SVG sprite is a little different than what you're used to, but it's still a piece of cake. *Tip: -To make your icons easily style able, we suggest adding a general class to the* `` *tag and a unique class name for -each different icon in the* `` *tag.* - -``` - - - -``` - -Sizing icons only needs basic CSS. All the icons are in a square format, so just set the `` tag with equal width -and height dimensions. - -``` -.icon { - width: 16px; - height: 16px; -} -``` - -Coloring icons is even easier. All you need to do is set the `fill` rule on the `` tag. - -``` -.icon-account-login { - fill: #f00; -} -``` - -To learn more about SVG Sprites, read [Chris Coyier's guide](http://css-tricks.com/svg-sprites-use-better-icon-fonts/). - -#### Using Open Iconic's Icon Font... - -##### …with Bootstrap - -You can find our Bootstrap stylesheets in `font/css/open-iconic-bootstrap.{css, less, scss, styl}` - -``` - -``` - -``` - -``` - -##### …with Foundation - -You can find our Foundation stylesheets in `font/css/open-iconic-foundation.{css, less, scss, styl}` - -``` - -``` - -``` - -``` - -##### …on its own - -You can find our default stylesheets in `font/css/open-iconic.{css, less, scss, styl}` - -``` - -``` - -``` - -``` - -## License - -### Icons - -All code (including SVG markup) is under the [MIT License](http://opensource.org/licenses/MIT). - -### Fonts - -All fonts are under the [SIL Licensed](http://scripts.sil.org/cms/scripts/page.php?item_id=OFL_web). \ No newline at end of file diff --git a/src/Client/wwwroot/css/open-iconic/font/css/open-iconic-bootstrap.min.css b/src/Client/wwwroot/css/open-iconic/font/css/open-iconic-bootstrap.min.css deleted file mode 100644 index 4664f2e..0000000 --- a/src/Client/wwwroot/css/open-iconic/font/css/open-iconic-bootstrap.min.css +++ /dev/null @@ -1 +0,0 @@ -@font-face{font-family:Icons;src:url(../fonts/open-iconic.eot);src:url(../fonts/open-iconic.eot?#iconic-sm) format('embedded-opentype'),url(../fonts/open-iconic.woff) format('woff'),url(../fonts/open-iconic.ttf) format('truetype'),url(../fonts/open-iconic.otf) format('opentype'),url(../fonts/open-iconic.svg#iconic-sm) format('svg');font-weight:400;font-style:normal}.oi{position:relative;top:1px;display:inline-block;speak:none;font-family:Icons;font-style:normal;font-weight:400;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.oi:empty:before{width:1em;text-align:center;box-sizing:content-box}.oi.oi-align-center:before{text-align:center}.oi.oi-align-left:before{text-align:left}.oi.oi-align-right:before{text-align:right}.oi.oi-flip-horizontal:before{-webkit-transform:scale(-1,1);-ms-transform:scale(-1,1);transform:scale(-1,1)}.oi.oi-flip-vertical:before{-webkit-transform:scale(1,-1);-ms-transform:scale(-1,1);transform:scale(1,-1)}.oi.oi-flip-horizontal-vertical:before{-webkit-transform:scale(-1,-1);-ms-transform:scale(-1,1);transform:scale(-1,-1)}.oi-account-login:before{content:'\e000'}.oi-account-logout:before{content:'\e001'}.oi-action-redo:before{content:'\e002'}.oi-action-undo:before{content:'\e003'}.oi-align-center:before{content:'\e004'}.oi-align-left:before{content:'\e005'}.oi-align-right:before{content:'\e006'}.oi-aperture:before{content:'\e007'}.oi-arrow-bottom:before{content:'\e008'}.oi-arrow-circle-bottom:before{content:'\e009'}.oi-arrow-circle-left:before{content:'\e00a'}.oi-arrow-circle-right:before{content:'\e00b'}.oi-arrow-circle-top:before{content:'\e00c'}.oi-arrow-left:before{content:'\e00d'}.oi-arrow-right:before{content:'\e00e'}.oi-arrow-thick-bottom:before{content:'\e00f'}.oi-arrow-thick-left:before{content:'\e010'}.oi-arrow-thick-right:before{content:'\e011'}.oi-arrow-thick-top:before{content:'\e012'}.oi-arrow-top:before{content:'\e013'}.oi-audio-spectrum:before{content:'\e014'}.oi-audio:before{content:'\e015'}.oi-badge:before{content:'\e016'}.oi-ban:before{content:'\e017'}.oi-bar-chart:before{content:'\e018'}.oi-basket:before{content:'\e019'}.oi-battery-empty:before{content:'\e01a'}.oi-battery-full:before{content:'\e01b'}.oi-beaker:before{content:'\e01c'}.oi-bell:before{content:'\e01d'}.oi-bluetooth:before{content:'\e01e'}.oi-bold:before{content:'\e01f'}.oi-bolt:before{content:'\e020'}.oi-book:before{content:'\e021'}.oi-bookmark:before{content:'\e022'}.oi-box:before{content:'\e023'}.oi-briefcase:before{content:'\e024'}.oi-british-pound:before{content:'\e025'}.oi-browser:before{content:'\e026'}.oi-brush:before{content:'\e027'}.oi-bug:before{content:'\e028'}.oi-bullhorn:before{content:'\e029'}.oi-calculator:before{content:'\e02a'}.oi-calendar:before{content:'\e02b'}.oi-camera-slr:before{content:'\e02c'}.oi-caret-bottom:before{content:'\e02d'}.oi-caret-left:before{content:'\e02e'}.oi-caret-right:before{content:'\e02f'}.oi-caret-top:before{content:'\e030'}.oi-cart:before{content:'\e031'}.oi-chat:before{content:'\e032'}.oi-check:before{content:'\e033'}.oi-chevron-bottom:before{content:'\e034'}.oi-chevron-left:before{content:'\e035'}.oi-chevron-right:before{content:'\e036'}.oi-chevron-top:before{content:'\e037'}.oi-circle-check:before{content:'\e038'}.oi-circle-x:before{content:'\e039'}.oi-clipboard:before{content:'\e03a'}.oi-clock:before{content:'\e03b'}.oi-cloud-download:before{content:'\e03c'}.oi-cloud-upload:before{content:'\e03d'}.oi-cloud:before{content:'\e03e'}.oi-cloudy:before{content:'\e03f'}.oi-code:before{content:'\e040'}.oi-cog:before{content:'\e041'}.oi-collapse-down:before{content:'\e042'}.oi-collapse-left:before{content:'\e043'}.oi-collapse-right:before{content:'\e044'}.oi-collapse-up:before{content:'\e045'}.oi-command:before{content:'\e046'}.oi-comment-square:before{content:'\e047'}.oi-compass:before{content:'\e048'}.oi-contrast:before{content:'\e049'}.oi-copywriting:before{content:'\e04a'}.oi-credit-card:before{content:'\e04b'}.oi-crop:before{content:'\e04c'}.oi-dashboard:before{content:'\e04d'}.oi-data-transfer-download:before{content:'\e04e'}.oi-data-transfer-upload:before{content:'\e04f'}.oi-delete:before{content:'\e050'}.oi-dial:before{content:'\e051'}.oi-document:before{content:'\e052'}.oi-dollar:before{content:'\e053'}.oi-double-quote-sans-left:before{content:'\e054'}.oi-double-quote-sans-right:before{content:'\e055'}.oi-double-quote-serif-left:before{content:'\e056'}.oi-double-quote-serif-right:before{content:'\e057'}.oi-droplet:before{content:'\e058'}.oi-eject:before{content:'\e059'}.oi-elevator:before{content:'\e05a'}.oi-ellipses:before{content:'\e05b'}.oi-envelope-closed:before{content:'\e05c'}.oi-envelope-open:before{content:'\e05d'}.oi-euro:before{content:'\e05e'}.oi-excerpt:before{content:'\e05f'}.oi-expand-down:before{content:'\e060'}.oi-expand-left:before{content:'\e061'}.oi-expand-right:before{content:'\e062'}.oi-expand-up:before{content:'\e063'}.oi-external-link:before{content:'\e064'}.oi-eye:before{content:'\e065'}.oi-eyedropper:before{content:'\e066'}.oi-file:before{content:'\e067'}.oi-fire:before{content:'\e068'}.oi-flag:before{content:'\e069'}.oi-flash:before{content:'\e06a'}.oi-folder:before{content:'\e06b'}.oi-fork:before{content:'\e06c'}.oi-fullscreen-enter:before{content:'\e06d'}.oi-fullscreen-exit:before{content:'\e06e'}.oi-globe:before{content:'\e06f'}.oi-graph:before{content:'\e070'}.oi-grid-four-up:before{content:'\e071'}.oi-grid-three-up:before{content:'\e072'}.oi-grid-two-up:before{content:'\e073'}.oi-hard-drive:before{content:'\e074'}.oi-header:before{content:'\e075'}.oi-headphones:before{content:'\e076'}.oi-heart:before{content:'\e077'}.oi-home:before{content:'\e078'}.oi-image:before{content:'\e079'}.oi-inbox:before{content:'\e07a'}.oi-infinity:before{content:'\e07b'}.oi-info:before{content:'\e07c'}.oi-italic:before{content:'\e07d'}.oi-justify-center:before{content:'\e07e'}.oi-justify-left:before{content:'\e07f'}.oi-justify-right:before{content:'\e080'}.oi-key:before{content:'\e081'}.oi-laptop:before{content:'\e082'}.oi-layers:before{content:'\e083'}.oi-lightbulb:before{content:'\e084'}.oi-link-broken:before{content:'\e085'}.oi-link-intact:before{content:'\e086'}.oi-list-rich:before{content:'\e087'}.oi-list:before{content:'\e088'}.oi-location:before{content:'\e089'}.oi-lock-locked:before{content:'\e08a'}.oi-lock-unlocked:before{content:'\e08b'}.oi-loop-circular:before{content:'\e08c'}.oi-loop-square:before{content:'\e08d'}.oi-loop:before{content:'\e08e'}.oi-magnifying-glass:before{content:'\e08f'}.oi-map-marker:before{content:'\e090'}.oi-map:before{content:'\e091'}.oi-media-pause:before{content:'\e092'}.oi-media-play:before{content:'\e093'}.oi-media-record:before{content:'\e094'}.oi-media-skip-backward:before{content:'\e095'}.oi-media-skip-forward:before{content:'\e096'}.oi-media-step-backward:before{content:'\e097'}.oi-media-step-forward:before{content:'\e098'}.oi-media-stop:before{content:'\e099'}.oi-medical-cross:before{content:'\e09a'}.oi-menu:before{content:'\e09b'}.oi-microphone:before{content:'\e09c'}.oi-minus:before{content:'\e09d'}.oi-monitor:before{content:'\e09e'}.oi-moon:before{content:'\e09f'}.oi-move:before{content:'\e0a0'}.oi-musical-note:before{content:'\e0a1'}.oi-paperclip:before{content:'\e0a2'}.oi-pencil:before{content:'\e0a3'}.oi-people:before{content:'\e0a4'}.oi-person:before{content:'\e0a5'}.oi-phone:before{content:'\e0a6'}.oi-pie-chart:before{content:'\e0a7'}.oi-pin:before{content:'\e0a8'}.oi-play-circle:before{content:'\e0a9'}.oi-plus:before{content:'\e0aa'}.oi-power-standby:before{content:'\e0ab'}.oi-print:before{content:'\e0ac'}.oi-project:before{content:'\e0ad'}.oi-pulse:before{content:'\e0ae'}.oi-puzzle-piece:before{content:'\e0af'}.oi-question-mark:before{content:'\e0b0'}.oi-rain:before{content:'\e0b1'}.oi-random:before{content:'\e0b2'}.oi-reload:before{content:'\e0b3'}.oi-resize-both:before{content:'\e0b4'}.oi-resize-height:before{content:'\e0b5'}.oi-resize-width:before{content:'\e0b6'}.oi-rss-alt:before{content:'\e0b7'}.oi-rss:before{content:'\e0b8'}.oi-script:before{content:'\e0b9'}.oi-share-boxed:before{content:'\e0ba'}.oi-share:before{content:'\e0bb'}.oi-shield:before{content:'\e0bc'}.oi-signal:before{content:'\e0bd'}.oi-signpost:before{content:'\e0be'}.oi-sort-ascending:before{content:'\e0bf'}.oi-sort-descending:before{content:'\e0c0'}.oi-spreadsheet:before{content:'\e0c1'}.oi-star:before{content:'\e0c2'}.oi-sun:before{content:'\e0c3'}.oi-tablet:before{content:'\e0c4'}.oi-tag:before{content:'\e0c5'}.oi-tags:before{content:'\e0c6'}.oi-target:before{content:'\e0c7'}.oi-task:before{content:'\e0c8'}.oi-terminal:before{content:'\e0c9'}.oi-text:before{content:'\e0ca'}.oi-thumb-down:before{content:'\e0cb'}.oi-thumb-up:before{content:'\e0cc'}.oi-timer:before{content:'\e0cd'}.oi-transfer:before{content:'\e0ce'}.oi-trash:before{content:'\e0cf'}.oi-underline:before{content:'\e0d0'}.oi-vertical-align-bottom:before{content:'\e0d1'}.oi-vertical-align-center:before{content:'\e0d2'}.oi-vertical-align-top:before{content:'\e0d3'}.oi-video:before{content:'\e0d4'}.oi-volume-high:before{content:'\e0d5'}.oi-volume-low:before{content:'\e0d6'}.oi-volume-off:before{content:'\e0d7'}.oi-warning:before{content:'\e0d8'}.oi-wifi:before{content:'\e0d9'}.oi-wrench:before{content:'\e0da'}.oi-x:before{content:'\e0db'}.oi-yen:before{content:'\e0dc'}.oi-zoom-in:before{content:'\e0dd'}.oi-zoom-out:before{content:'\e0de'} \ No newline at end of file diff --git a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.eot b/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.eot deleted file mode 100644 index f98177d..0000000 Binary files a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.eot and /dev/null differ diff --git a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.otf b/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.otf deleted file mode 100644 index f6bd684..0000000 Binary files a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.otf and /dev/null differ diff --git a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.svg b/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.svg deleted file mode 100644 index 6c29493..0000000 --- a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.svg +++ /dev/null @@ -1,544 +0,0 @@ - - - - - - Created by FontForge 20120731 at Tue Jul 1 20:39:22 2014 - By P.J. Onori - Created by P.J. Onori with FontForge 2.0 (http://fontforge.sf.net) - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.ttf b/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.ttf deleted file mode 100644 index fab6048..0000000 Binary files a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.ttf and /dev/null differ diff --git a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.woff b/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.woff deleted file mode 100644 index f930998..0000000 Binary files a/src/Client/wwwroot/css/open-iconic/font/fonts/open-iconic.woff and /dev/null differ diff --git a/src/Client/wwwroot/favicon.ico b/src/Client/wwwroot/favicon.ico deleted file mode 100644 index 63e859b..0000000 Binary files a/src/Client/wwwroot/favicon.ico and /dev/null differ diff --git a/src/Client/wwwroot/icon-192.png b/src/Client/wwwroot/icon-192.png deleted file mode 100644 index 166f56d..0000000 Binary files a/src/Client/wwwroot/icon-192.png and /dev/null differ diff --git a/src/Client/wwwroot/index.html b/src/Client/wwwroot/index.html deleted file mode 100644 index ce03f68..0000000 --- a/src/Client/wwwroot/index.html +++ /dev/null @@ -1,26 +0,0 @@ - - - - - - - NSeguin.Dev.Web - - - - - - - -
Loading...
- -
- An unhandled error has occurred. - Reload - 🗙 -
- - - - - \ No newline at end of file diff --git a/src/NSeguin.Dev.Web.Client/NSeguin.Dev.Web.Client.csproj b/src/NSeguin.Dev.Web.Client/NSeguin.Dev.Web.Client.csproj new file mode 100644 index 0000000..92e4044 --- /dev/null +++ b/src/NSeguin.Dev.Web.Client/NSeguin.Dev.Web.Client.csproj @@ -0,0 +1,23 @@ + + + + net8.0 + enable + enable + true + Default + + + + + + + + + + <_ContentIncludedByDefault Remove="wwwroot\css\app.css" /> + <_ContentIncludedByDefault Remove="wwwroot\css\bootstrap\bootstrap.min.css" /> + <_ContentIncludedByDefault Remove="wwwroot\css\bootstrap\bootstrap.min.css.map" /> + + + \ No newline at end of file diff --git a/src/Client/Pages/Counter.razor b/src/NSeguin.Dev.Web.Client/Pages/Counter.razor similarity index 81% rename from src/Client/Pages/Counter.razor rename to src/NSeguin.Dev.Web.Client/Pages/Counter.razor index 005d33c..e7d4f94 100644 --- a/src/Client/Pages/Counter.razor +++ b/src/NSeguin.Dev.Web.Client/Pages/Counter.razor @@ -1,4 +1,5 @@ @page "/counter" +@rendermode InteractiveAuto Counter @@ -9,11 +10,10 @@ @code { - private int currentCount; + private int currentCount = 0; private void IncrementCount() { currentCount++; } - -} \ No newline at end of file +} diff --git a/src/NSeguin.Dev.Web.Client/Program.cs b/src/NSeguin.Dev.Web.Client/Program.cs new file mode 100644 index 0000000..89f3f30 --- /dev/null +++ b/src/NSeguin.Dev.Web.Client/Program.cs @@ -0,0 +1,4 @@ +using Microsoft.AspNetCore.Components.WebAssembly.Hosting; + +var builder = WebAssemblyHostBuilder.CreateDefault(args); +await builder.Build().RunAsync(); diff --git a/src/NSeguin.Dev.Web.Client/UserInfo.cs b/src/NSeguin.Dev.Web.Client/UserInfo.cs new file mode 100644 index 0000000..7a14212 --- /dev/null +++ b/src/NSeguin.Dev.Web.Client/UserInfo.cs @@ -0,0 +1,9 @@ +namespace NSeguin.Dev.Web.Client; + +// Add properties to this class and update the server and client AuthenticationStateProviders +// to expose more information about the authenticated user to the client. +public class UserInfo +{ + public required string UserId { get; set; } + public required string Email { get; set; } +} diff --git a/src/NSeguin.Dev.Web.Client/_Imports.razor b/src/NSeguin.Dev.Web.Client/_Imports.razor new file mode 100644 index 0000000..3c3b9ae --- /dev/null +++ b/src/NSeguin.Dev.Web.Client/_Imports.razor @@ -0,0 +1,9 @@ +@using System.Net.Http +@using System.Net.Http.Json +@using Microsoft.AspNetCore.Components.Forms +@using Microsoft.AspNetCore.Components.Routing +@using Microsoft.AspNetCore.Components.Web +@using static Microsoft.AspNetCore.Components.Web.RenderMode +@using Microsoft.AspNetCore.Components.Web.Virtualization +@using Microsoft.JSInterop +@using NSeguin.Dev.Web.Client diff --git a/src/Server/appsettings.Development.json b/src/NSeguin.Dev.Web.Client/wwwroot/appsettings.Development.json similarity index 71% rename from src/Server/appsettings.Development.json rename to src/NSeguin.Dev.Web.Client/wwwroot/appsettings.Development.json index a783479..0c208ae 100644 --- a/src/Server/appsettings.Development.json +++ b/src/NSeguin.Dev.Web.Client/wwwroot/appsettings.Development.json @@ -1,8 +1,8 @@ { "Logging": { "LogLevel": { - "Default": "Debug", + "Default": "Information", "Microsoft.AspNetCore": "Warning" } } -} \ No newline at end of file +} diff --git a/src/NSeguin.Dev.Web.Client/wwwroot/appsettings.json b/src/NSeguin.Dev.Web.Client/wwwroot/appsettings.json new file mode 100644 index 0000000..3c6896e --- /dev/null +++ b/src/NSeguin.Dev.Web.Client/wwwroot/appsettings.json @@ -0,0 +1,13 @@ +{ + "AzureAdB2C": { + "Authority": "https://nseguindev.b2clogin.com/nseguindev.onmicrosoft.com/B2C_1_signin", + "ClientId": "eebf1b00-74e5-4d9e-a05c-0ceab9b382e9", + "ValidateAuthority": false + }, + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} \ No newline at end of file diff --git a/src/NSeguin.Dev.Web.sln b/src/NSeguin.Dev.Web.sln new file mode 100644 index 0000000..651a2fa --- /dev/null +++ b/src/NSeguin.Dev.Web.sln @@ -0,0 +1,50 @@ +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 16 +VisualStudioVersion = 16.0.0.0 +MinimumVisualStudioVersion = 16.0.0.0 +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "NSeguin.Dev.Web", "NSeguin.Dev.Web\NSeguin.Dev.Web.csproj", "{D9C8497C-2D8B-4860-8D20-082B80C3D684}" +EndProject +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "NSeguin.Dev.Web.Client", "NSeguin.Dev.Web.Client\NSeguin.Dev.Web.Client.csproj", "{F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}" +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 + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Debug|Any CPU.Build.0 = Debug|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Debug|x64.ActiveCfg = Debug|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Debug|x64.Build.0 = Debug|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Debug|x86.ActiveCfg = Debug|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Debug|x86.Build.0 = Debug|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Release|Any CPU.ActiveCfg = Release|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Release|Any CPU.Build.0 = Release|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Release|x64.ActiveCfg = Release|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Release|x64.Build.0 = Release|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Release|x86.ActiveCfg = Release|Any CPU + {F48BF4EE-9549-49B2-928C-7EC3AF0CD0AB}.Release|x86.Build.0 = Release|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Debug|Any CPU.Build.0 = Debug|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Debug|x64.ActiveCfg = Debug|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Debug|x64.Build.0 = Debug|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Debug|x86.ActiveCfg = Debug|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Debug|x86.Build.0 = Debug|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Release|Any CPU.ActiveCfg = Release|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Release|Any CPU.Build.0 = Release|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Release|x64.ActiveCfg = Release|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Release|x64.Build.0 = Release|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Release|x86.ActiveCfg = Release|Any CPU + {D9C8497C-2D8B-4860-8D20-082B80C3D684}.Release|x86.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {4599CBB5-6EB8-4DA8-9825-78545A7C0D4B} + EndGlobalSection +EndGlobal diff --git a/src/NSeguin.Dev.Web/Components/App.razor b/src/NSeguin.Dev.Web/Components/App.razor new file mode 100644 index 0000000..4acbb25 --- /dev/null +++ b/src/NSeguin.Dev.Web/Components/App.razor @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/NSeguin.Dev.Web/Components/Layout/MainLayout.razor b/src/NSeguin.Dev.Web/Components/Layout/MainLayout.razor new file mode 100644 index 0000000..6a9042a --- /dev/null +++ b/src/NSeguin.Dev.Web/Components/Layout/MainLayout.razor @@ -0,0 +1,24 @@ +@inherits LayoutComponentBase + +
+ + +
+
+ + About +
+ +
+ @Body +
+
+
+ +
+ An unhandled error has occurred. + Reload + 🗙 +
\ No newline at end of file diff --git a/src/Client/Shared/MainLayout.razor.css b/src/NSeguin.Dev.Web/Components/Layout/MainLayout.razor.css similarity index 58% rename from src/Client/Shared/MainLayout.razor.css rename to src/NSeguin.Dev.Web/Components/Layout/MainLayout.razor.css index 32ad916..038baf1 100644 --- a/src/Client/Shared/MainLayout.razor.css +++ b/src/NSeguin.Dev.Web/Components/Layout/MainLayout.razor.css @@ -21,27 +21,23 @@ main { 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:not(.auth) { - display: none; + .top-row ::deep a, .top-row ::deep .btn-link { + white-space: nowrap; + margin-left: 1.5rem; + text-decoration: none; } - .top-row.auth { + .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; } @@ -78,4 +74,23 @@ main { padding-left: 2rem !important; padding-right: 1.5rem !important; } -} \ No newline at end of file +} + +#blazor-error-ui { + background: lightyellow; + bottom: 0; + box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2); + display: none; + left: 0; + padding: 0.6rem 1.25rem 0.7rem 1.25rem; + position: fixed; + width: 100%; + z-index: 1000; +} + + #blazor-error-ui .dismiss { + cursor: pointer; + position: absolute; + right: 0.75rem; + top: 0.5rem; + } diff --git a/src/NSeguin.Dev.Web/Components/Layout/NavMenu.razor b/src/NSeguin.Dev.Web/Components/Layout/NavMenu.razor new file mode 100644 index 0000000..4d47878 --- /dev/null +++ b/src/NSeguin.Dev.Web/Components/Layout/NavMenu.razor @@ -0,0 +1,30 @@ + + + + + + diff --git a/src/NSeguin.Dev.Web/Components/Layout/NavMenu.razor.css b/src/NSeguin.Dev.Web/Components/Layout/NavMenu.razor.css new file mode 100644 index 0000000..4e15395 --- /dev/null +++ b/src/NSeguin.Dev.Web/Components/Layout/NavMenu.razor.css @@ -0,0 +1,105 @@ +.navbar-toggler { + appearance: none; + cursor: pointer; + width: 3.5rem; + height: 2.5rem; + color: white; + position: absolute; + top: 0.5rem; + right: 1rem; + border: 1px solid rgba(255, 255, 255, 0.1); + background: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e") no-repeat center/1.75rem rgba(255, 255, 255, 0.1); +} + +.navbar-toggler:checked { + background-color: rgba(255, 255, 255, 0.5); +} + +.top-row { + 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 .nav-link { + color: #d7d7d7; + background: none; + border: none; + border-radius: 4px; + height: 3rem; + display: flex; + align-items: center; + line-height: 3rem; + width: 100%; + } + +.nav-item ::deep a.active { + background-color: rgba(255,255,255,0.37); + color: white; +} + +.nav-item ::deep .nav-link:hover { + background-color: rgba(255,255,255,0.1); + color: white; +} + +.nav-scrollable { + display: none; +} + +.navbar-toggler:checked ~ .nav-scrollable { + display: block; +} + +@media (min-width: 641px) { + .navbar-toggler { + display: none; + } + + .nav-scrollable { + /* Never collapse the sidebar for wide screens */ + display: block; + + /* Allow sidebar to scroll for tall menus */ + height: calc(100vh - 3.5rem); + overflow-y: auto; + } +} diff --git a/src/NSeguin.Dev.Web/Components/Pages/Authentication.razor b/src/NSeguin.Dev.Web/Components/Pages/Authentication.razor new file mode 100644 index 0000000..9ee0b39 --- /dev/null +++ b/src/NSeguin.Dev.Web/Components/Pages/Authentication.razor @@ -0,0 +1,42 @@ +@page "/authentication/{action}" +@using System.Security.Claims + +@code{ + private const string Login = "login"; + private const string Logout = "logout"; + public ClaimsPrincipal? User { get; private set; } + + [CascadingParameter] + public required Task AuthenticationState { get; set; } + + [Parameter] + public string? Action { get; set; } + + [Inject] + public required NavigationManager NavigationManager { get; set; } + + private void SignOut() + { + NavigationManager.NavigateTo("MicrosoftIdentity/Account/SignOut", true); + } + + private void SignIn() + { + NavigationManager.NavigateTo("MicrosoftIdentity/Account/SignIn", true); + } + + protected override async Task OnInitializedAsync() + { + var authenticationState = await AuthenticationState; + User = authenticationState.User; + if (Action == Logout) + { + SignOut(); + } + else if (Action == Login) + { + SignIn(); + } + } + +} \ No newline at end of file diff --git a/src/NSeguin.Dev.Web/Components/Pages/Error.razor b/src/NSeguin.Dev.Web/Components/Pages/Error.razor new file mode 100644 index 0000000..576cc2d --- /dev/null +++ b/src/NSeguin.Dev.Web/Components/Pages/Error.razor @@ -0,0 +1,36 @@ +@page "/Error" +@using System.Diagnostics + +Error + +

Error.

+

An error occurred while processing your request.

+ +@if (ShowRequestId) +{ +

+ Request ID: @RequestId +

+} + +

Development Mode

+

+ Swapping to Development environment will display more detailed information about the error that occurred. +

+

+ The Development environment shouldn't be enabled for deployed applications. + It can result in displaying sensitive information from exceptions to end users. + For local debugging, enable the Development environment by setting the ASPNETCORE_ENVIRONMENT environment variable to Development + and restarting the app. +

+ +@code{ + [CascadingParameter] + private HttpContext? HttpContext { get; set; } + + private string? RequestId { get; set; } + private bool ShowRequestId => !string.IsNullOrEmpty(RequestId); + + protected override void OnInitialized() => + RequestId = Activity.Current?.Id ?? HttpContext?.TraceIdentifier; +} diff --git a/src/NSeguin.Dev.Web/Components/Pages/Home.razor b/src/NSeguin.Dev.Web/Components/Pages/Home.razor new file mode 100644 index 0000000..9001e0b --- /dev/null +++ b/src/NSeguin.Dev.Web/Components/Pages/Home.razor @@ -0,0 +1,7 @@ +@page "/" + +Home + +

Hello, world!

+ +Welcome to your new app. diff --git a/src/NSeguin.Dev.Web/Components/Pages/Weather.razor b/src/NSeguin.Dev.Web/Components/Pages/Weather.razor new file mode 100644 index 0000000..43a1ecb --- /dev/null +++ b/src/NSeguin.Dev.Web/Components/Pages/Weather.razor @@ -0,0 +1,64 @@ +@page "/weather" +@attribute [StreamRendering] + +Weather + +

Weather

+ +

This component demonstrates showing data.

+ +@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() + { + // Simulate asynchronous loading to demonstrate streaming rendering + await Task.Delay(500); + + var startDate = DateOnly.FromDateTime(DateTime.Now); + var summaries = new[] { "Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering", "Scorching" }; + forecasts = Enumerable.Range(1, 5).Select(index => new WeatherForecast + { + Date = startDate.AddDays(index), + TemperatureC = Random.Shared.Next(-20, 55), + Summary = summaries[Random.Shared.Next(summaries.Length)] + }).ToArray(); + } + + private class WeatherForecast + { + public DateOnly Date { get; set; } + public int TemperatureC { get; set; } + public string? Summary { get; set; } + public int TemperatureF => 32 + (int)(TemperatureC / 0.5556); + } +} diff --git a/src/Client/App.razor b/src/NSeguin.Dev.Web/Components/Routes.razor similarity index 81% rename from src/Client/App.razor rename to src/NSeguin.Dev.Web/Components/Routes.razor index 6beefc2..889bc65 100644 --- a/src/Client/App.razor +++ b/src/NSeguin.Dev.Web/Components/Routes.razor @@ -1,5 +1,6 @@ - - +@using NSeguin.Dev.Web.Components.Layout + + @@ -13,7 +14,6 @@ } - Not found diff --git a/src/Client/Shared/LoginDisplay.razor b/src/NSeguin.Dev.Web/Components/Shared/LoginDisplay.razor similarity index 75% rename from src/Client/Shared/LoginDisplay.razor rename to src/NSeguin.Dev.Web/Components/Shared/LoginDisplay.razor index 3b8cef4..060e096 100644 --- a/src/Client/Shared/LoginDisplay.razor +++ b/src/NSeguin.Dev.Web/Components/Shared/LoginDisplay.razor @@ -1,4 +1,5 @@ -@using Microsoft.AspNetCore.Components.WebAssembly.Authentication +@using Microsoft.AspNetCore.Components.Authorization + @inject NavigationManager Navigation @@ -14,6 +15,6 @@ @code{ private void BeginLogout(MouseEventArgs args) { - Navigation.NavigateToLogout("authentication/logout"); + Navigation.NavigateTo("authentication/logout"); } } \ No newline at end of file diff --git a/src/Client/Shared/RedirectToLogin.razor b/src/NSeguin.Dev.Web/Components/Shared/RedirectToLogin.razor similarity index 82% rename from src/Client/Shared/RedirectToLogin.razor rename to src/NSeguin.Dev.Web/Components/Shared/RedirectToLogin.razor index 426fc6d..2db61cf 100644 --- a/src/Client/Shared/RedirectToLogin.razor +++ b/src/NSeguin.Dev.Web/Components/Shared/RedirectToLogin.razor @@ -1,11 +1,8 @@ @inject NavigationManager Navigation @code { - protected override void OnInitialized() { - Console.WriteLine(Navigation.Uri); Navigation.NavigateTo($"authentication/login?returnUrl={Uri.EscapeDataString(Navigation.Uri)}"); } - -} \ No newline at end of file +} diff --git a/src/Client/_Imports.razor b/src/NSeguin.Dev.Web/Components/_Imports.razor similarity index 69% rename from src/Client/_Imports.razor rename to src/NSeguin.Dev.Web/Components/_Imports.razor index e2a4a6b..f0eb67c 100644 --- a/src/Client/_Imports.razor +++ b/src/NSeguin.Dev.Web/Components/_Imports.razor @@ -4,8 +4,10 @@ @using Microsoft.AspNetCore.Components.Forms @using Microsoft.AspNetCore.Components.Routing @using Microsoft.AspNetCore.Components.Web +@using static Microsoft.AspNetCore.Components.Web.RenderMode @using Microsoft.AspNetCore.Components.Web.Virtualization -@using Microsoft.AspNetCore.Components.WebAssembly.Http @using Microsoft.JSInterop +@using NSeguin.Dev.Web @using NSeguin.Dev.Web.Client -@using NSeguin.Dev.Web.Client.Shared \ No newline at end of file +@using NSeguin.Dev.Web.Components +@using NSeguin.Dev.Web.Components.Shared \ No newline at end of file diff --git a/src/NSeguin.Dev.Web/NSeguin.Dev.Web.csproj b/src/NSeguin.Dev.Web/NSeguin.Dev.Web.csproj new file mode 100644 index 0000000..5c6ca66 --- /dev/null +++ b/src/NSeguin.Dev.Web/NSeguin.Dev.Web.csproj @@ -0,0 +1,39 @@ + + + + net8.0 + enable + enable + + + + + + + + + + + + + + + + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web.Client\wwwroot\appsettings.Development.json" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web.Client\wwwroot\appsettings.json" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web.Client\Pages\Counter.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web.Client\_Imports.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\appsettings.Development.json" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\appsettings.json" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Properties\launchSettings.json" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Components\App.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Components\Layout\MainLayout.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Components\Layout\NavMenu.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Components\Pages\Error.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Components\Pages\Home.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Components\Pages\Weather.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Components\Routes.razor" /> + <_ContentIncludedByDefault Remove="NSeguin.Dev.Web\Components\_Imports.razor" /> + + + \ No newline at end of file diff --git a/src/NSeguin.Dev.Web/Program.cs b/src/NSeguin.Dev.Web/Program.cs new file mode 100644 index 0000000..1573ca8 --- /dev/null +++ b/src/NSeguin.Dev.Web/Program.cs @@ -0,0 +1,80 @@ +using Azure.Identity; + +using Microsoft.AspNetCore.Authentication.OpenIdConnect; +using Microsoft.Identity.Web; +using Microsoft.Identity.Web.UI; +using Microsoft.IdentityModel.Protocols.OpenIdConnect; + +using NSeguin.Dev.Web.Client.Pages; +using NSeguin.Dev.Web.Components; + +var builder = WebApplication.CreateBuilder(args); +builder.Services.Configure( + options => + { + // This lambda determines whether user consent for non-essential cookies is needed for a given request. + options.CheckConsentNeeded = context => true; + options.MinimumSameSitePolicy = SameSiteMode.Unspecified; + // Handling SameSite cookie according to https://docs.microsoft.com/en-us/aspnet/core/security/samesite?view=aspnetcore-3.1 + options.HandleSameSiteCookieCompatibility(); + }); + +builder.Services.AddControllersWithViews().AddMicrosoftIdentityUI(); +builder.Services.AddRazorPages(); +builder.Services.AddRazorComponents() + .AddInteractiveServerComponents() + .AddInteractiveWebAssemblyComponents(); + +if (builder.Environment.IsStaging() || builder.Environment.IsProduction()) +{ + builder.Configuration.AddAzureAppConfiguration( + a => + { + a.Connect( + Environment.GetEnvironmentVariable("AZURE_APP_CONFIGURATION_CONNECTION_STRING")) + .ConfigureKeyVault(kv => kv.SetCredential(new DefaultAzureCredential())); + }); +} + +builder.Services.AddAuthentication(OpenIdConnectDefaults.AuthenticationScheme) + .AddMicrosoftIdentityWebApp( + builder.Configuration.GetSection("AzureAdB2C"), + subscribeToOpenIdConnectMiddlewareDiagnosticsEvents: true); + +builder.Services.Configure( + OpenIdConnectDefaults.AuthenticationScheme, + options => + { + options.ResponseType = OpenIdConnectResponseType.Code; + options.Scope.Add(options.ClientId); + }); + +var app = builder.Build(); + +// Configure the HTTP request pipeline. +if (app.Environment.IsDevelopment()) +{ + app.UseWebAssemblyDebugging(); +} +else +{ + app.UseExceptionHandler("/Error", createScopeForErrors: true); + // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts. + app.UseHsts(); +} + +app.UseHttpsRedirection(); +app.UseBlazorFrameworkFiles(); +app.UseStaticFiles(); +app.UseRouting(); +app.UseAuthentication(); +app.UseAuthorization(); +app.UseAntiforgery(); +app.MapRazorComponents() + .AddAdditionalAssemblies(typeof(Counter).Assembly) + .AddInteractiveServerRenderMode() + .AddInteractiveWebAssemblyRenderMode(); + +app.MapControllers(); +app.MapFallbackToFile("index.html"); +app.Run(); diff --git a/src/NSeguin.Dev.Web/Properties/launchSettings.json b/src/NSeguin.Dev.Web/Properties/launchSettings.json new file mode 100644 index 0000000..4d89521 --- /dev/null +++ b/src/NSeguin.Dev.Web/Properties/launchSettings.json @@ -0,0 +1,41 @@ +{ + "$schema": "http://json.schemastore.org/launchsettings.json", + "iisSettings": { + "windowsAuthentication": false, + "anonymousAuthentication": true, + "iisExpress": { + "applicationUrl": "http://localhost:25070", + "sslPort": 44305 + } + }, + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", + "applicationUrl": "http://localhost:5172", + "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:7259;http://localhost:5172", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "IIS Express": { + "commandName": "IISExpress", + "launchBrowser": true, + "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } + } diff --git a/src/NSeguin.Dev.Web/appsettings.Development.json b/src/NSeguin.Dev.Web/appsettings.Development.json new file mode 100644 index 0000000..0c208ae --- /dev/null +++ b/src/NSeguin.Dev.Web/appsettings.Development.json @@ -0,0 +1,8 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} diff --git a/src/NSeguin.Dev.Web/appsettings.json b/src/NSeguin.Dev.Web/appsettings.json new file mode 100644 index 0000000..0741ee2 --- /dev/null +++ b/src/NSeguin.Dev.Web/appsettings.json @@ -0,0 +1,23 @@ +{ + "AzureAppConfiguration": { + "Endpoint": "https://nseguin-dev-app-config.azconfig.io" + }, + "AzureAdB2C": { + "Instance": "https://nseguindev.b2clogin.com/", + "ClientId": "db78f018-cfcd-4bb4-b81e-c139a1f76af6", + "Domain": "nseguindev.onmicrosoft.com", + "Scopes": "https://nseguindev.onmicrosoft.com/db78f018-cfcd-4bb4-b81e-c139a1f76af6/access_as_user", + "SignUpSignInPolicyId": "B2C_1_signin", + "SignedOutCallbackPath": "/signout-oidc" + }, + "AzureKeyVault": { + "VaultUri": "https://nseguin-dev-key-vault.vault.azure.net/" + }, + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + }, + "AllowedHosts": "*" +} \ No newline at end of file diff --git a/src/Client/wwwroot/css/app.css b/src/NSeguin.Dev.Web/wwwroot/app.css similarity index 79% rename from src/Client/wwwroot/css/app.css rename to src/NSeguin.Dev.Web/wwwroot/app.css index a6ecb1a..2bd9b78 100644 --- a/src/Client/wwwroot/css/app.css +++ b/src/NSeguin.Dev.Web/wwwroot/app.css @@ -1,15 +1,9 @@ -@import url('open-iconic/font/css/open-iconic-bootstrap.min.css'); - html, body { font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; } -h1:focus { - outline: none; -} - a, .btn-link { - color: #0071c1; + color: #006bb7; } .btn-primary { @@ -18,39 +12,28 @@ a, .btn-link { 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; } +h1:focus { + outline: none; +} + .valid.modified:not([type=checkbox]) { outline: 1px solid #26b050; } .invalid { - outline: 1px solid red; + outline: 1px solid #e50000; } .validation-message { - color: red; -} - -#blazor-error-ui { - background: lightyellow; - bottom: 0; - box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2); - display: none; - left: 0; - padding: 0.6rem 1.25rem 0.7rem 1.25rem; - position: fixed; - width: 100%; - z-index: 1000; -} - -#blazor-error-ui .dismiss { - cursor: pointer; - position: absolute; - right: 0.75rem; - top: 0.5rem; + color: #e50000; } .blazor-error-boundary { @@ -59,6 +42,10 @@ a, .btn-link { color: white; } -.blazor-error-boundary::after { - content: "An error has occurred." -} \ No newline at end of file + .blazor-error-boundary::after { + content: "An error has occurred." + } + +.darker-border-checkbox.form-check-input { + border-color: #929292; +} diff --git a/src/Client/wwwroot/css/bootstrap/bootstrap.min.css b/src/NSeguin.Dev.Web/wwwroot/bootstrap/bootstrap.min.css similarity index 100% rename from src/Client/wwwroot/css/bootstrap/bootstrap.min.css rename to src/NSeguin.Dev.Web/wwwroot/bootstrap/bootstrap.min.css diff --git a/src/Client/wwwroot/css/bootstrap/bootstrap.min.css.map b/src/NSeguin.Dev.Web/wwwroot/bootstrap/bootstrap.min.css.map similarity index 100% rename from src/Client/wwwroot/css/bootstrap/bootstrap.min.css.map rename to src/NSeguin.Dev.Web/wwwroot/bootstrap/bootstrap.min.css.map diff --git a/src/NSeguin.Dev.Web/wwwroot/favicon.png b/src/NSeguin.Dev.Web/wwwroot/favicon.png new file mode 100644 index 0000000..8422b59 Binary files /dev/null and b/src/NSeguin.Dev.Web/wwwroot/favicon.png differ diff --git a/src/Server/Controllers/WeatherForecastController.cs b/src/Server/Controllers/WeatherForecastController.cs deleted file mode 100644 index 7c514a7..0000000 --- a/src/Server/Controllers/WeatherForecastController.cs +++ /dev/null @@ -1,41 +0,0 @@ -using Microsoft.AspNetCore.Authorization; -using Microsoft.AspNetCore.Mvc; -using Microsoft.Identity.Web.Resource; - -using NSeguin.Dev.Web.Shared; - -namespace NSeguin.Dev.Web.Server.Controllers; - -[Authorize] -[ApiController] -[Route("[controller]")] -[RequiredScope(RequiredScopesConfigurationKey = "AzureAdB2C:Scopes")] -public class WeatherForecastController : ControllerBase -{ - private static readonly string[] Summaries = - { - "Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering", - "Scorching" - }; - - private readonly ILogger _logger; - - public WeatherForecastController(ILogger logger) - { - _logger = logger; - } - - [HttpGet] - public IEnumerable Get() - { - return Enumerable.Range(1, 5) - .Select( - index => new WeatherForecast - { - Date = DateTime.Now.AddDays(index), - TemperatureC = Random.Shared.Next(-20, 55), - Summary = Summaries[Random.Shared.Next(Summaries.Length)] - }) - .ToArray(); - } -} diff --git a/src/Server/NSeguin.Dev.Web.Server.csproj b/src/Server/NSeguin.Dev.Web.Server.csproj deleted file mode 100644 index 242f14d..0000000 --- a/src/Server/NSeguin.Dev.Web.Server.csproj +++ /dev/null @@ -1,25 +0,0 @@ - - - - net8.0 - false - NSeguin.Dev.Web.Server-b424bded-ba79-482e-9db0-4095696fe958 - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/src/Server/Pages/Error.cshtml b/src/Server/Pages/Error.cshtml deleted file mode 100644 index 90b7aca..0000000 --- a/src/Server/Pages/Error.cshtml +++ /dev/null @@ -1,42 +0,0 @@ -@page -@model NSeguin.Dev.Web.Server.Pages.ErrorModel - - - - - - - - Error - - - - - -
-
-

Error.

-

An error occurred while processing your request.

- - @if (Model.ShowRequestId) - { -

- Request ID: @Model.RequestId -

- } - -

Development Mode

-

- Swapping to the Development environment displays detailed information about the error that occurred. -

-

- The Development environment shouldn't be enabled for deployed applications. - It can result in displaying sensitive information from exceptions to end users. - For local debugging, enable the Development environment by setting the ASPNETCORE_ENVIRONMENT environment variable to Development - and restarting the app. -

-
-
- - - \ No newline at end of file diff --git a/src/Server/Pages/Error.cshtml.cs b/src/Server/Pages/Error.cshtml.cs deleted file mode 100644 index 649d32c..0000000 --- a/src/Server/Pages/Error.cshtml.cs +++ /dev/null @@ -1,26 +0,0 @@ -using System.Diagnostics; - -using Microsoft.AspNetCore.Mvc; -using Microsoft.AspNetCore.Mvc.RazorPages; - -namespace NSeguin.Dev.Web.Server.Pages; - -[ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)] -[IgnoreAntiforgeryToken] -public class ErrorModel : PageModel -{ - private readonly ILogger _logger; - - public ErrorModel(ILogger logger) - { - _logger = logger; - } - - public bool ShowRequestId => !string.IsNullOrEmpty(RequestId); - public string? RequestId { get; set; } - - public void OnGet() - { - RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier; - } -} diff --git a/src/Server/Program.cs b/src/Server/Program.cs deleted file mode 100644 index 6ceb269..0000000 --- a/src/Server/Program.cs +++ /dev/null @@ -1,37 +0,0 @@ -using Microsoft.AspNetCore.Authentication.JwtBearer; -using Microsoft.Identity.Web; - -WebApplicationBuilder builder = WebApplication.CreateBuilder(args); - -// Add services to the container. -builder.Services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme) - .AddMicrosoftIdentityWebApi( - builder.Configuration.GetSection("AzureAdB2C"), - subscribeToJwtBearerMiddlewareDiagnosticsEvents: true); - -builder.Services.AddControllersWithViews(); -builder.Services.AddRazorPages(); -WebApplication app = builder.Build(); - -// Configure the HTTP request pipeline. -if (app.Environment.IsDevelopment()) -{ - app.UseWebAssemblyDebugging(); -} -else -{ - app.UseExceptionHandler("/Error"); - // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts. - app.UseHsts(); -} - -app.UseHttpsRedirection(); -app.UseBlazorFrameworkFiles(); -app.UseStaticFiles(); -app.UseRouting(); -app.UseAuthentication(); -app.UseAuthorization(); -app.MapRazorPages(); -app.MapControllers(); -app.MapFallbackToFile("index.html"); -app.Run(); diff --git a/src/Server/Properties/launchSettings.json b/src/Server/Properties/launchSettings.json deleted file mode 100644 index 423a21d..0000000 --- a/src/Server/Properties/launchSettings.json +++ /dev/null @@ -1,30 +0,0 @@ -{ - "iisSettings": { - "windowsAuthentication": false, - "anonymousAuthentication": true, - "iisExpress": { - "applicationUrl": "http://localhost:21374", - "sslPort": 44327 - } - }, - "profiles": { - "NSeguin.Dev.Web.Server": { - "commandName": "Project", - "dotnetRunMessages": true, - "launchBrowser": true, - "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", - "applicationUrl": "https://localhost:7091;http://localhost:5198", - "environmentVariables": { - "ASPNETCORE_ENVIRONMENT": "Development" - } - }, - "IIS Express": { - "commandName": "IISExpress", - "launchBrowser": true, - "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}", - "environmentVariables": { - "ASPNETCORE_ENVIRONMENT": "Development" - } - } - } -} \ No newline at end of file diff --git a/src/Server/appsettings.json b/src/Server/appsettings.json deleted file mode 100644 index 35a2aba..0000000 --- a/src/Server/appsettings.json +++ /dev/null @@ -1,17 +0,0 @@ -{ - "AzureAdB2C": { - "Instance": "https://nseguindev.b2clogin.com", - "ClientId": "db78f018-cfcd-4bb4-b81e-c139a1f76af6", - "Domain": "nseguindev.onmicrosoft.com", - "Scopes": "access_as_user", - "SignUpSignInPolicyId": "b2c_1_signin", - "RedirectUri": "http://localhost:7091/authentication/login-callback" - }, - "Logging": { - "LogLevel": { - "Default": "Information", - "Microsoft.AspNetCore": "Warning" - } - }, - "AllowedHosts": "*" -} \ No newline at end of file diff --git a/src/Shared/NSeguin.Dev.Web.Shared.csproj b/src/Shared/NSeguin.Dev.Web.Shared.csproj deleted file mode 100644 index 2719da6..0000000 --- a/src/Shared/NSeguin.Dev.Web.Shared.csproj +++ /dev/null @@ -1,10 +0,0 @@ - - - - net8.0 - - - - - - \ No newline at end of file diff --git a/src/Shared/WeatherForecast.cs b/src/Shared/WeatherForecast.cs deleted file mode 100644 index d44d4ed..0000000 --- a/src/Shared/WeatherForecast.cs +++ /dev/null @@ -1,11 +0,0 @@ -namespace NSeguin.Dev.Web.Shared; - -public class WeatherForecast -{ - public int TemperatureF => 32 + (int)(TemperatureC / 0.5556); - public DateTime Date { get; set; } - - public int TemperatureC { get; set; } - - public string? Summary { get; set; } -} diff --git a/targets/Imports.targets b/targets/Imports.targets index 0aeda3c..48bd4fa 100644 --- a/targets/Imports.targets +++ b/targets/Imports.targets @@ -25,9 +25,8 @@ - - - - + + + \ No newline at end of file diff --git a/targets/NativeAot.targets b/targets/NativeAot.targets new file mode 100644 index 0000000..3c48f37 --- /dev/null +++ b/targets/NativeAot.targets @@ -0,0 +1,14 @@ + + + + + false + + + + + true + + + + \ No newline at end of file diff --git a/targets/NativeAotCompatible.targets b/targets/NativeAotCompatible.targets new file mode 100644 index 0000000..c1df222 --- /dev/null +++ b/targets/NativeAotCompatible.targets @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/targets/Settings.props b/targets/Settings.props index cdd7529..32b1cd2 100644 --- a/targets/Settings.props +++ b/targets/Settings.props @@ -2,7 +2,11 @@ enable enable - 12 + preview + + + + true diff --git a/targets/StyleCop.props b/targets/StyleCop.props index f140856..ef7e9a5 100644 --- a/targets/StyleCop.props +++ b/targets/StyleCop.props @@ -6,12 +6,12 @@ - - $(MSBuildThisFileDirectory)../stylecop.json + + stylecop.json - - $(MSBuildThisFileDirectory)../StyleCop.ruleset + + StyleCop.ruleset