I've складав докупи відповідний фільтр для списку блогів: мова, діапазон дат, сорт і парування, які залишаються гарячими з HTMX.
Вона закривається зараз, але, як завжди, останні 20% - це гострий біт: дати, кеші і далі, чи працює він локально, але не після свопінгу?Якщо ви хочете, щоб глибинний пірнань з багатьма кодами і діаграмами, подивіться:
/ blog/ filterbarprogressand grees
/blog/calendar-daysВиблиски календаря з' являються з крихітнихNameLinkUrlPagination зберігає поточні фільтри (// Server: cache and vary for index
[ResponseCache(Duration = 300, VaryByHeader = "hx-request",
VaryByQueryKeys = new[] { "page", "pageSize", nameof(startDate), nameof(endDate), nameof(language), nameof(orderBy), nameof(orderDir) },
Location = ResponseCacheLocation.Any)]
[OutputCache(Duration = 3600, VaryByHeaderNames = new[] { "hx-request" },
VaryByQueryKeys = new[] { nameof(page), nameof(pageSize), nameof(startDate), nameof(endDate), nameof(language), nameof(orderBy), nameof(orderDir) })]
public async Task<IActionResult> Index(int page = 1, int pageSize = 20, DateTime? startDate = null, DateTime? endDate = null,
string language = MarkdownBaseService.EnglishLanguage, string orderBy = "date", string orderDir = "desc")
{
var posts = await blogViewService.GetPagedPosts(page, pageSize, language: language, startDate: startDate, endDate: endDate);
posts.LinkUrl = Url.Action("Index", "Blog", new { startDate, endDate, language, orderBy, orderDir });
if (Request.IsHtmx()) return PartialView("_BlogSummaryList", posts);
return View("Index", posts);
}
Видимість і контроль за вмістом
IsHidden: Тепер дописи можна позначати як приховані (ScheduledPublishDate: У поштових скриньках може бутиIsPinned: Дописи позначені// BlogService now filters appropriately
var now = DateTimeOffset.UtcNow;
postQuery = postQuery.Where(x =>
!x.IsHidden &&
(x.ScheduledPublishDate == null || x.ScheduledPublishDate <= now));
// Pinned posts sorted first on page 1
if (isFirstPage)
{
postQuery = postQuery.OrderByDescending(x => x.IsPinned)
.ThenByDescending(x => x.PublishedDate.DateTime);
}
/blog/date-rangeСтворитикінцева точка<clear-param>: Додано<clear-param name="startDate">Clear Date</clear-param>
<clear-param all="true" exclude="language">Clear All Filters</clear-param>
new URL(window.location.href)Фіксація: почати зlangSelect.addEventListener('change', async function(){
const u = new URL(window.location.href);
u.searchParams.set('language', langSelect.value);
u.searchParams.set('page','1');
const [ob,od] = (orderSelect.value||'date_desc').split('_');
u.searchParams.set('orderBy', ob);
u.searchParams.set('orderDir', od);
if(input._flatpickr && input._flatpickr.selectedDates.length===2){
const [s,e] = input._flatpickr.selectedDates;
u.searchParams.set('startDate', s.toISOString().substring(0,10));
u.searchParams.set('endDate', e.toISOString().substring(0,10));
}
applyNavigation(u);
});
і перевизначити лише змінені параметри; якщо у Flatpicker є вибір, перезастосуйте його.
fp.redraw()Виправлення: Вилучити/ відтворити Flatpicker на init, а потім отримати підсвічування за видимий місяць і викликhtmx:afterSwap.; also re-run init on
<html class>Фіксація: спостереженняfp.redraw()і викликsequenceDiagram
participant User
participant JS as blog-index.js
participant HT as HTMX
participant S as Server
User->>JS: Change language/order or pick a date range
JS->>JS: Update URLSearchParams, pushState
JS->>HT: htmx.ajax('GET', /blog?...)
HT->>S: GET /blog
S-->>HT: HTML partial (_BlogSummaryList)
HT-->>User: Swap #content
Виявлення запиту на виявлення сторінки HTMXpagerequestПанель фільтрування використовує нетиповий
public static bool IsPageRequest(this HttpRequest request)
{
return request.Headers.ContainsKey("pagerequest") &&
request.Headers["pagerequest"] == "true";
}
заголовок, щоб допомогти серверу розрізняти між завантаженням повної сторінки і частковими свопінгами HTMX:
**За допомогою цього пункту можна повернути лише частковий перегляд запитів HTMX і повний перегляд сторінок для безпосередньої навігації.**Примітка
Підтримка клавіатури під час захоплення діапазонів
© 2026 Scott Galloway — Unlicense — All content and source code on this site is free to use, copy, modify, and sell.