Skip to content

Commit d25882b

Browse files
Merge pull request #36 from bingzer/main
Adding hx-headers-* server-side processing to hx-headers
2 parents 6b91f41 + 11d3851 commit d25882b

File tree

6 files changed

+222
-0
lines changed

6 files changed

+222
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
using JetBrains.Annotations;
2+
using Microsoft.AspNetCore.Razor.TagHelpers;
3+
using System.Collections.Generic;
4+
using System;
5+
using System.Text.Json;
6+
using System.Linq;
7+
8+
namespace Htmx.TagHelpers
9+
{
10+
/// <summary>
11+
/// Targets any element that has hx-get, hx-post, hx-put, hx-patch, and hx-delete.
12+
/// https://htmx.org/attributes/hx-headers/
13+
/// </summary>
14+
[PublicAPI]
15+
[HtmlTargetElement("*", Attributes = "[hx-get]")]
16+
[HtmlTargetElement("*", Attributes = "[hx-post]")]
17+
[HtmlTargetElement("*", Attributes = "[hx-put]")]
18+
[HtmlTargetElement("*", Attributes = "[hx-delete]")]
19+
[HtmlTargetElement("*", Attributes = "[hx-patch]")]
20+
public class HtmxHeadersTagHelper : TagHelper
21+
{
22+
/// <summary>
23+
/// Dictionary of hx-headers
24+
/// </summary>
25+
[HtmlAttributeName(DictionaryAttributePrefix = "hx-headers-")]
26+
public IDictionary<string, string?> HeaderAttributes { get; set; } = new Dictionary<string, string?>(StringComparer.OrdinalIgnoreCase);
27+
28+
/// <inheritdoc />
29+
public override void Process(TagHelperContext context, TagHelperOutput output)
30+
{
31+
var existingHxHeaders = output.Attributes["hx-headers"]?.Value;
32+
if (existingHxHeaders != null || !HeaderAttributes.Any())
33+
{
34+
return;
35+
}
36+
37+
// serialize
38+
var json = JsonSerializer.Serialize(HeaderAttributes);
39+
40+
output.Attributes.Add("hx-headers", json);
41+
}
42+
}
43+
}

test/Htmx.Tests/Htmx.Tests.csproj

+1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
</ItemGroup>
2626

2727
<ItemGroup>
28+
<ProjectReference Include="..\..\src\Htmx.TagHelpers\Htmx.TagHelpers.csproj" />
2829
<ProjectReference Include="..\..\src\Htmx\Htmx.csproj" />
2930
</ItemGroup>
3031

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
using Htmx.TagHelpers;
2+
using Microsoft.AspNetCore.Mvc.Rendering;
3+
using Microsoft.AspNetCore.Razor.TagHelpers;
4+
using System.Collections.Generic;
5+
using System.Text.Json;
6+
using System.Text.Json.Nodes;
7+
using System.Threading.Tasks;
8+
using Xunit;
9+
10+
namespace Htmx.Tests.TagHelpers
11+
{
12+
public class HtmxHeadersTagHelperTests
13+
{
14+
private readonly HtmxHeadersTagHelper _tagHelper;
15+
private readonly TagHelperContext _context;
16+
private readonly TagHelperOutput _output;
17+
18+
public HtmxHeadersTagHelperTests()
19+
{
20+
_tagHelper = new HtmxHeadersTagHelper();
21+
22+
_context = new TagHelperContext(
23+
new TagHelperAttributeList(),
24+
new Dictionary<object, object>(),
25+
"test_unique_id");
26+
27+
_output = new TagHelperOutput(
28+
"div",
29+
new TagHelperAttributeList(),
30+
(useCachedResult, encoder) =>
31+
{
32+
var tagHelperContent = new DefaultTagHelperContent();
33+
var tagBuilder = new TagBuilder("div");
34+
tagBuilder.Attributes.Add("hx-post", "url");
35+
tagHelperContent.SetHtmlContent(tagBuilder);
36+
return Task.FromResult<TagHelperContent>(tagHelperContent);
37+
});
38+
}
39+
40+
[Fact]
41+
public void Is_HeaderAttributes_NotNull()
42+
{
43+
Assert.NotNull(_tagHelper.HeaderAttributes);
44+
Assert.Empty(_tagHelper.HeaderAttributes);
45+
}
46+
47+
[Fact]
48+
public void Process_AddsHeadersAttribute_WhenNoExistingHeadersAndHeaderAttributesPresent()
49+
{
50+
// Arrange
51+
_tagHelper.HeaderAttributes.Add("Key1", "Value1");
52+
_tagHelper.HeaderAttributes.Add("Key2", "Value2");
53+
54+
// Act
55+
_tagHelper.Process(_context, _output);
56+
57+
// Assert
58+
Assert.True(_output.Attributes.TryGetAttribute("hx-headers", out var headersAttribute));
59+
Assert.NotNull(headersAttribute);
60+
61+
var json = JsonSerializer.Deserialize<JsonObject>(headersAttribute.Value.ToString()!)!;
62+
Assert.Equal("Value1", json["Key1"]!.GetValue<string>());
63+
Assert.Equal("Value2", json["Key2"]!.GetValue<string>());
64+
}
65+
66+
[Fact]
67+
public void Process_DoesNotAddHeadersAttribute_WhenExistingHeadersPresent()
68+
{
69+
// Arrange
70+
var existingHeaders = new TagHelperAttribute("hx-headers", "{\"ExistingKey1\":\"ExistingValue1\"}");
71+
_output.Attributes.Add(existingHeaders);
72+
73+
// Act
74+
_tagHelper.Process(_context, _output);
75+
76+
// Assert
77+
var json = JsonSerializer.Deserialize<JsonObject>(_output.Attributes["hx-headers"].Value.ToString()!)!;
78+
Assert.Equal("ExistingValue1", json["ExistingKey1"]!.GetValue<string>());
79+
}
80+
81+
[Fact]
82+
public void Process_DoesNotAddHeadersAttribute_WhenNoHeaderAttributes()
83+
{
84+
// Act
85+
_tagHelper.Process(_context, _output);
86+
87+
// Assert
88+
Assert.False(_output.Attributes.TryGetAttribute("hx-headers", out _));
89+
}
90+
}
91+
92+
}

