├── .github
└── workflows
│ ├── ci.yml
│ └── release.yml
├── .gitignore
├── Build.ps1
├── LICENSE
├── NuGet.config
├── Push.ps1
├── README.md
├── Tailwind.Extensions.AspNetCore.sln
├── Tailwind.Extensions.AspNetCore.sln.DotSettings.user
└── src
├── SampleApps
└── BlazorServer
│ ├── .dockerignore
│ ├── App.razor
│ ├── BlazorServer.csproj
│ ├── Dockerfile
│ ├── Pages
│ ├── Error.cshtml
│ ├── Error.cshtml.cs
│ ├── Index.razor
│ ├── _Host.cshtml
│ └── _Layout.cshtml
│ ├── Program.cs
│ ├── Properties
│ └── launchSettings.json
│ ├── Shared
│ ├── MainLayout.razor
│ ├── MainLayout.razor.css
│ ├── NavMenu.razor
│ ├── NavMenu.razor.css
│ └── SurveyPrompt.razor
│ ├── Styles
│ └── input.css
│ ├── _Imports.razor
│ ├── appsettings.Development.json
│ ├── appsettings.json
│ ├── tailwind.config.js
│ └── wwwroot
│ ├── css
│ └── open-iconic
│ │ ├── FONT-LICENSE
│ │ ├── ICON-LICENSE
│ │ ├── README.md
│ │ └── font
│ │ ├── css
│ │ └── open-iconic-bootstrap.min.css
│ │ └── fonts
│ │ ├── open-iconic.eot
│ │ ├── open-iconic.otf
│ │ ├── open-iconic.svg
│ │ ├── open-iconic.ttf
│ │ └── open-iconic.woff
│ └── favicon.ico
├── Tailwind.Extensions.AspNetCore.Tests
├── GlobalUsings.cs
├── Tailwind.Extensions.AspNetCore.Tests.csproj
└── TailwindCliHostedServiceTests.cs
└── Tailwind.Extensions.AspNetCore
├── Guard.cs
├── Npm
└── NodeScriptRunner.cs
├── Tailwind.Extensions.AspNetCore.csproj
├── TailwindHostedService.cs
├── TailwindMiddleware.cs
├── TailwindMiddlewareExtensions.cs
├── TailwindProcessInterop.cs
└── Util
├── EventedStreamReader.cs
├── EventedStreamStringReader.cs
├── LoggerFinder.cs
├── TaskTimeoutExtensions.cs
└── TcpPortFinder.cs
/.github/workflows/ci.yml:
--------------------------------------------------------------------------------
1 | name: CI
2 |
3 | on:
4 | push:
5 | branches:
6 | - master
7 | - main
8 | - dev
9 | pull_request:
10 | branches:
11 | - master
12 | - main
13 | - dev
14 |
15 | jobs:
16 | build:
17 | runs-on: windows-latest
18 | steps:
19 | - name: Checkout
20 | uses: actions/checkout@v2
21 | with:
22 | fetch-depth: 0
23 | - name: Setup .NET 6
24 | uses: actions/setup-dotnet@v1
25 | with:
26 | dotnet-version: '6.0.x'
27 | - name: Build
28 | run: ./Build.ps1
29 | shell: pwsh
30 | - name: Artifacts
31 | uses: actions/upload-artifact@v2
32 | with:
33 | name: artifacts
34 | path: artifacts/**/*
--------------------------------------------------------------------------------
/.github/workflows/release.yml:
--------------------------------------------------------------------------------
1 | on:
2 | push:
3 | tags:
4 | - '*.*.*'
5 | jobs:
6 | build:
7 | runs-on: windows-latest
8 | steps:
9 | - name: Checkout
10 | uses: actions/checkout@v2
11 | with:
12 | fetch-depth: 0
13 | - name: Setup .NET 6
14 | uses: actions/setup-dotnet@v1
15 | with:
16 | dotnet-version: '6.0.x'
17 | - name: Build
18 | run: ./Build.ps1
19 | shell: pwsh
20 | - name: Push to NuGet
21 | env:
22 | NUGET_URL: https://api.nuget.org/v3/index.json
23 | NUGET_API_KEY: ${{ secrets.NUGET_API_KEY }}
24 | run: ./Push.ps1
25 | shell: pwsh
26 | - name: Artifacts
27 | uses: actions/upload-artifact@v2
28 | with:
29 | name: artifacts
30 | path: artifacts/**/*
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | bin/
2 | obj/
3 | /packages/
4 | riderModule.iml
5 | /_ReSharper.Caches/
6 | /.idea/
7 | /artifacts
8 | **/node_modules/
9 | /src/SampleApps/BlazorServer/wwwroot/css/output.css
10 |
--------------------------------------------------------------------------------
/Build.ps1:
--------------------------------------------------------------------------------
1 | # Taken from MediatR.Extensions.DependencyInjection https://github.com/jbogard/MediatR.Extensions.Microsoft.DependencyInjection/
2 |
3 | <#
4 | .SYNOPSIS
5 | This is a helper function that runs a scriptblock and checks the PS variable $lastexitcode
6 | to see if an error occcured. If an error is detected then an exception is thrown.
7 | This function allows you to run command-line programs without having to
8 | explicitly check the $lastexitcode variable.
9 | .EXAMPLE
10 | exec { svn info $repository_trunk } "Error executing SVN. Please verify SVN command-line client is installed"
11 | #>
12 | function Exec
13 | {
14 | [CmdletBinding()]
15 | param(
16 | [Parameter(Position=0,Mandatory=1)][scriptblock]$cmd,
17 | [Parameter(Position=1,Mandatory=0)][string]$errorMessage = ($msgs.error_bad_command -f $cmd)
18 | )
19 | & $cmd
20 | if ($lastexitcode -ne 0) {
21 | throw ("Exec: " + $errorMessage)
22 | }
23 | }
24 |
25 | $artifacts = ".\artifacts"
26 |
27 | if(Test-Path $artifacts) { Remove-Item $artifacts -Force -Recurse }
28 |
29 | exec { & dotnet clean -c Release }
30 |
31 | exec { & dotnet build -c Release }
32 |
33 | exec { & dotnet test -c Release -r $artifacts --no-build -l trx --verbosity=normal }
34 |
35 | exec { dotnet pack .\src\Tailwind.Extensions.AspNetCore -c Release -o $artifacts --no-build }
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2022 Jon Hilton
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/NuGet.config:
--------------------------------------------------------------------------------
1 |
2 |
3 |
--------------------------------------------------------------------------------
/Push.ps1:
--------------------------------------------------------------------------------
1 | # Taken from MediatR.Extensions.DependencyInjection https://github.com/jbogard/MediatR.Extensions.Microsoft.DependencyInjection/
2 |
3 | $scriptName = $MyInvocation.MyCommand.Name
4 | $artifacts = "./artifacts"
5 |
6 | if ([string]::IsNullOrEmpty($Env:NUGET_API_KEY)) {
7 | Write-Host "${scriptName}: NUGET_API_KEY is empty or not set. Skipped pushing package(s)."
8 | } else {
9 | Get-ChildItem $artifacts -Filter "*.nupkg" | ForEach-Object {
10 | Write-Host "$($scriptName): Pushing $($_.Name)"
11 | dotnet nuget push $_ --source $Env:NUGET_URL --api-key $Env:NUGET_API_KEY
12 | if ($lastexitcode -ne 0) {
13 | throw ("Exec: " + $errorMessage)
14 | }
15 | }
16 | }
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | _Disclaimer: This project is not affiliated with or supported by Tailwind Labs._
2 |
3 | # Use Tailwind's JIT mode with `dotnet run` and `dotnet watch run`
4 |
5 | Makes it possible to use the new "Just In Time" builds for Tailwind (Tailwind 3+) with ASP.NET Core.
6 |
7 | Works for Blazor WASM applications that are hosted via ASP.NET Core, and Blazor Server applications.
8 |
9 | Note it doesn't work with Blazor WASM apps that aren't hosted via ASP.NET Core.
10 |
11 | ## Usage
12 |
13 | Create a new Hosted Blazor WASM, or Blazor Server project.
14 |
15 | CD to the Client App's folder (if Blazor WASM) or Blazor Server App's folder.
16 |
17 | Run these commands:
18 | ``` powershell
19 | npm install -D tailwindcss cross-env
20 | npx tailwindcss init
21 | ```
22 |
23 | This will install Tailwind and the handy cross-env utility via NPM, then create a `tailwind.config.js` file.
24 |
25 | Now update the `tailwind.config.js` file to include all your .razor and .cshtml files.
26 |
27 | ``` javascript
28 | module.exports = {
29 | content: ["**/*.razor", "**/*.cshtml", "**/*.html"],
30 | theme: {
31 | extend: {},
32 | },
33 | plugins: [],
34 | }
35 | ```
36 |
37 | Now you'll want to create the Tailwind input stylesheet. This is the stylesheet that Tailwind will then pick up and build.
38 |
39 | Here's the minimum you'll need...
40 |
41 | **Styles\input.css**
42 |
43 | ``` css
44 | @tailwind base;
45 | @tailwind components;
46 | @tailwind utilities;
47 | ```
48 |
49 | Finally, update your `package.json` file to add this script.
50 |
51 | ``` json
52 | "scripts": {
53 | "tailwind": "cross-env NODE_ENV=development ./node_modules/tailwindcss/lib/cli.js -i ./Styles/input.css -o ./wwwroot/css/output.css --watch"
54 | },
55 | ```
56 |
57 | Make sure **.Styles/input.css** is pointing to the css file you created in the last step. You can control where the resulting css file is created by specifying your own value for the `-o` parameter.
58 |
59 | That takes care of the Tailwind setup, now we just need to make this run when you run your ASP.NET Core project during Development.
60 |
61 | Make sure you switch to the Server App's folder if you're using Blazor WASM (hosted).
62 |
63 | Run this command to install the Tailwind AspNetCore NuGet package.
64 |
65 | ``` powershell
66 | dotnet add package Tailwind.Extensions.AspNetCore --version 1.0.0-beta2
67 | ```
68 |
69 | Now head over to `Program.cs` and add this code before `app.Run()`;
70 |
71 | ``` csharp
72 | if (app.Environment.IsDevelopment())
73 | {
74 | _ = app.RunTailwind("tailwind", "./");
75 | }
76 | ```
77 |
78 | The second argument is the path to the folder containing your package.json file. If you're using Blazor WASM you'll probably need something like this...
79 |
80 | ``` csharp
81 | if (app.Environment.IsDevelopment())
82 | {
83 | _ = app.RunTailwind("tailwind", "../Client/");
84 | }
85 | ```
86 |
87 | You'll also need to add this `using` statement:
88 |
89 | ``` csharp
90 | using Tailwind;
91 | ```
92 |
93 | Note we're using the discard parameter `_` when we call `app.RunTailwind`. This is because the method is async, but we don't want to wait for it to complete (as this would cause your app to wait for it to finish, and we want it continue running in the background alongside your app). Using the `_` parameter here stops your IDE nagging at you to await the call 😀
94 |
95 | Now, run `dotnet watch run` and try modifying your Razor components (using Tailwind's utility classes).
96 |
97 | You should see logs indicating that tailwind has rebuilt the CSS stylesheet successfully.
98 |
--------------------------------------------------------------------------------
/Tailwind.Extensions.AspNetCore.sln:
--------------------------------------------------------------------------------
1 |
2 | Microsoft Visual Studio Solution File, Format Version 12.00
3 | Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Tailwind.Extensions.AspNetCore", "src\Tailwind.Extensions.AspNetCore\Tailwind.Extensions.AspNetCore.csproj", "{3A29D45C-D8D5-4D52-8E4E-F295F3EF14C3}"
4 | EndProject
5 | Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "BlazorServer", "src\SampleApps\BlazorServer\BlazorServer.csproj", "{AA1AF6B6-6B5B-45B4-867C-9B04CAFDFB84}"
6 | EndProject
7 | Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Tailwind.Extensions.AspNetCore.Tests", "src\Tailwind.Extensions.AspNetCore.Tests\Tailwind.Extensions.AspNetCore.Tests.csproj", "{FF0BEEBB-F708-4F4B-9839-A82AE71835C4}"
8 | EndProject
9 | Global
10 | GlobalSection(SolutionConfigurationPlatforms) = preSolution
11 | Debug|Any CPU = Debug|Any CPU
12 | Release|Any CPU = Release|Any CPU
13 | EndGlobalSection
14 | GlobalSection(ProjectConfigurationPlatforms) = postSolution
15 | {3A29D45C-D8D5-4D52-8E4E-F295F3EF14C3}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
16 | {3A29D45C-D8D5-4D52-8E4E-F295F3EF14C3}.Debug|Any CPU.Build.0 = Debug|Any CPU
17 | {3A29D45C-D8D5-4D52-8E4E-F295F3EF14C3}.Release|Any CPU.ActiveCfg = Release|Any CPU
18 | {3A29D45C-D8D5-4D52-8E4E-F295F3EF14C3}.Release|Any CPU.Build.0 = Release|Any CPU
19 | {AA1AF6B6-6B5B-45B4-867C-9B04CAFDFB84}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
20 | {AA1AF6B6-6B5B-45B4-867C-9B04CAFDFB84}.Debug|Any CPU.Build.0 = Debug|Any CPU
21 | {AA1AF6B6-6B5B-45B4-867C-9B04CAFDFB84}.Release|Any CPU.ActiveCfg = Release|Any CPU
22 | {AA1AF6B6-6B5B-45B4-867C-9B04CAFDFB84}.Release|Any CPU.Build.0 = Release|Any CPU
23 | {FF0BEEBB-F708-4F4B-9839-A82AE71835C4}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
24 | {FF0BEEBB-F708-4F4B-9839-A82AE71835C4}.Debug|Any CPU.Build.0 = Debug|Any CPU
25 | {FF0BEEBB-F708-4F4B-9839-A82AE71835C4}.Release|Any CPU.ActiveCfg = Release|Any CPU
26 | {FF0BEEBB-F708-4F4B-9839-A82AE71835C4}.Release|Any CPU.Build.0 = Release|Any CPU
27 | EndGlobalSection
28 | EndGlobal
29 |
--------------------------------------------------------------------------------
/Tailwind.Extensions.AspNetCore.sln.DotSettings.user:
--------------------------------------------------------------------------------
1 |
2 | <SessionState ContinuousTestingMode="0" IsActive="True" Name="MissingOptionsThrowsError" xmlns="urn:schemas-jetbrains-com:jetbrains-ut-session">
3 | <Project Location="C:\Users\hilto\RiderProjects\Tailwind.Extensions.AspNetCore\Tailwind.Extensions.AspNetCore.Tests" Presentation="<Tailwind.Extensions.AspNetCore.Tests>" />
4 | </SessionState>
5 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/.dockerignore:
--------------------------------------------------------------------------------
1 | **/.dockerignore
2 | **/.env
3 | **/.git
4 | **/.gitignore
5 | **/.project
6 | **/.settings
7 | **/.toolstarget
8 | **/.vs
9 | **/.vscode
10 | **/.idea
11 | **/*.*proj.user
12 | **/*.dbmdl
13 | **/*.jfm
14 | **/azds.yaml
15 | **/bin
16 | **/charts
17 | **/docker-compose*
18 | **/Dockerfile*
19 | **/node_modules
20 | **/npm-debug.log
21 | **/obj
22 | **/secrets.dev.yaml
23 | **/values.dev.yaml
24 | LICENSE
25 | README.md
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/App.razor:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 | Not found
8 |
9 | Sorry, there's nothing at this address.
10 |
11 |
12 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/BlazorServer.csproj:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | net6.0
5 | enable
6 | enable
7 | Linux
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 | <_ContentIncludedByDefault Remove="wwwroot\css\bootstrap\bootstrap.min.css" />
16 | <_ContentIncludedByDefault Remove="wwwroot\css\bootstrap\bootstrap.min.css.map" />
17 |
18 |
19 |
20 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM mcr.microsoft.com/dotnet/aspnet:6.0 AS base
2 | WORKDIR /app
3 | EXPOSE 80
4 | EXPOSE 443
5 |
6 | FROM mcr.microsoft.com/dotnet/sdk:6.0 AS build
7 | WORKDIR /src
8 | COPY ["src/SampleApps/BlazorServer/BlazorServer.csproj", "BlazorServer/"]
9 | RUN dotnet restore "src/SampleApps/BlazorServer/BlazorServer.csproj"
10 | COPY . .
11 | WORKDIR "/src/BlazorServer"
12 | RUN dotnet build "BlazorServer.csproj" -c Release -o /app/build
13 |
14 | FROM build AS publish
15 | RUN dotnet publish "BlazorServer.csproj" -c Release -o /app/publish
16 |
17 | FROM base AS final
18 | WORKDIR /app
19 | COPY --from=publish /app/publish .
20 | ENTRYPOINT ["dotnet", "BlazorServer.dll"]
21 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Pages/Error.cshtml:
--------------------------------------------------------------------------------
1 | @page
2 | @model BlazorServer.Pages.ErrorModel
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 | Error
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
Error.
19 |
An error occurred while processing your request.
20 |
21 | @if (Model.ShowRequestId)
22 | {
23 |
24 | Request ID: @Model.RequestId
25 |
26 | }
27 |
28 |
Development Mode
29 |
30 | Swapping to the Development environment displays detailed information about the error that occurred.
31 |
32 |
33 | The Development environment shouldn't be enabled for deployed applications.
34 | It can result in displaying sensitive information from exceptions to end users.
35 | For local debugging, enable the Development environment by setting the ASPNETCORE_ENVIRONMENT environment variable to Development
36 | and restarting the app.
37 |
38 |
39 |
40 |
41 |
42 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Pages/Error.cshtml.cs:
--------------------------------------------------------------------------------
1 | using System.Diagnostics;
2 | using Microsoft.AspNetCore.Mvc;
3 | using Microsoft.AspNetCore.Mvc.RazorPages;
4 |
5 | namespace BlazorServer.Pages;
6 |
7 | [ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)]
8 | [IgnoreAntiforgeryToken]
9 | public class ErrorModel : PageModel
10 | {
11 | public string? RequestId { get; set; }
12 |
13 | public bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
14 |
15 | private readonly ILogger _logger;
16 |
17 | public ErrorModel(ILogger logger)
18 | {
19 | _logger = logger;
20 | }
21 |
22 | public void OnGet()
23 | {
24 | RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier;
25 | }
26 | }
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Pages/Index.razor:
--------------------------------------------------------------------------------
1 | @page "/"
2 |
3 |
4 |
Hello World
5 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Pages/_Host.cshtml:
--------------------------------------------------------------------------------
1 | @page "/"
2 | @namespace BlazorServer.Pages
3 | @addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers
4 | @{
5 | Layout = "_Layout";
6 | }
7 |
8 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Pages/_Layout.cshtml:
--------------------------------------------------------------------------------
1 | @using Microsoft.AspNetCore.Components.Web
2 | @namespace BlazorServer.Pages
3 | @addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 | @RenderBody()
17 |
18 |
19 |
20 | An error has occurred. This application may no longer respond until reloaded.
21 |
22 |
23 | An unhandled exception has occurred. See browser dev tools for details.
24 |
25 |
Reload
26 |
🗙
27 |
28 |
29 |
30 |
31 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Program.cs:
--------------------------------------------------------------------------------
1 | using Tailwind;
2 |
3 | var builder = WebApplication.CreateBuilder(args);
4 |
5 | // Add services to the container.
6 | builder.Services.AddRazorPages();
7 | builder.Services.AddServerSideBlazor();
8 |
9 | // use this to automatically watch for css changes using Tailwind CLI
10 | // configure via appsettings "Tailwind"
11 | builder.UseTailwindCli();
12 |
13 | var app = builder.Build();
14 |
15 | // Configure the HTTP request pipeline.
16 | if (!app.Environment.IsDevelopment())
17 | {
18 | app.UseExceptionHandler("/Error");
19 | // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
20 | app.UseHsts();
21 | }
22 |
23 | app.UseHttpsRedirection();
24 |
25 | app.UseStaticFiles();
26 |
27 | app.UseRouting();
28 |
29 | app.MapBlazorHub();
30 | app.MapFallbackToPage("/_Host");
31 |
32 | if (app.Environment.IsDevelopment())
33 | {
34 | // app.RunTailwind("tailwind", "./");
35 | }
36 |
37 | app.Run();
38 |
39 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Properties/launchSettings.json:
--------------------------------------------------------------------------------
1 | {
2 | "iisSettings": {
3 | "windowsAuthentication": false,
4 | "anonymousAuthentication": true,
5 | "iisExpress": {
6 | "applicationUrl": "http://localhost:17460",
7 | "sslPort": 44325
8 | }
9 | },
10 | "profiles": {
11 | "BlazorServer": {
12 | "commandName": "Project",
13 | "dotnetRunMessages": true,
14 | "launchBrowser": true,
15 | "applicationUrl": "https://localhost:7076;http://localhost:5076",
16 | "environmentVariables": {
17 | "ASPNETCORE_ENVIRONMENT": "Development"
18 | }
19 | },
20 | "IIS Express": {
21 | "commandName": "IISExpress",
22 | "launchBrowser": true,
23 | "environmentVariables": {
24 | "ASPNETCORE_ENVIRONMENT": "Development"
25 | }
26 | }
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Shared/MainLayout.razor:
--------------------------------------------------------------------------------
1 | @inherits LayoutComponentBase
2 |
3 | Blazor Server Tailwind Demo
4 |
5 |
6 |
7 | @Body
8 |
9 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Shared/MainLayout.razor.css:
--------------------------------------------------------------------------------
1 | .page {
2 | position: relative;
3 | display: flex;
4 | flex-direction: column;
5 | }
6 |
7 | main {
8 | flex: 1;
9 | }
10 |
11 | .sidebar {
12 | background-image: linear-gradient(180deg, rgb(5, 39, 103) 0%, #3a0647 70%);
13 | }
14 |
15 | .top-row {
16 | background-color: #f7f7f7;
17 | border-bottom: 1px solid #d6d5d5;
18 | justify-content: flex-end;
19 | height: 3.5rem;
20 | display: flex;
21 | align-items: center;
22 | }
23 |
24 | .top-row ::deep a, .top-row .btn-link {
25 | white-space: nowrap;
26 | margin-left: 1.5rem;
27 | }
28 |
29 | .top-row a:first-child {
30 | overflow: hidden;
31 | text-overflow: ellipsis;
32 | }
33 |
34 | @media (max-width: 640.98px) {
35 | .top-row:not(.auth) {
36 | display: none;
37 | }
38 |
39 | .top-row.auth {
40 | justify-content: space-between;
41 | }
42 |
43 | .top-row a, .top-row .btn-link {
44 | margin-left: 0;
45 | }
46 | }
47 |
48 | @media (min-width: 641px) {
49 | .page {
50 | flex-direction: row;
51 | }
52 |
53 | .sidebar {
54 | width: 250px;
55 | height: 100vh;
56 | position: sticky;
57 | top: 0;
58 | }
59 |
60 | .top-row {
61 | position: sticky;
62 | top: 0;
63 | z-index: 1;
64 | }
65 |
66 | .top-row, article {
67 | padding-left: 2rem !important;
68 | padding-right: 1.5rem !important;
69 | }
70 | }
71 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Shared/NavMenu.razor:
--------------------------------------------------------------------------------
1 |
9 |
10 |
29 |
30 | @code {
31 | private bool collapseNavMenu = true;
32 |
33 | private string? NavMenuCssClass => collapseNavMenu ? "collapse" : null;
34 |
35 | private void ToggleNavMenu()
36 | {
37 | collapseNavMenu = !collapseNavMenu;
38 | }
39 |
40 | }
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Shared/NavMenu.razor.css:
--------------------------------------------------------------------------------
1 | .navbar-toggler {
2 | background-color: rgba(255, 255, 255, 0.1);
3 | }
4 |
5 | .top-row {
6 | height: 3.5rem;
7 | background-color: rgba(0,0,0,0.4);
8 | }
9 |
10 | .navbar-brand {
11 | font-size: 1.1rem;
12 | }
13 |
14 | .oi {
15 | width: 2rem;
16 | font-size: 1.1rem;
17 | vertical-align: text-top;
18 | top: -2px;
19 | }
20 |
21 | .nav-item {
22 | font-size: 0.9rem;
23 | padding-bottom: 0.5rem;
24 | }
25 |
26 | .nav-item:first-of-type {
27 | padding-top: 1rem;
28 | }
29 |
30 | .nav-item:last-of-type {
31 | padding-bottom: 1rem;
32 | }
33 |
34 | .nav-item ::deep a {
35 | color: #d7d7d7;
36 | border-radius: 4px;
37 | height: 3rem;
38 | display: flex;
39 | align-items: center;
40 | line-height: 3rem;
41 | }
42 |
43 | .nav-item ::deep a.active {
44 | background-color: rgba(255,255,255,0.25);
45 | color: white;
46 | }
47 |
48 | .nav-item ::deep a:hover {
49 | background-color: rgba(255,255,255,0.1);
50 | color: white;
51 | }
52 |
53 | @media (min-width: 641px) {
54 | .navbar-toggler {
55 | display: none;
56 | }
57 |
58 | .collapse {
59 | /* Never collapse the sidebar for wide screens */
60 | display: block;
61 | }
62 | }
63 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Shared/SurveyPrompt.razor:
--------------------------------------------------------------------------------
1 |
2 |
3 |
@Title
4 |
5 |
6 | Please take our
7 | brief survey
8 |
9 | and tell us what you think.
10 |
11 |
12 | @code {
13 | // Demonstrates how a parent component can supply parameters
14 | [Parameter]
15 | public string? Title { get; set; }
16 |
17 | }
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/Styles/input.css:
--------------------------------------------------------------------------------
1 | @tailwind base;
2 | @tailwind components;
3 | @tailwind utilities;
4 |
5 | .valid.modified:not([type=checkbox]) {
6 | outline: 1px solid #26b050;
7 | }
8 |
9 | .invalid {
10 | outline: 1px solid red;
11 | }
12 |
13 | .validation-message {
14 | color: red;
15 | }
16 |
17 | #blazor-error-ui {
18 | background: lightyellow;
19 | bottom: 0;
20 | box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2);
21 | display: none;
22 | left: 0;
23 | padding: 0.6rem 1.25rem 0.7rem 1.25rem;
24 | position: fixed;
25 | width: 100%;
26 | z-index: 1000;
27 | }
28 |
29 | #blazor-error-ui .dismiss {
30 | cursor: pointer;
31 | position: absolute;
32 | right: 0.75rem;
33 | top: 0.5rem;
34 | }
35 |
36 | .blazor-error-boundary {
37 | background: url() no-repeat 1rem/1.8rem, #b32121;
38 | padding: 1rem 1rem 1rem 3.7rem;
39 | color: white;
40 | }
41 |
42 | .blazor-error-boundary::after {
43 | content: "An error has occurred."
44 | }
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/_Imports.razor:
--------------------------------------------------------------------------------
1 | @using System.Net.Http
2 | @using Microsoft.AspNetCore.Authorization
3 | @using Microsoft.AspNetCore.Components.Authorization
4 | @using Microsoft.AspNetCore.Components.Forms
5 | @using Microsoft.AspNetCore.Components.Routing
6 | @using Microsoft.AspNetCore.Components.Web
7 | @using Microsoft.AspNetCore.Components.Web.Virtualization
8 | @using Microsoft.JSInterop
9 | @using BlazorServer
10 | @using BlazorServer.Shared
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/appsettings.Development.json:
--------------------------------------------------------------------------------
1 | {
2 | "DetailedErrors": true,
3 | "Logging": {
4 | "LogLevel": {
5 | "Default": "Information",
6 | "Microsoft.AspNetCore": "Warning"
7 | }
8 | },
9 | "Tailwind": {
10 | "InputFile": "./Styles/input.css",
11 | "OutputFile": "./wwwroot/css/output.css",
12 | "TailwindCliPath": ""
13 | }
14 | }
15 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/appsettings.json:
--------------------------------------------------------------------------------
1 | {
2 | "Logging": {
3 | "LogLevel": {
4 | "Default": "Information",
5 | "Microsoft.AspNetCore": "Warning"
6 | }
7 | },
8 | "AllowedHosts": "*"
9 | }
10 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/tailwind.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | content: ["**/*.razor", "**/*.cshtml", "**/*.html"],
3 | theme: {
4 | extend: {},
5 | },
6 | plugins: [],
7 | }
8 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/FONT-LICENSE:
--------------------------------------------------------------------------------
1 | SIL OPEN FONT LICENSE Version 1.1
2 |
3 | Copyright (c) 2014 Waybury
4 |
5 | PREAMBLE
6 | The goals of the Open Font License (OFL) are to stimulate worldwide
7 | development of collaborative font projects, to support the font creation
8 | efforts of academic and linguistic communities, and to provide a free and
9 | open framework in which fonts may be shared and improved in partnership
10 | with others.
11 |
12 | The OFL allows the licensed fonts to be used, studied, modified and
13 | redistributed freely as long as they are not sold by themselves. The
14 | fonts, including any derivative works, can be bundled, embedded,
15 | redistributed and/or sold with any software provided that any reserved
16 | names are not used by derivative works. The fonts and derivatives,
17 | however, cannot be released under any other type of license. The
18 | requirement for fonts to remain under this license does not apply
19 | to any document created using the fonts or their derivatives.
20 |
21 | DEFINITIONS
22 | "Font Software" refers to the set of files released by the Copyright
23 | Holder(s) under this license and clearly marked as such. This may
24 | include source files, build scripts and documentation.
25 |
26 | "Reserved Font Name" refers to any names specified as such after the
27 | copyright statement(s).
28 |
29 | "Original Version" refers to the collection of Font Software components as
30 | distributed by the Copyright Holder(s).
31 |
32 | "Modified Version" refers to any derivative made by adding to, deleting,
33 | or substituting -- in part or in whole -- any of the components of the
34 | Original Version, by changing formats or by porting the Font Software to a
35 | new environment.
36 |
37 | "Author" refers to any designer, engineer, programmer, technical
38 | writer or other person who contributed to the Font Software.
39 |
40 | PERMISSION & CONDITIONS
41 | Permission is hereby granted, free of charge, to any person obtaining
42 | a copy of the Font Software, to use, study, copy, merge, embed, modify,
43 | redistribute, and sell modified and unmodified copies of the Font
44 | Software, subject to the following conditions:
45 |
46 | 1) Neither the Font Software nor any of its individual components,
47 | in Original or Modified Versions, may be sold by itself.
48 |
49 | 2) Original or Modified Versions of the Font Software may be bundled,
50 | redistributed and/or sold with any software, provided that each copy
51 | contains the above copyright notice and this license. These can be
52 | included either as stand-alone text files, human-readable headers or
53 | in the appropriate machine-readable metadata fields within text or
54 | binary files as long as those fields can be easily viewed by the user.
55 |
56 | 3) No Modified Version of the Font Software may use the Reserved Font
57 | Name(s) unless explicit written permission is granted by the corresponding
58 | Copyright Holder. This restriction only applies to the primary font name as
59 | presented to the users.
60 |
61 | 4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
62 | Software shall not be used to promote, endorse or advertise any
63 | Modified Version, except to acknowledge the contribution(s) of the
64 | Copyright Holder(s) and the Author(s) or with their explicit written
65 | permission.
66 |
67 | 5) The Font Software, modified or unmodified, in part or in whole,
68 | must be distributed entirely under this license, and must not be
69 | distributed under any other license. The requirement for fonts to
70 | remain under this license does not apply to any document created
71 | using the Font Software.
72 |
73 | TERMINATION
74 | This license becomes null and void if any of the above conditions are
75 | not met.
76 |
77 | DISCLAIMER
78 | THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
79 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
80 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
81 | OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
82 | COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
83 | INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
84 | DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
85 | FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
86 | OTHER DEALINGS IN THE FONT SOFTWARE.
87 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/ICON-LICENSE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2014 Waybury
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in
13 | all copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21 | THE SOFTWARE.
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/README.md:
--------------------------------------------------------------------------------
1 | [Open Iconic v1.1.1](http://useiconic.com/open)
2 | ===========
3 |
4 | ### Open Iconic is the open source sibling of [Iconic](http://useiconic.com). It is a hyper-legible collection of 223 icons with a tiny footprint—ready to use with Bootstrap and Foundation. [View the collection](http://useiconic.com/open#icons)
5 |
6 |
7 |
8 | ## What's in Open Iconic?
9 |
10 | * 223 icons designed to be legible down to 8 pixels
11 | * Super-light SVG files - 61.8 for the entire set
12 | * SVG sprite—the modern replacement for icon fonts
13 | * Webfont (EOT, OTF, SVG, TTF, WOFF), PNG and WebP formats
14 | * Webfont stylesheets (including versions for Bootstrap and Foundation) in CSS, LESS, SCSS and Stylus formats
15 | * PNG and WebP raster images in 8px, 16px, 24px, 32px, 48px and 64px.
16 |
17 |
18 | ## Getting Started
19 |
20 | #### For code samples and everything else you need to get started with Open Iconic, check out our [Icons](http://useiconic.com/open#icons) and [Reference](http://useiconic.com/open#reference) sections.
21 |
22 | ### General Usage
23 |
24 | #### Using Open Iconic's SVGs
25 |
26 | 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).
27 |
28 | ```
29 |
30 | ```
31 |
32 | #### Using Open Iconic's SVG Sprite
33 |
34 | 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.
35 |
36 | 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.*
37 |
38 | ```
39 |
40 |
41 |
42 | ```
43 |
44 | 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.
45 |
46 | ```
47 | .icon {
48 | width: 16px;
49 | height: 16px;
50 | }
51 | ```
52 |
53 | Coloring icons is even easier. All you need to do is set the `fill` rule on the `` tag.
54 |
55 | ```
56 | .icon-account-login {
57 | fill: #f00;
58 | }
59 | ```
60 |
61 | To learn more about SVG Sprites, read [Chris Coyier's guide](http://css-tricks.com/svg-sprites-use-better-icon-fonts/).
62 |
63 | #### Using Open Iconic's Icon Font...
64 |
65 |
66 | ##### …with Bootstrap
67 |
68 | You can find our Bootstrap stylesheets in `font/css/open-iconic-bootstrap.{css, less, scss, styl}`
69 |
70 |
71 | ```
72 |
73 | ```
74 |
75 |
76 | ```
77 |
78 | ```
79 |
80 | ##### …with Foundation
81 |
82 | You can find our Foundation stylesheets in `font/css/open-iconic-foundation.{css, less, scss, styl}`
83 |
84 | ```
85 |
86 | ```
87 |
88 |
89 | ```
90 |
91 | ```
92 |
93 | ##### …on its own
94 |
95 | You can find our default stylesheets in `font/css/open-iconic.{css, less, scss, styl}`
96 |
97 | ```
98 |
99 | ```
100 |
101 | ```
102 |
103 | ```
104 |
105 |
106 | ## License
107 |
108 | ### Icons
109 |
110 | All code (including SVG markup) is under the [MIT License](http://opensource.org/licenses/MIT).
111 |
112 | ### Fonts
113 |
114 | All fonts are under the [SIL Licensed](http://scripts.sil.org/cms/scripts/page.php?item_id=OFL_web).
115 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/css/open-iconic-bootstrap.min.css:
--------------------------------------------------------------------------------
1 | @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'}
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/Practical-ASP-NET/Tailwind.Extensions.AspNetCore/0f01d2bfe18ebe6fa32bc16e0b525e7755bc00d6/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.eot
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.otf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/Practical-ASP-NET/Tailwind.Extensions.AspNetCore/0f01d2bfe18ebe6fa32bc16e0b525e7755bc00d6/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.otf
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
6 |
7 |
8 | Created by FontForge 20120731 at Tue Jul 1 20:39:22 2014
9 | By P.J. Onori
10 | Created by P.J. Onori with FontForge 2.0 (http://fontforge.sf.net)
11 |
12 |
13 |
14 |
27 |
28 |
30 |
32 |
34 |
36 |
38 |
40 |
42 |
45 |
47 |
49 |
51 |
53 |
55 |
57 |
59 |
61 |
63 |
65 |
67 |
69 |
71 |
74 |
76 |
79 |
81 |
84 |
86 |
88 |
91 |
93 |
95 |
98 |
100 |
102 |
104 |
106 |
109 |
112 |
115 |
117 |
121 |
123 |
125 |
127 |
130 |
132 |
134 |
136 |
138 |
141 |
143 |
145 |
147 |
149 |
151 |
153 |
155 |
157 |
159 |
162 |
165 |
167 |
169 |
172 |
174 |
177 |
179 |
181 |
183 |
185 |
189 |
191 |
194 |
196 |
198 |
200 |
202 |
205 |
207 |
209 |
211 |
213 |
215 |
218 |
220 |
222 |
224 |
226 |
228 |
230 |
232 |
234 |
236 |
238 |
241 |
243 |
245 |
247 |
249 |
251 |
253 |
256 |
259 |
261 |
263 |
265 |
267 |
269 |
272 |
274 |
276 |
280 |
282 |
285 |
287 |
289 |
292 |
295 |
298 |
300 |
302 |
304 |
306 |
309 |
312 |
314 |
316 |
318 |
320 |
322 |
324 |
326 |
330 |
334 |
338 |
340 |
343 |
345 |
347 |
349 |
351 |
353 |
355 |
358 |
360 |
363 |
365 |
367 |
369 |
371 |
373 |
375 |
377 |
379 |
381 |
383 |
386 |
388 |
390 |
392 |
394 |
396 |
399 |
401 |
404 |
406 |
408 |
410 |
412 |
414 |
416 |
419 |
421 |
423 |
425 |
428 |
431 |
435 |
438 |
440 |
442 |
444 |
446 |
448 |
451 |
453 |
455 |
457 |
460 |
462 |
464 |
466 |
468 |
471 |
473 |
477 |
479 |
481 |
483 |
486 |
488 |
490 |
492 |
494 |
496 |
499 |
501 |
504 |
506 |
509 |
512 |
515 |
517 |
520 |
522 |
524 |
526 |
529 |
532 |
534 |
536 |
539 |
542 |
543 |
544 |
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/Practical-ASP-NET/Tailwind.Extensions.AspNetCore/0f01d2bfe18ebe6fa32bc16e0b525e7755bc00d6/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.ttf
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/Practical-ASP-NET/Tailwind.Extensions.AspNetCore/0f01d2bfe18ebe6fa32bc16e0b525e7755bc00d6/src/SampleApps/BlazorServer/wwwroot/css/open-iconic/font/fonts/open-iconic.woff
--------------------------------------------------------------------------------
/src/SampleApps/BlazorServer/wwwroot/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/Practical-ASP-NET/Tailwind.Extensions.AspNetCore/0f01d2bfe18ebe6fa32bc16e0b525e7755bc00d6/src/SampleApps/BlazorServer/wwwroot/favicon.ico
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore.Tests/GlobalUsings.cs:
--------------------------------------------------------------------------------
1 | global using Xunit;
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore.Tests/Tailwind.Extensions.AspNetCore.Tests.csproj:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | net6.0
5 | enable
6 | enable
7 |
8 | false
9 | true
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 | runtime; build; native; contentfiles; analyzers; buildtransitive
18 | all
19 |
20 |
21 | runtime; build; native; contentfiles; analyzers; buildtransitive
22 | all
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore.Tests/TailwindCliHostedServiceTests.cs:
--------------------------------------------------------------------------------
1 | using FakeItEasy;
2 | using Microsoft.Extensions.Hosting;
3 | using Microsoft.Extensions.Logging;
4 | using Microsoft.Extensions.Options;
5 |
6 | namespace Tailwind.Extensions.AspNetCore.Tests;
7 |
8 | public class TailwindCliHostedServiceTests
9 | {
10 | [Fact]
11 | public async Task MissingOptionsShouldThrowException()
12 | {
13 | var tailwindOptions = new TailwindOptions
14 | {
15 | InputFile = "", // empty InputFile
16 | OutputFile = "", // empty OutputFile
17 | TailwindCliPath = null // null TailwindCliPath
18 | };
19 | var options = Options.Create(tailwindOptions);
20 | var tailwindProcess = A.Fake();
21 | var hostEnvironment = A.Fake();
22 | var logger = A.Fake>();
23 |
24 | A.CallTo(() => hostEnvironment.EnvironmentName).Returns(Environments.Development);
25 | var tailwindHostedService = new TailwindHostedService(options, hostEnvironment, tailwindProcess, logger);
26 |
27 | await Assert.ThrowsAsync(async () =>
28 | await tailwindHostedService.StartAsync(CancellationToken.None));
29 | }
30 |
31 | [Theory]
32 | [InlineData(null, "tailwind")]
33 | [InlineData("", "tailwind")]
34 | [InlineData("tailwindoverride.exe", "tailwindoverride.exe")]
35 | [InlineData("tailwindoverride", "tailwindoverride")]
36 | public async Task SpecifiedTailwindCliPathOverridesDefault(string tailwindCliPath, string expectedCliPath)
37 | {
38 | var tailwindOptions = new TailwindOptions
39 | {
40 | InputFile = "input.css", // empty InputFile
41 | OutputFile = "output.css", // empty OutputFile
42 | TailwindCliPath = tailwindCliPath // null TailwindCliPath
43 | };
44 | var options = Options.Create(tailwindOptions);
45 | var tailwindProcess = A.Fake();
46 | var hostEnvironment = A.Fake();
47 | var logger = A.Fake>();
48 |
49 | A.CallTo(() => hostEnvironment.EnvironmentName).Returns(Environments.Development);
50 |
51 | var tailwindHostedService = new TailwindHostedService(options, hostEnvironment, tailwindProcess, logger);
52 | await tailwindHostedService.StartAsync(CancellationToken.None);
53 |
54 | A.CallTo(() => tailwindProcess.StartProcess(expectedCliPath,A.Ignored)).MustHaveHappened();
55 | }
56 | }
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/Guard.cs:
--------------------------------------------------------------------------------
1 | namespace Tailwind;
2 |
3 | public static class Guard
4 | {
5 | public static void AgainstNull(string? parameterName, string? errorMessage = "")
6 | {
7 | if (string.IsNullOrEmpty(parameterName))
8 | {
9 | throw new ArgumentNullException($"{parameterName} must not be null or white space. ${errorMessage}");
10 | }
11 | }
12 | }
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/Npm/NodeScriptRunner.cs:
--------------------------------------------------------------------------------
1 | // Licensed to the .NET Foundation under one or more agreements.
2 | // The .NET Foundation licenses this file to you under the MIT license.
3 |
4 | using System.Diagnostics;
5 | using System.Text.RegularExpressions;
6 | using Microsoft.AspNetCore.NodeServices.Util;
7 | using Microsoft.Extensions.Logging;
8 |
9 | // This is under the NodeServices namespace because post 2.1 it will be moved to that package
10 | namespace Microsoft.AspNetCore.NodeServices.Npm;
11 |
12 | ///
13 | /// Executes the script entries defined in a package.json file,
14 | /// capturing any output written to stdio.
15 | ///
16 | internal class NodeScriptRunner : IDisposable
17 | {
18 | private Process? _npmProcess;
19 | public EventedStreamReader StdOut { get; }
20 | public EventedStreamReader StdErr { get; }
21 |
22 | private static readonly Regex AnsiColorRegex = new Regex("\x001b\\[[0-9;]*m", RegexOptions.None, TimeSpan.FromSeconds(1));
23 |
24 | public NodeScriptRunner(string workingDirectory, string scriptName, string? arguments, IDictionary? envVars, string pkgManagerCommand, DiagnosticSource diagnosticSource, CancellationToken applicationStoppingToken)
25 | {
26 | if (string.IsNullOrEmpty(workingDirectory))
27 | {
28 | throw new ArgumentException("Cannot be null or empty.", nameof(workingDirectory));
29 | }
30 |
31 | if (string.IsNullOrEmpty(scriptName))
32 | {
33 | throw new ArgumentException("Cannot be null or empty.", nameof(scriptName));
34 | }
35 |
36 | if (string.IsNullOrEmpty(pkgManagerCommand))
37 | {
38 | throw new ArgumentException("Cannot be null or empty.", nameof(pkgManagerCommand));
39 | }
40 |
41 | var exeToRun = pkgManagerCommand;
42 | var completeArguments = $"run {scriptName} -- {arguments ?? string.Empty}";
43 | if (OperatingSystem.IsWindows())
44 | {
45 | // On Windows, the node executable is a .cmd file, so it can't be executed
46 | // directly (except with UseShellExecute=true, but that's no good, because
47 | // it prevents capturing stdio). So we need to invoke it via "cmd /c".
48 | exeToRun = "cmd";
49 | completeArguments = $"/c {pkgManagerCommand} {completeArguments}";
50 | }
51 |
52 | var processStartInfo = new ProcessStartInfo(exeToRun)
53 | {
54 | Arguments = completeArguments,
55 | UseShellExecute = false,
56 | RedirectStandardInput = true,
57 | RedirectStandardOutput = true,
58 | RedirectStandardError = true,
59 | WorkingDirectory = workingDirectory
60 | };
61 |
62 | if (envVars != null)
63 | {
64 | foreach (var keyValuePair in envVars)
65 | {
66 | processStartInfo.Environment[keyValuePair.Key] = keyValuePair.Value;
67 | }
68 | }
69 |
70 | _npmProcess = LaunchNodeProcess(processStartInfo, pkgManagerCommand);
71 | StdOut = new EventedStreamReader(_npmProcess.StandardOutput);
72 | StdErr = new EventedStreamReader(_npmProcess.StandardError);
73 |
74 | applicationStoppingToken.Register(((IDisposable)this).Dispose);
75 |
76 | if (diagnosticSource.IsEnabled("Microsoft.AspNetCore.NodeServices.Npm.NpmStarted"))
77 | {
78 | diagnosticSource.Write(
79 | "Microsoft.AspNetCore.NodeServices.Npm.NpmStarted",
80 | new
81 | {
82 | processStartInfo = processStartInfo,
83 | process = _npmProcess
84 | });
85 | }
86 | }
87 |
88 | public void AttachToLogger(ILogger logger, bool treatErrorsAsInfo = false)
89 | {
90 | // When the node task emits complete lines, pass them through to the real logger
91 | StdOut.OnReceivedLine += line =>
92 | {
93 | if (!string.IsNullOrWhiteSpace(line))
94 | {
95 | // Node tasks commonly emit ANSI colors, but it wouldn't make sense to forward
96 | // those to loggers (because a logger isn't necessarily any kind of terminal)
97 | logger.LogInformation(StripAnsiColors(line));
98 | }
99 | };
100 |
101 | StdErr.OnReceivedLine += line =>
102 | {
103 | if (!string.IsNullOrWhiteSpace(line))
104 | {
105 | if(treatErrorsAsInfo)
106 | logger.LogInformation(StripAnsiColors(line));
107 | else
108 | logger.LogError(StripAnsiColors(line));
109 | }
110 | };
111 |
112 | // But when it emits incomplete lines, assume this is progress information and
113 | // hence just pass it through to StdOut regardless of logger config.
114 | StdErr.OnReceivedChunk += chunk =>
115 | {
116 | Debug.Assert(chunk.Array != null);
117 |
118 | var containsNewline = Array.IndexOf(
119 | chunk.Array, '\n', chunk.Offset, chunk.Count) >= 0;
120 | if (!containsNewline)
121 | {
122 | Console.Write(chunk.Array, chunk.Offset, chunk.Count);
123 | }
124 | };
125 | }
126 |
127 | private static string StripAnsiColors(string line)
128 | => AnsiColorRegex.Replace(line, string.Empty);
129 |
130 | private static Process LaunchNodeProcess(ProcessStartInfo startInfo, string commandName)
131 | {
132 | try
133 | {
134 | var process = Process.Start(startInfo)!;
135 |
136 | // See equivalent comment in OutOfProcessNodeInstance.cs for why
137 | process.EnableRaisingEvents = true;
138 |
139 | return process;
140 | }
141 | catch (Exception ex)
142 | {
143 | var message = $"Failed to start '{commandName}'. To resolve this:.\n\n"
144 | + $"[1] Ensure that '{commandName}' is installed and can be found in one of the PATH directories.\n"
145 | + $" Current PATH enviroment variable is: { Environment.GetEnvironmentVariable("PATH") }\n"
146 | + " Make sure the executable is in one of those directories, or update your PATH.\n\n"
147 | + "[2] See the InnerException for further details of the cause.";
148 | throw new InvalidOperationException(message, ex);
149 | }
150 | }
151 |
152 | void IDisposable.Dispose()
153 | {
154 | if (_npmProcess != null && !_npmProcess.HasExited)
155 | {
156 | _npmProcess.Kill(entireProcessTree: true);
157 | _npmProcess = null;
158 | }
159 | }
160 | }
161 |
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/Tailwind.Extensions.AspNetCore.csproj:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | net6.0
5 | enable
6 | enable
7 | Tailwind extensions for ASP.NET Core
8 | Tailwind
9 | Tailwind.Extensions.AspNetCore
10 | Tailwind.Extensions.AspNetCore
11 | tailwind;devtools;
12 | latest
13 | v
14 | https://github.com/Practical-ASP-NET/Tailwind.Extensions.AspNetCore
15 | https://github.com/Practical-ASP-NET/Tailwind.Extensions.AspNetCore/blob/main/LICENSE
16 | https://github.com/Practical-ASP-NET/Tailwind.Extensions.AspNetCore
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/TailwindHostedService.cs:
--------------------------------------------------------------------------------
1 | using System.Diagnostics;
2 | using Microsoft.Extensions.Configuration;
3 | using Microsoft.Extensions.Hosting;
4 | using Microsoft.Extensions.Logging;
5 | using Microsoft.Extensions.Options;
6 |
7 | namespace Tailwind;
8 |
9 | public class TailwindOptions
10 | {
11 | public string? InputFile { get; set; }
12 | public string? OutputFile { get; set; }
13 | public string? TailwindCliPath { get; set; }
14 | }
15 |
16 | ///
17 | /// Launches the Tailwind CLI in watch mode (if the CLI is installed and exists in the system Path)
18 | /// Runs as a background service (will be shut down when the app is stopped)
19 | /// Only works in development environments
20 | ///
21 | public class TailwindHostedService : IHostedService, IDisposable
22 | {
23 | private Process? _process;
24 | private readonly IHostEnvironment _hostEnvironment;
25 | private readonly TailwindOptions _options;
26 | private readonly ITailwindProcessInterop _tailwindProcess;
27 | private readonly ILogger _logger;
28 |
29 | public TailwindHostedService(IOptions options, IHostEnvironment hostEnvironment,
30 | ITailwindProcessInterop tailwindProcess, ILogger logger)
31 | {
32 | _logger = logger;
33 | _tailwindProcess = tailwindProcess;
34 | _options = options.Value;
35 | _hostEnvironment = hostEnvironment;
36 | }
37 |
38 | public Task StartAsync(CancellationToken cancellationToken)
39 | {
40 | // don't want this running in production, only useful for development time generation of the Tailwind file
41 | // use another mechanism such as CI/CD to generate the production-ready CSS style sheet
42 | if (!_hostEnvironment.IsDevelopment())
43 | return Task.CompletedTask;
44 |
45 |
46 | var input = _options.InputFile;
47 | var output = _options.OutputFile;
48 |
49 | Guard.AgainstNull(input, "check Tailwind configuration");
50 | Guard.AgainstNull(output, "check Tailwind configuration");
51 |
52 | _logger.LogInformation($"tailwind -i {input} -o {output} --watch");
53 |
54 | var processName = string.IsNullOrEmpty(_options.TailwindCliPath) ? "tailwind" : _options.TailwindCliPath;
55 | _process = _tailwindProcess.StartProcess(processName, $"-i {input} -o {output} --watch");
56 |
57 | return Task.CompletedTask;
58 | }
59 |
60 | public Task StopAsync(CancellationToken cancellationToken)
61 | {
62 | _process?.Kill();
63 | return Task.CompletedTask;
64 | }
65 |
66 | public void Dispose()
67 | {
68 | _process?.Dispose();
69 | }
70 | }
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/TailwindMiddleware.cs:
--------------------------------------------------------------------------------
1 | // Licensed to the .NET Foundation under one or more agreements.
2 | // The .NET Foundation licenses this file to you under the MIT license.
3 |
4 | using System.Diagnostics;
5 | using System.Text.RegularExpressions;
6 | using Microsoft.AspNetCore.Builder;
7 | using Microsoft.AspNetCore.NodeServices.Npm;
8 | using Microsoft.AspNetCore.NodeServices.Util;
9 | using Microsoft.AspNetCore.SpaServices.Util;
10 | using Microsoft.Extensions.DependencyInjection;
11 | using Microsoft.Extensions.Hosting;
12 | using Microsoft.Extensions.Logging;
13 |
14 |
15 | namespace Tailwind;
16 |
17 | internal static class TailwindMiddleware
18 | {
19 | private const string LogCategoryName = "NodeServices";
20 |
21 | private static readonly TimeSpan
22 | RegexMatchTimeout =
23 | TimeSpan.FromSeconds(5); // This is a development-time only feature, so a very long timeout is fine
24 |
25 | public static async Task Attach(IApplicationBuilder appBuilder, string scriptName, string workingDir)
26 | {
27 | var applicationStoppingToken = appBuilder.ApplicationServices.GetRequiredService()
28 | .ApplicationStopping;
29 | var logger = LoggerFinder.GetOrCreateLogger(appBuilder, LogCategoryName);
30 | var diagnosticSource = appBuilder.ApplicationServices.GetRequiredService();
31 | await ExecuteScript(workingDir, scriptName, logger, diagnosticSource, applicationStoppingToken);
32 | }
33 |
34 | private static async Task ExecuteScript(
35 | string sourcePath, string scriptName, ILogger logger,
36 | DiagnosticSource diagnosticSource, CancellationToken applicationStoppingToken)
37 | {
38 | var envVars = new Dictionary() { };
39 |
40 | var scriptRunner = new NodeScriptRunner(
41 | sourcePath, scriptName, null, envVars, "npm", diagnosticSource, applicationStoppingToken);
42 | scriptRunner.AttachToLogger(logger, true);
43 |
44 | using var stdErrReader = new EventedStreamStringReader(scriptRunner.StdErr);
45 | try
46 | {
47 | await scriptRunner.StdOut.WaitForMatch(
48 | new Regex("Done in", RegexOptions.None, RegexMatchTimeout));
49 | }
50 | catch (EndOfStreamException ex)
51 | {
52 | throw new InvalidOperationException(
53 | $"The npm script '{scriptName}' exited without indicating that the " +
54 | "Tailwind CLI had finished. The error output was: " +
55 | $"{stdErrReader.ReadAsString()}", ex);
56 | }
57 | }
58 | }
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/TailwindMiddlewareExtensions.cs:
--------------------------------------------------------------------------------
1 | using Microsoft.AspNetCore.Builder;
2 | using Microsoft.Extensions.DependencyInjection;
3 |
4 | namespace Tailwind;
5 |
6 | ///
7 | /// Extension methods for enabling tailwind middleware support.
8 | ///
9 | public static class TailwindMiddlewareExtensions
10 | {
11 | ///
12 | /// Automatically runs an npm script
13 | /// This feature should probably only be used in development.
14 | ///
15 | /// The .
16 | /// The name of the script in your package.json file that launches Tailwind.
17 | /// The directory where your package.json file resides
18 | public static Task RunTailwind(
19 | this IApplicationBuilder applicationBuilder,
20 | string npmScript, string workingDir = "./")
21 | {
22 | if (applicationBuilder == null)
23 | {
24 | throw new ArgumentNullException(nameof(applicationBuilder));
25 | }
26 |
27 | return TailwindMiddleware.Attach(applicationBuilder, npmScript, workingDir);
28 | }
29 |
30 | ///
31 | /// Run the Tailwind Cli in watch mode in the background (Development environments only
32 | ///
33 | ///
34 | public static void UseTailwindCli(this WebApplicationBuilder webApplicationBuilder)
35 | {
36 | webApplicationBuilder.Services.Configure(webApplicationBuilder.Configuration.GetSection("Tailwind"));
37 | webApplicationBuilder.Services.AddSingleton();
38 | webApplicationBuilder.Services.AddHostedService();
39 | }
40 | }
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/TailwindProcessInterop.cs:
--------------------------------------------------------------------------------
1 | using System.Diagnostics;
2 | using System.Runtime.InteropServices;
3 |
4 | namespace Tailwind;
5 |
6 | public interface ITailwindProcessInterop
7 | {
8 | Process? StartProcess(string command, string arguments);
9 | }
10 |
11 | public class TailwindProcessInterop : ITailwindProcessInterop
12 | {
13 | public Process? StartProcess(string command, string arguments)
14 | {
15 | string cmdName;
16 | if (!command.EndsWith(".exe") && RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
17 | {
18 | cmdName = $"{command}.exe";
19 | }
20 | else
21 | cmdName = command;
22 |
23 | try
24 | {
25 | var startInfo = new ProcessStartInfo
26 | {
27 | FileName = cmdName,
28 | Arguments = arguments,
29 | RedirectStandardOutput = true,
30 | RedirectStandardError = true,
31 | UseShellExecute = false,
32 | CreateNoWindow = true
33 | };
34 |
35 | var process = Process.Start(startInfo);
36 |
37 | if (process == null)
38 | {
39 | Console.WriteLine($"Could not start process: {cmdName}");
40 | return null;
41 | }
42 |
43 | process.OutputDataReceived += (sender, data) =>
44 | {
45 | if (!string.IsNullOrEmpty(data.Data))
46 | {
47 | Console.WriteLine("tailwind: " + data.Data);
48 | }
49 | };
50 |
51 | process.ErrorDataReceived += (sender, data) =>
52 | {
53 | if (!string.IsNullOrEmpty(data.Data))
54 | {
55 | Console.WriteLine("tailwind: " +data.Data);
56 | }
57 | };
58 |
59 | process.BeginOutputReadLine();
60 | process.BeginErrorReadLine();
61 |
62 | return process;
63 | }
64 | catch (Exception ex)
65 | {
66 | Console.WriteLine($"Error starting process: {ex.Message}");
67 | throw;
68 | }
69 | }
70 | }
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/Util/EventedStreamReader.cs:
--------------------------------------------------------------------------------
1 | // Licensed to the .NET Foundation under one or more agreements.
2 | // The .NET Foundation licenses this file to you under the MIT license.
3 |
4 | using System.Text;
5 | using System.Text.RegularExpressions;
6 |
7 | namespace Microsoft.AspNetCore.NodeServices.Util;
8 |
9 | ///
10 | /// Wraps a to expose an evented API, issuing notifications
11 | /// when the stream emits partial lines, completed lines, or finally closes.
12 | ///
13 | internal class EventedStreamReader
14 | {
15 | public delegate void OnReceivedChunkHandler(ArraySegment chunk);
16 | public delegate void OnReceivedLineHandler(string line);
17 | public delegate void OnStreamClosedHandler();
18 |
19 | public event OnReceivedChunkHandler? OnReceivedChunk;
20 | public event OnReceivedLineHandler? OnReceivedLine;
21 | public event OnStreamClosedHandler? OnStreamClosed;
22 |
23 | private readonly StreamReader _streamReader;
24 | private readonly StringBuilder _linesBuffer;
25 |
26 | public EventedStreamReader(StreamReader streamReader)
27 | {
28 | _streamReader = streamReader ?? throw new ArgumentNullException(nameof(streamReader));
29 | _linesBuffer = new StringBuilder();
30 | Task.Factory.StartNew(Run, CancellationToken.None, TaskCreationOptions.DenyChildAttach, TaskScheduler.Default);
31 | }
32 |
33 | public Task WaitForMatch(Regex regex)
34 | {
35 | var tcs = new TaskCompletionSource();
36 | var completionLock = new object();
37 |
38 | OnReceivedLineHandler? onReceivedLineHandler = null;
39 | OnStreamClosedHandler? onStreamClosedHandler = null;
40 |
41 | void ResolveIfStillPending(Action applyResolution)
42 | {
43 | lock (completionLock)
44 | {
45 | if (!tcs.Task.IsCompleted)
46 | {
47 | OnReceivedLine -= onReceivedLineHandler;
48 | OnStreamClosed -= onStreamClosedHandler;
49 | applyResolution();
50 | }
51 | }
52 | }
53 |
54 | onReceivedLineHandler = line =>
55 | {
56 | var match = regex.Match(line);
57 | if (match.Success)
58 | {
59 | ResolveIfStillPending(() => tcs.SetResult(match));
60 | }
61 | };
62 |
63 | onStreamClosedHandler = () =>
64 | {
65 | ResolveIfStillPending(() => tcs.SetException(new EndOfStreamException()));
66 | };
67 |
68 | OnReceivedLine += onReceivedLineHandler;
69 | OnStreamClosed += onStreamClosedHandler;
70 |
71 | return tcs.Task;
72 | }
73 |
74 | private async Task Run()
75 | {
76 | var buf = new char[8 * 1024];
77 | while (true)
78 | {
79 | var chunkLength = await _streamReader.ReadAsync(buf, 0, buf.Length);
80 | if (chunkLength == 0)
81 | {
82 | if (_linesBuffer.Length > 0)
83 | {
84 | OnCompleteLine(_linesBuffer.ToString());
85 | _linesBuffer.Clear();
86 | }
87 |
88 | OnClosed();
89 | break;
90 | }
91 |
92 | OnChunk(new ArraySegment(buf, 0, chunkLength));
93 |
94 | int lineBreakPos;
95 | var startPos = 0;
96 |
97 | // get all the newlines
98 | while ((lineBreakPos = Array.IndexOf(buf, '\n', startPos, chunkLength - startPos)) >= 0 && startPos < chunkLength)
99 | {
100 | var length = (lineBreakPos + 1) - startPos;
101 | _linesBuffer.Append(buf, startPos, length);
102 | OnCompleteLine(_linesBuffer.ToString());
103 | _linesBuffer.Clear();
104 | startPos = lineBreakPos + 1;
105 | }
106 |
107 | // get the rest
108 | if (lineBreakPos < 0 && startPos < chunkLength)
109 | {
110 | _linesBuffer.Append(buf, startPos, chunkLength - startPos);
111 | }
112 | }
113 | }
114 |
115 | private void OnChunk(ArraySegment chunk)
116 | {
117 | var dlg = OnReceivedChunk;
118 | dlg?.Invoke(chunk);
119 | }
120 |
121 | private void OnCompleteLine(string line)
122 | {
123 | var dlg = OnReceivedLine;
124 | dlg?.Invoke(line);
125 | }
126 |
127 | private void OnClosed()
128 | {
129 | var dlg = OnStreamClosed;
130 | dlg?.Invoke();
131 | }
132 | }
133 |
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/Util/EventedStreamStringReader.cs:
--------------------------------------------------------------------------------
1 | // Licensed to the .NET Foundation under one or more agreements.
2 | // The .NET Foundation licenses this file to you under the MIT license.
3 |
4 | using System.Text;
5 |
6 | namespace Microsoft.AspNetCore.NodeServices.Util;
7 |
8 | ///
9 | /// Captures the completed-line notifications from a ,
10 | /// combining the data into a single .
11 | ///
12 | internal class EventedStreamStringReader : IDisposable
13 | {
14 | private readonly EventedStreamReader _eventedStreamReader;
15 | private bool _isDisposed;
16 | private readonly StringBuilder _stringBuilder = new StringBuilder();
17 |
18 | public EventedStreamStringReader(EventedStreamReader eventedStreamReader)
19 | {
20 | _eventedStreamReader = eventedStreamReader
21 | ?? throw new ArgumentNullException(nameof(eventedStreamReader));
22 | _eventedStreamReader.OnReceivedLine += OnReceivedLine;
23 | }
24 |
25 | public string ReadAsString() => _stringBuilder.ToString();
26 |
27 | private void OnReceivedLine(string line) => _stringBuilder.AppendLine(line);
28 |
29 | public void Dispose()
30 | {
31 | if (!_isDisposed)
32 | {
33 | _eventedStreamReader.OnReceivedLine -= OnReceivedLine;
34 | _isDisposed = true;
35 | }
36 | }
37 | }
38 |
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/Util/LoggerFinder.cs:
--------------------------------------------------------------------------------
1 | // Licensed to the .NET Foundation under one or more agreements.
2 | // The .NET Foundation licenses this file to you under the MIT license.
3 |
4 | using Microsoft.AspNetCore.Builder;
5 | using Microsoft.Extensions.DependencyInjection;
6 | using Microsoft.Extensions.Logging;
7 | using Microsoft.Extensions.Logging.Abstractions;
8 |
9 | namespace Microsoft.AspNetCore.SpaServices.Util;
10 |
11 | internal static class LoggerFinder
12 | {
13 | public static ILogger GetOrCreateLogger(
14 | IApplicationBuilder appBuilder,
15 | string logCategoryName)
16 | {
17 | // If the DI system gives us a logger, use it. Otherwise, set up a default one
18 | var loggerFactory = appBuilder.ApplicationServices.GetService();
19 | var logger = loggerFactory != null
20 | ? loggerFactory.CreateLogger(logCategoryName)
21 | : NullLogger.Instance;
22 | return logger;
23 | }
24 | }
25 |
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/Util/TaskTimeoutExtensions.cs:
--------------------------------------------------------------------------------
1 | // Licensed to the .NET Foundation under one or more agreements.
2 | // The .NET Foundation licenses this file to you under the MIT license.
3 |
4 | namespace Microsoft.AspNetCore.SpaServices.Extensions.Util;
5 |
6 | internal static class TaskTimeoutExtensions
7 | {
8 | public static async Task WithTimeout(this Task task, TimeSpan timeoutDelay, string message)
9 | {
10 | if (task == await Task.WhenAny(task, Task.Delay(timeoutDelay)))
11 | {
12 | task.Wait(); // Allow any errors to propagate
13 | }
14 | else
15 | {
16 | throw new TimeoutException(message);
17 | }
18 | }
19 |
20 | public static async Task WithTimeout(this Task task, TimeSpan timeoutDelay, string message)
21 | {
22 | if (task == await Task.WhenAny(task, Task.Delay(timeoutDelay)))
23 | {
24 | return task.Result;
25 | }
26 | else
27 | {
28 | throw new TimeoutException(message);
29 | }
30 | }
31 | }
32 |
--------------------------------------------------------------------------------
/src/Tailwind.Extensions.AspNetCore/Util/TcpPortFinder.cs:
--------------------------------------------------------------------------------
1 | // Licensed to the .NET Foundation under one or more agreements.
2 | // The .NET Foundation licenses this file to you under the MIT license.
3 |
4 | using System.Net;
5 | using System.Net.Sockets;
6 |
7 | namespace Microsoft.AspNetCore.SpaServices.Util;
8 |
9 | internal static class TcpPortFinder
10 | {
11 | public static int FindAvailablePort()
12 | {
13 | var listener = new TcpListener(IPAddress.Loopback, 0);
14 | listener.Start();
15 | try
16 | {
17 | return ((IPEndPoint)listener.LocalEndpoint).Port;
18 | }
19 | finally
20 | {
21 | listener.Stop();
22 | }
23 | }
24 | }
25 |
--------------------------------------------------------------------------------