Skip to content

Commit 05faa28

Browse files
Merge pull request #32 from khalidabuhakmeh/mat-allow-existing-triggers
Allow existing triggers to be merged in with new trigger definitions
2 parents 783286e + 321f5f7 commit 05faa28

File tree

2 files changed

+77
-24
lines changed

2 files changed

+77
-24
lines changed

src/Htmx/HtmxResponseHeaders.cs

+43-13
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,9 @@
11
using System;
22
using System.Collections.Generic;
33
using System.Linq;
4+
using System.Text;
45
using System.Text.Json;
6+
using System.Text.Json.Nodes;
57
using JetBrains.Annotations;
68
using Microsoft.AspNetCore.Http;
79

@@ -183,7 +185,7 @@ public HtmxResponseHeaders WithTrigger(string eventName, object? detail = null,
183185
}
184186
else
185187
{
186-
_triggers[timing].Add(eventName, detail ?? string.Empty);
188+
_triggers[timing].TryAdd(eventName, detail ?? string.Empty);
187189
}
188190

189191
return this;
@@ -196,38 +198,66 @@ internal HtmxResponseHeaders Process()
196198
{
197199
if (_triggers.ContainsKey(HtmxTriggerTiming.Default))
198200
{
199-
if (_headers.ContainsKey(Keys.Trigger))
200-
{
201-
throw new Exception("You must use either WithTrigger(..) or Trigger(..), but not both.");
202-
}
201+
ParsePossibleExistingTriggers(Keys.Trigger, HtmxTriggerTiming.Default);
203202

204203
_headers[Keys.Trigger] = BuildTriggerHeader(HtmxTriggerTiming.Default);
205204
}
206205

207206
if (_triggers.ContainsKey(HtmxTriggerTiming.AfterSettle))
208207
{
209-
if (_headers.ContainsKey(Keys.TriggerAfterSettle))
210-
{
211-
throw new Exception("You must use either WithTrigger(..) or TriggerAfterSettle(..), but not both.");
212-
}
208+
ParsePossibleExistingTriggers(Keys.TriggerAfterSettle, HtmxTriggerTiming.AfterSettle);
213209

214210
_headers[Keys.TriggerAfterSettle] = BuildTriggerHeader(HtmxTriggerTiming.AfterSettle);
215211
}
216212

217213
// ReSharper disable once InvertIf
218214
if (_triggers.ContainsKey(HtmxTriggerTiming.AfterSwap))
219215
{
220-
if (_headers.ContainsKey(Keys.TriggerAfterSwap))
221-
{
222-
throw new Exception("You must use either WithTrigger(..) or TriggerAfterSwap(..), but not both.");
223-
}
216+
ParsePossibleExistingTriggers(Keys.TriggerAfterSwap, HtmxTriggerTiming.AfterSwap);
224217

225218
_headers[Keys.TriggerAfterSwap] = BuildTriggerHeader(HtmxTriggerTiming.AfterSwap);
226219
}
227220

228221
return this;
229222
}
230223

224+
/// <summary>
225+
/// Checks to see if the response has an existing header defined by headerKey. If it does the
226+
/// header loads all of the triggers locally so they aren't overwritten by Htmx.
227+
/// </summary>
228+
/// <param name="headerKey"></param>
229+
/// <param name="timing"></param>
230+
private void ParsePossibleExistingTriggers(string headerKey, HtmxTriggerTiming timing)
231+
{
232+
if (!_headers.ContainsKey(headerKey))
233+
return;
234+
235+
var header = _headers[headerKey];
236+
// Attempt to parse existing header as Json, if fails it is a simplified event key
237+
// assume if the string starts with '{' and ends with '}', that it is JSON
238+
if (header.Any(h => h is ['{', .., '}']))
239+
{
240+
var reader = new Utf8JsonReader(Encoding.UTF8.GetBytes(header));
241+
// this might still throw :(
242+
var jsonObject = JsonNode.Parse(ref reader)?.AsObject();
243+
// Load any existing triggers
244+
foreach (var (key, value) in jsonObject!)
245+
WithTrigger(key, value, timing);
246+
}
247+
else
248+
{
249+
foreach (var headerValue in _headers[headerKey])
250+
{
251+
if (headerValue is null) continue;
252+
253+
var eventNames = headerValue.Split(',');
254+
255+
foreach (var eventName in eventNames)
256+
WithTrigger(eventName, null, timing);
257+
}
258+
}
259+
}
260+
231261
private string BuildTriggerHeader(HtmxTriggerTiming timing)
232262
{
233263
// Reduce the payload if the user has only specified 1 trigger with no value

test/Htmx.Tests/HtmxHttpResponseExtensionsTests.cs

+34-11
Original file line numberDiff line numberDiff line change
@@ -122,24 +122,47 @@ public void Can_add_multiple_triggers_with_detail()
122122
}
123123

124124
[Fact]
125-
public void Cant_use_legacy_trigger_and_with_trigger()
125+
public void Can_use_existing_trigger_with_trigger()
126126
{
127-
Response.Htmx(h => h.Trigger("cool"));
128-
Assert.Throws<Exception>(() => Response.Htmx(h => h.WithTrigger("neat")));
127+
const string expected = @"{""neat"":"""",""cool"":""""}";
128+
129+
Response.Htmx(h => h.WithTrigger("cool"));
130+
Response.Htmx(h => h.WithTrigger("neat"));
131+
132+
Assert.True(Headers.ContainsKey(Keys.Trigger));
133+
Assert.Equal(expected, Headers[Keys.Trigger]);
129134
}
130-
135+
131136
[Fact]
132-
public void Cant_use_legacy_triggeraftersettle_and_with_trigger()
137+
public void Can_use_existing_trigger_with_multiple_triggers_with_detail()
133138
{
134-
Response.Htmx(h => h.TriggerAfterSettle("cool"));
135-
Assert.Throws<Exception>(() => Response.Htmx(h => h.WithTrigger("neat", timing: HtmxTriggerTiming.AfterSettle)));
139+
const string expected = @"{""cool"":{""magic"":""something""},""neat"":{""moremagic"":false},""wow"":""""}";
140+
141+
Response.Htmx(h => h.WithTrigger("wow"));
142+
Response.Htmx(h =>
143+
{
144+
h.WithTrigger("cool", new { magic = "something" });
145+
h.WithTrigger("neat", new { moremagic = false });
146+
});
147+
148+
Assert.True(Headers.ContainsKey(Keys.Trigger));
149+
Assert.Equal(expected, Headers[Keys.Trigger]);
136150
}
137-
151+
138152
[Fact]
139-
public void Cant_use_legacy_triggerafterswap_and_with_trigger()
153+
public void Can_use_existing_trigger_with_detail_with_multiple_triggers_with_detail()
140154
{
141-
Response.Htmx(h => h.TriggerAfterSwap("cool"));
142-
Assert.Throws<Exception>(() => Response.Htmx(h => h.WithTrigger("neat", timing: HtmxTriggerTiming.AfterSwap)));
155+
const string expected = @"{""cool"":{""magic"":""something""},""neat"":{""moremagic"":false},""wow"":{""display"":true}}";
156+
157+
Response.Htmx(h => h.WithTrigger("wow", new { display = true }));
158+
Response.Htmx(h =>
159+
{
160+
h.WithTrigger("cool", new { magic = "something" });
161+
h.WithTrigger("neat", new { moremagic = false });
162+
});
163+
164+
Assert.True(Headers.ContainsKey(Keys.Trigger));
165+
Assert.Equal(expected, Headers[Keys.Trigger]);
143166
}
144167

145168
[Fact]

0 commit comments

Comments
 (0)