test/Sample/Pages/Headers.cshtml

+63
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
@page
2+
@using Microsoft.AspNetCore.Antiforgery;
3+
@inject IAntiforgery antiforgery;
4+
@model HxRequestsModel
5+
6+
<div>
7+
<button type="button"
8+
hx-post
9+
hx-page="./Headers"
10+
hx-headers-New-Key1="NewValue1"
11+
hx-headers-New-Key2="NewValue2"
12+
hx-swap="innerHTML"
13+
hx-target="#post-result">
14+
Click me (using hx-headers-key="value")
15+
</button>
16+
</div>
17+
18+
<div>
19+
<button type="button"
20+
hx-post
21+
hx-page="./Headers"
22+
hx-headers='{ "Existing-Key1": "ExistingValue1", "Existing-Key2": "ExistingValue1" }'
23+
hx-headers-New-Key1="NewValue1"
24+
hx-headers-New-Key2="NewValue2"
25+
hx-swap="innerHTML"
26+
hx-target="#post-result">
27+
Click me (with existing hx-headers)
28+
</button>
29+
</div>
30+
31+
<div>
32+
Testing: https://en.wikipedia.org/wiki/List_of_HTTP_header_fields
33+
<br/>
34+
<button type="button"
35+
hx-post
36+
hx-page="./Headers"
37+
hx-headers-X-Forwarded-Host="en.wikipedia.org:8080"
38+
hx-headers-X-Front-End-Https="on"
39+
hx-headers-X-Forwarded-For="client1, proxy1, proxy2"
40+
hx-headers-Upgrade-Insecure-Requests="1"
41+
hx-headers-X-LongCustom-Header="Stuff"
42+
hx-swap="innerHTML"
43+
hx-target="#post-result">
44+
Click me (List_of_HTTP_header_fields)
45+
</button>
46+
</div>
47+
48+
<div class="mt-2">
49+
<code>requestConfig.headers</code>
50+
<textarea class="w-100" id="post-result" rows="10"></textarea>
51+
</div>
52+
53+
@section Scripts {
54+
<script type="text/javascript">
55+
window.addEventListener("htmx:afterRequest", evt => {
56+
let headers = evt.detail.requestConfig.headers;
57+
let json = JSON.stringify(headers, null, 2);
58+
59+
document.querySelector('#post-result').innerHTML = json;
60+
});
61+
</script>
62+
}
63+

test/Sample/Pages/Headers.cshtml.cs

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
using Htmx;
2+
using Microsoft.AspNetCore.Mvc;
3+
using Microsoft.AspNetCore.Mvc.RazorPages;
4+
using System.Text.Json;
5+
6+
namespace Sample.Pages
7+
{
8+
[ValidateAntiForgeryToken]
9+
public class HxRequestsModel : PageModel
10+
{
11+
public IActionResult OnPost()
12+
{
13+
// list of headers
14+
var headers = Request.Headers.ToList();
15+
var html = "<pre>" + JsonSerializer.Serialize(headers, new JsonSerializerOptions { WriteIndented = true }) + "</pre>";
16+
17+
return Content(html, "text/html");
18+
}
19+
}
20+
}

test/Sample/Pages/Shared/_Layout.cshtml

+3
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,9 @@
3737
<li class="nav-item">
3838
<a class="nav-link text-dark" asp-area="" asp-page="/Response">Response.Htmx()</a>
3939
</li>
40+
<li class="nav-item">
41+
<a class="nav-link text-dark" asp-area="" asp-page="/Headers"><code>hx-headers</code></a>
42+
</li>
4043
<li class="nav-item">
4144
<a class="nav-link text-dark" asp-area="" asp-page="/Privacy">Privacy</a>
4245
</li>

0 commit comments

Comments
 (0)