$(document).ready(() => { let selectedAuthors = new Map(); let selectedGenres = new Map(); let currentPage = 1; let pageSize = 20; let totalBooks = 0; Promise.all([ fetch("/api/authors").then((response) => response.json()), fetch("/api/genres").then((response) => response.json()), ]) .then(([authorsData, genresData]) => { const $dropdown = $("#author-dropdown"); authorsData.authors.forEach((author) => { $("
") .addClass("p-2 hover:bg-gray-100 cursor-pointer author-item") .attr("data-id", author.id) .attr("data-name", author.name) .text(author.name) .appendTo($dropdown); }); const $list = $("#genres-list"); genresData.genres.forEach((genre) => { $("
  • ") .addClass("mb-1") .html( ``, ) .appendTo($list); }); initializeAuthorDropdown(); initializeFilters(); loadBooks(); }) .catch((error) => console.error("Error loading data:", error)); function loadBooks() { const searchQuery = $("#book-search-input").val().trim(); const params = new URLSearchParams(); if (searchQuery.length >= 3) { params.append("q", searchQuery); } selectedAuthors.forEach((name, id) => { params.append("author_ids", id); }); selectedGenres.forEach((name, id) => { params.append("genre_ids", id); }); params.append("page", currentPage); params.append("size", pageSize); const url = `/api/books/filter?${params.toString()}`; showLoadingState(); fetch(url) .then((response) => { if (!response.ok) { throw new Error(`HTTP error! status: ${response.status}`); } return response.json(); }) .then((data) => { totalBooks = data.total; renderBooks(data.books); renderPagination(); }) .catch((error) => { console.error("Error loading books:", error); showErrorState(); }); } function renderBooks(books) { const $container = $("#books-container"); $container.empty(); if (books.length === 0) { $container.html(`

    Книги не найдены

    Попробуйте изменить параметры поиска или фильтры

    `); return; } books.forEach((book) => { const authorsText = book.authors.map((a) => a.name).join(", ") || "Автор неизвестен"; const genresText = book.genres.map((g) => g.name).join(", ") || "Без жанра"; const $bookCard = $(`

    ${escapeHtml(book.title)}

    Авторы: ${escapeHtml(authorsText)}

    ${escapeHtml(book.description || "Описание отсутствует")}

    ${book.genres .map( (g) => ` ${escapeHtml(g.name)} `, ) .join("")}
    `); $container.append($bookCard); }); $container.on("click", ".book-card", function () { const bookId = $(this).data("id"); window.location.href = `/book/${bookId}`; }); } function renderPagination() { $("#pagination-container").remove(); const totalPages = Math.ceil(totalBooks / pageSize); if (totalPages <= 1) return; const $pagination = $(`
    `); const $pageNumbers = $pagination.find("#page-numbers"); const pages = generatePageNumbers(currentPage, totalPages); pages.forEach((page) => { if (page === "...") { $pageNumbers.append(`...`); } else { const isActive = page === currentPage; $pageNumbers.append(` `); } }); $("#books-container").after($pagination); $("#prev-page").on("click", function () { if (currentPage > 1) { currentPage--; loadBooks(); scrollToTop(); } }); $("#next-page").on("click", function () { if (currentPage < totalPages) { currentPage++; loadBooks(); scrollToTop(); } }); $(".page-btn").on("click", function () { const page = parseInt($(this).data("page")); if (page !== currentPage) { currentPage = page; loadBooks(); scrollToTop(); } }); } function generatePageNumbers(current, total) { const pages = []; const delta = 2; for (let i = 1; i <= total; i++) { if ( i === 1 || i === total || (i >= current - delta && i <= current + delta) ) { pages.push(i); } else if (pages[pages.length - 1] !== "...") { pages.push("..."); } } return pages; } function scrollToTop() { $("html, body").animate({ scrollTop: 0 }, 300); } function showLoadingState() { const $container = $("#books-container"); $container.html(`
    ${Array(3) .fill() .map( () => `
    `, ) .join("")}
    `); } function showErrorState() { const $container = $("#books-container"); $container.html(`

    Ошибка загрузки

    Не удалось загрузить список книг

    `); $("#retry-btn").on("click", loadBooks); } function escapeHtml(text) { if (!text) return ""; const div = document.createElement("div"); div.textContent = text; return div.innerHTML; } function initializeAuthorDropdown() { const $input = $("#author-search-input"); const $dropdown = $("#author-dropdown"); const $container = $("#selected-authors-container"); function updateHighlights() { $dropdown.find(".author-item").each(function () { const id = $(this).attr("data-id"); const isSelected = selectedAuthors.has(parseInt(id)); $(this) .toggleClass("bg-gray-300 text-gray-600", isSelected) .toggleClass("hover:bg-gray-100", !isSelected); }); } function filterDropdown(query) { const lowerQuery = query.toLowerCase(); $dropdown.find(".author-item").each(function () { $(this).toggle($(this).text().toLowerCase().includes(lowerQuery)); }); } function renderChips() { $container.find(".author-chip").remove(); selectedAuthors.forEach((name, id) => { $(` ${escapeHtml(name)} `).insertBefore($input); }); updateHighlights(); } function toggleAuthor(id, name) { id = parseInt(id); if (selectedAuthors.has(id)) { selectedAuthors.delete(id); } else { selectedAuthors.add(id, name); selectedAuthors.set(id, name); } $input.val(""); filterDropdown(""); renderChips(); } $input.on("focus", () => $dropdown.removeClass("hidden")); $input.on("input", function () { filterDropdown($(this).val().toLowerCase()); $dropdown.removeClass("hidden"); }); $(document).on("click", (e) => { if ( !$(e.target).closest("#selected-authors-container, #author-dropdown") .length ) { $dropdown.addClass("hidden"); } }); $dropdown.on("click", ".author-item", function (e) { e.stopPropagation(); toggleAuthor($(this).attr("data-id"), $(this).attr("data-name")); $input.focus(); }); $container.on("click", ".remove-author", function (e) { e.stopPropagation(); selectedAuthors.delete(parseInt($(this).attr("data-id"))); renderChips(); $input.focus(); }); $container.on("click", (e) => { if (!$(e.target).closest(".author-chip").length) { $input.focus(); } }); window.renderAuthorChips = renderChips; window.updateAuthorHighlights = updateHighlights; } function initializeFilters() { const $bookSearch = $("#book-search-input"); const $applyBtn = $("#apply-filters-btn"); const $resetBtn = $("#reset-filters-btn"); $("#genres-list").on("change", "input[type='checkbox']", function () { const id = parseInt($(this).attr("data-id")); const name = $(this).attr("data-name"); if ($(this).is(":checked")) { selectedGenres.set(id, name); } else { selectedGenres.delete(id); } }); $applyBtn.on("click", function () { currentPage = 1; loadBooks(); }); $resetBtn.on("click", function () { $bookSearch.val(""); selectedAuthors.clear(); $("#selected-authors-container .author-chip").remove(); if (window.updateAuthorHighlights) window.updateAuthorHighlights(); selectedGenres.clear(); $("#genres-list input[type='checkbox']").prop("checked", false); currentPage = 1; loadBooks(); }); let searchTimeout; $bookSearch.on("input", function () { clearTimeout(searchTimeout); const query = $(this).val().trim(); if (query.length >= 3 || query.length === 0) { searchTimeout = setTimeout(() => { currentPage = 1; loadBooks(); }, 500); } }); $bookSearch.on("keypress", function (e) { if (e.which === 13) { clearTimeout(searchTimeout); currentPage = 1; loadBooks(); } }); } const $guestLink = $("#guest-link"); const $userBtn = $("#user-btn"); const $userDropdown = $("#user-dropdown"); const $userArrow = $("#user-arrow"); const $userAvatar = $("#user-avatar"); const $dropdownName = $("#dropdown-name"); const $dropdownUsername = $("#dropdown-username"); const $dropdownEmail = $("#dropdown-email"); const $logoutBtn = $("#logout-btn"); let isDropdownOpen = false; function openDropdown() { isDropdownOpen = true; $userDropdown.removeClass("hidden"); $userArrow.addClass("rotate-180"); } function closeDropdown() { isDropdownOpen = false; $userDropdown.addClass("hidden"); $userArrow.removeClass("rotate-180"); } $userBtn.on("click", function (e) { e.stopPropagation(); isDropdownOpen ? closeDropdown() : openDropdown(); }); $(document).on("click", function (e) { if (isDropdownOpen && !$(e.target).closest("#user-menu-container").length) { closeDropdown(); } }); $(document).on("keydown", function (e) { if (e.key === "Escape" && isDropdownOpen) { closeDropdown(); } }); $logoutBtn.on("click", function () { localStorage.removeItem("access_token"); localStorage.removeItem("refresh_token"); window.location.reload(); }); function showGuest() { $guestLink.removeClass("hidden"); $userBtn.addClass("hidden").removeClass("flex"); closeDropdown(); } function showUser(user) { $guestLink.addClass("hidden"); $userBtn.removeClass("hidden").addClass("flex"); const displayName = user.full_name || user.username; const firstLetter = displayName.charAt(0).toUpperCase(); $userAvatar.text(firstLetter); $dropdownName.text(displayName); $dropdownUsername.text("@" + user.username); $dropdownEmail.text(user.email); } function updateUserAvatar(email) { if (!email) return; const cleanEmail = email.trim().toLowerCase(); const emailHash = sha256(cleanEmail); const avatarUrl = `https://www.gravatar.com/avatar/${emailHash}?d=identicon&s=200`; const avatarImg = document.getElementById("user-avatar"); if (avatarImg) { avatarImg.src = avatarUrl; } } const token = localStorage.getItem("access_token"); if (!token) { showGuest(); } else { fetch("/api/auth/me", { headers: { Authorization: "Bearer " + token }, }) .then((response) => { if (response.ok) return response.json(); throw new Error("Unauthorized"); }) .then((user) => { showUser(user); updateUserAvatar(user.email); document.getElementById("user-btn").classList.remove("hidden"); document.getElementById("guest-link").classList.add("hidden"); }) .catch(() => { localStorage.removeItem("access_token"); localStorage.removeItem("refresh_token"); showGuest(); }); } });