Sweep all .cshtml files for encoding corruption; add pre-commit guard

Replace all corruption variants with HTML entities across 226 view files:
- 3-char UTF-8-as-Win1252 sequences (ae-corruption)
- Standalone smart/curly quotes that break C# Razor expressions
- Partially re-corrupted variants where the 3rd byte was normalised to ASCII

tools/Fix-Encoding.ps1: re-runnable sweep; uses [char] code points so the
script itself never contains a literal non-ASCII character; supports -DryRun

.githooks/pre-commit: blocks commits containing the ae-corruption byte
signature (xc3xa2xe2x82xac); git core.hooksPath = .githooks so the
hook is repo-committed and active for all future work on this machine.

Build clean; 225 unit tests pass.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-05-20 21:37:10 -04:00
parent 21b39161a3
commit a0bdd2b5b4
252 changed files with 1785 additions and 1633 deletions
@@ -31,7 +31,7 @@
<div class="col-md-4">
<label class="form-label fw-semibold">Vendor <span class="text-danger">*</span></label>
<select asp-for="VendorId" asp-items="vendors" class="form-select" required>
<option value=""> select vendor </option>
<option value="">&mdash; select vendor &mdash;</option>
</select>
</div>
<div class="col-md-3">
@@ -42,7 +42,7 @@
<div class="col-md-5">
<label class="form-label fw-semibold">AP Account <span class="text-danger">*</span></label>
<select asp-for="APAccountId" asp-items="apAccounts" class="form-select" required>
<option value=""> select account </option>
<option value="">&mdash; select account &mdash;</option>
</select>
</div>
<div class="col-12">
@@ -98,7 +98,7 @@
tr.innerHTML = `
<td>
<select name="lineAccountIds" class="form-select form-select-sm">
<option value=""> optional </option>
<option value="">&mdash; optional &mdash;</option>
${expenseAccounts.map(a => `<option value="${a.value}">${escHtml(a.text)}</option>`).join('')}
</select>
</td>
@@ -79,7 +79,7 @@
<dd class="col-8">@Model.CreditDate.ToString("MMMM d, yyyy")</dd>
<dt class="col-4 text-muted">AP Account</dt>
<dd class="col-8">@Model.APAccount?.AccountNumber @Model.APAccount?.Name</dd>
<dd class="col-8">@Model.APAccount?.AccountNumber &ndash; @Model.APAccount?.Name</dd>
@if (!string.IsNullOrWhiteSpace(Model.Memo))
{
@@ -128,7 +128,7 @@
@if (line.AccountId.HasValue && accountMap.TryGetValue(line.AccountId.Value, out var acct))
{ @acct }
else
{ <span class="text-muted"></span> }
{ <span class="text-muted">&mdash;</span> }
</td>
<td class="text-muted small">@line.Description</td>
<td class="text-end">@line.Amount.ToString("C")</td>
@@ -205,7 +205,7 @@
<td>
<a asp-controller="Bills" asp-action="Details" asp-route-id="@bill.Id">@bill.BillNumber</a>
</td>
<td class="text-muted small">@(bill.DueDate?.ToString("MMM d, yyyy") ?? "")</td>
<td class="text-muted small">@(bill.DueDate?.ToString("MMM d, yyyy") ?? "&mdash;")</td>
<td class="text-end">@bill.BalanceDue.ToString("C")</td>
<td class="text-end" style="width:150px">
<form asp-action="Apply" method="post" class="d-inline">
@@ -166,7 +166,7 @@
}
else
{
<span class="text-muted"></span>
<span class="text-muted">&mdash;</span>
}
</td>
<td>