razorwire-mvc
RazorWire MVC Example
This sample is the concrete proof behind the RazorWire package README. It shows how returned Razor fragments, islands, and SSE fit into a normal ASP.NET Core MVC app without a separate client rendering stack.
Start Here: Return Razor Fragments
Run the application from the repository root:
dotnet run --project examples/razorwire-mvc/RazorWireWebExample.csprojThis assumes you are in a clone of this repository with the .NET 10 SDK installed.
If you
cd examples/razorwire-mvcfirst,dotnet runalso works from there.Open
http://localhost:5233/Reactivityor use the port printed in the console if it differs.Wait for the
Permanent Islandsidebar to load.Click the
+button in the counter widget.Watch
Instance ScoreandSession Scoreupdate in place without a full page reload.
That is the core RazorWire workflow in one interaction: a normal MVC form posts, the controller returns targeted Razor fragments, and the UI updates only where it needs to.
What Just Happened
/Reactivity
-> loads the Permanent Island from /Reactivity/Sidebar
-> renders the Counter view component inside that island
-> posts the counter form to ReactivityController.IncrementCounter
-> returns a RazorWire stream with targeted updates
-> updates the two counters and replaces the hidden input for the next click
Files Behind the Hero Flow
examples/razorwire-mvc/Views/Reactivity/Index.cshtmlloads the permanent island withsrc="/Reactivity/Sidebar".examples/razorwire-mvc/Views/Shared/_Sidebar.cshtmlhosts the island content and invokes theCounterview component.examples/razorwire-mvc/Views/Shared/Components/Counter/Default.cshtmlrenders the counter values plus theIncrementCounterform.examples/razorwire-mvc/Controllers/ReactivityController.csreturns the targeted stream updates.examples/razorwire-mvc/Views/Reactivity/_CounterInput.cshtmlreplaces the hiddenclientCountinput after each click.
Proof Slice
examples/razorwire-mvc/Views/Shared/Components/Counter/Default.cshtml
<div id="instance-score-value" class="text-2xl font-black text-indigo-600 tabular-nums">@Model</div>
<div id="session-score-value" class="text-2xl font-black text-indigo-400 tabular-nums">0</div>
<form asp-controller="Reactivity" asp-action="IncrementCounter" method="post" rw-active="true" data-counter-form>
<input type="hidden" name="clientCount" id="client-count-input" value="0" />
<button type="submit">+</button>
</form>
examples/razorwire-mvc/Controllers/ReactivityController.cs
[HttpPost]
[ValidateAntiForgeryToken]
public IActionResult IncrementCounter([FromForm] int clientCount)
{
CounterViewComponent.Increment();
clientCount++;
if (Request.IsTurboRequest())
{
return this.RazorWireStream()
.Update("instance-score-value", CounterViewComponent.Count.ToString())
.Update("session-score-value", clientCount.ToString())
.ReplacePartial("client-count-input", "_CounterInput", clientCount)
.BuildResult();
}
var referer = Request.Headers["Referer"].ToString();
return Url.IsLocalUrl(referer) ? Redirect(referer) : RedirectToAction(nameof(Index));
}
examples/razorwire-mvc/Views/Reactivity/_CounterInput.cshtml
<input type='hidden' name='clientCount' id='client-count-input' value='@Model' />
If Your Result Differs
- If the page loads on a different port, use the URL printed by
dotnet run. - If clicking
+gives you a bare400 Bad Request, check the package docs for Security & Anti-Forgery. That is the first thing to verify when you copy this pattern into another page or app. - If the form does not update in place, check the same anti-forgery guidance first, then confirm you are still posting with
rw-active="true"and returning a RazorWire stream fromIncrementCounter. - If you want the broader sample context instead of the focused proof, continue below.
Broader Sample Features
Islands
The sample uses rw:island to load and persist independent UI regions.
ReactivityController.Sidebar()returns the permanent sidebar island.ReactivityController.UserList()returns theUserListview component inside its own island.Views/Home/Index.cshtml,Views/Reactivity/Index.cshtml, andViews/Navigation/Index.cshtmlall reuse the samepermanent-islandso it can persist across page transitions.
Live Updates over SSE
The sample also demonstrates live multi-client updates.
Views/Reactivity/Index.cshtmlincludes<rw:stream-source id="rw-stream-reactivity" channel="reactivity" permanent="true" />.ReactivityController.PublishMessage()pushes new messages to every connected client.ReactivityController.BroadcastUserPresenceAsync()updates the user list and online count across sessions.
Registration and Message Publishing
The reactivity page includes two additional form flows:
Views/Reactivity/_UserRegistration.cshtmlposts toRegisterUserand swaps the register and message forms.Views/Reactivity/_MessageForm.cshtmlposts toPublishMessageand prepends messages into the live feed.
Those flows are richer than the counter demo, but the counter is the cleanest first proof because it does not depend on stream-hub context to feel convincing.
Project Structure
Controllers/ReactivityController.cs: main demo controller for islands, form posts, and stream responses.Views/Reactivity/: reactivity page plus registration, message, and counter partials.Views/Shared/: shared island and view component rendering.ViewComponents/: view component entry points such asCounterandUserList.Services/: in-memory sample services such asUserPresenceServiceandMessageStore.
Development Notes
To enable Razor Runtime Compilation and live static asset updates in the sample, run in the Development environment, for example with ASPNETCORE_ENVIRONMENT=Development.
Local assets such as site.js and site.css automatically receive version hashes for cache busting. You can still use asp-append-version="true" explicitly if you want to make that behavior obvious in markup.