$(document).ready(() => { const pathParts = window.location.pathname.split("/"); const authorId = pathParts[pathParts.length - 1]; if (!authorId || isNaN(authorId)) { showErrorState("Некорректный ID автора"); return; } loadAuthor(authorId); function loadAuthor(id) { showLoadingState(); fetch(`/api/authors/${id}`) .then((response) => { if (!response.ok) { if (response.status === 404) { throw new Error("Автор не найден"); } throw new Error(`HTTP error! status: ${response.status}`); } return response.json(); }) .then((author) => { renderAuthor(author); renderBooks(author.books); document.title = `LiB - ${author.name}`; }) .catch((error) => { console.error("Error loading author:", error); showErrorState(error.message); }); } function renderAuthor(author) { const $card = $("#author-card"); const firstLetter = author.name.charAt(0).toUpperCase(); const booksCount = author.books ? author.books.length : 0; const booksWord = getWordForm(booksCount, ["книга", "книги", "книг"]); $card.html(`
${firstLetter}

${escapeHtml(author.name)}

ID: ${author.id}
${booksCount} ${booksWord} в библиотеке
Вернуться к списку авторов
`); } function renderBooks(books) { const $container = $("#books-container"); $container.empty(); if (!books || books.length === 0) { $container.html(`

У этого автора пока нет книг в библиотеке

`); return; } const $grid = $('
'); books.forEach((book) => { const $bookCard = $(`

${escapeHtml(book.title)}

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

`); $grid.append($bookCard); }); $container.append($grid); $container.off("click", ".book-card").on("click", ".book-card", function () { const bookId = $(this).data("id"); window.location.href = `/book/${bookId}`; }); } function showLoadingState() { const $authorCard = $("#author-card"); const $booksContainer = $("#books-container"); $authorCard.html(`
`); $booksContainer.html(`
${Array(3) .fill() .map( () => `
` ) .join("")}
`); } function showErrorState(message) { const $authorCard = $("#author-card"); const $booksSection = $("#books-section"); $booksSection.hide(); $authorCard.html(`

${escapeHtml(message)}

Не удалось загрузить информацию об авторе

К списку авторов
`); $("#retry-btn").on("click", function () { $booksSection.show(); loadAuthor(authorId); }); } function getWordForm(number, forms) { const cases = [2, 0, 1, 1, 1, 2]; const index = number % 100 > 4 && number % 100 < 20 ? 2 : cases[Math.min(number % 10, 5)]; return forms[index]; } function escapeHtml(text) { if (!text) return ""; const div = document.createElement("div"); div.textContent = text; return div.innerHTML; } 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(); }); } });