Skip to content

Commit 95d2764

Browse files
Merge pull request #34 from JalexSocial/mat-corspolicy-builder-extension
Htmx.Net doesn't work in a cross origin context without a CORS policy (Fixes #33)
2 parents 05faa28 + 305fdb8 commit 95d2764

File tree

3 files changed

+52
-0
lines changed

3 files changed

+52
-0
lines changed

Readme.md

+42
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,22 @@ Request.IsHtmx(out var values);
3939

4040
Read more about the other header values on the [official documentation page](https://htmx.org/reference/#request_headers).
4141

42+
#### Browser Caching
43+
44+
As a special note, please be mindful that if your server can render different content for the same URL depending on some other headers, you need to use the Vary response HTTP header. For example, if your server renders the full HTML when Request.IsHtmx() is false, and it renders a fragment of that HTML when Request.IsHtmx() is true, you need to add Vary: HX-Request. That causes the cache to be keyed based on a composite of the response URL and the HX-Request request header — rather than being based just on the response URL.
45+
46+
```c#
47+
// in a Razor Page
48+
if (Request.IsHtmx())
49+
{
50+
Response.Headers.Add("Vary", "HX-Request");
51+
return Partial("_Form", this)
52+
}
53+
54+
return Page();
55+
```
56+
57+
4258
### HttpResponse
4359

4460
We can set Http Response headers using the `Htmx` extension method, which passes an action and `HtmxResponseHeaders` object.
@@ -64,6 +80,32 @@ Response.Htmx(h => {
6480
});
6581
```
6682

83+
#### CORS Policy
84+
85+
By default, all Htmx requests and responses will be blocked in a cross-origin context.
86+
87+
If you configure your application in a cross-origin context, then setting a CORS policy in ASP.NET Core also allows you to define specific restrictions on request and response headers,
88+
enabling fine-grained control over the data that can be exchanged between your web application and different origins.
89+
90+
This library provides a simple approach to exposing Htmx headers to your CORS policy:
91+
92+
```c#
93+
var MyAllowSpecificOrigins = "_myAllowSpecificOrigins";
94+
95+
var builder = WebApplication.CreateBuilder(args);
96+
97+
builder.Services.AddCors(options =>
98+
{
99+
options.AddPolicy(name: MyAllowSpecificOrigins,
100+
policy =>
101+
{
102+
policy.WithOrigins("http://example.com", "http://www.contoso.com")
103+
.WithHeaders(HtmxRequestHeaders.Keys.All) // Add htmx request headers
104+
.WithExposedHeaders(HtmxResponseHeaders.Keys.All) // Add htmx response headers
105+
});
106+
});
107+
```
108+
67109
## Htmx.TagHelpers
68110

69111
### Getting Started

src/Htmx/HtmxRequestHeaders.cs

+5
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,11 @@ public static class Keys
2020
public const string TriggerName = "HX-Trigger-Name";
2121
public const string Trigger = "HX-Trigger";
2222
public const string Boosted = "HX-Boosted";
23+
24+
public static string[] All { get; } = new[]
25+
{
26+
CurrentUrl, HistoryRestoreRequest, Prompt, Request, Target, TriggerName, Trigger, Boosted
27+
};
2328
}
2429

2530
public HtmxRequestHeaders(HttpRequest request)

src/Htmx/HtmxResponseHeaders.cs

+5
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,11 @@ public static class Keys
3232
public const string Reswap = "HX-Reswap";
3333
public const string Retarget = "HX-Retarget";
3434
public const string ReplaceUrl = "HX-Replace-Url";
35+
36+
public static string[] All { get; } = new[]
37+
{
38+
PushUrl, Location, Redirect, Refresh, Trigger, TriggerAfterSettle, TriggerAfterSwap, Reswap, Retarget, ReplaceUrl
39+
};
3540
}
3641

3742
internal HtmxResponseHeaders(IHeaderDictionary headers)

0 commit comments

Comments
 (0)