$(document).ready(function () { 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) => { const $div = $("
", { class: "p-2 hover:bg-gray-100 cursor-pointer", "data-value": author.name, text: author.name, }); $dropdown.append($div); }); const $list = $("#genres-list"); genresData.genres.forEach((genre) => { const $li = $("
  • ", { class: "mb-1" }); $li.html(` `); $list.append($li); }); initializeAuthorDropdown(); }) .catch((error) => console.error("Error loading data:", error)); function initializeAuthorDropdown() { const $authorSearchInput = $("#author-search-input"); const $authorDropdown = $("#author-dropdown"); const $selectedAuthorsContainer = $("#selected-authors-container"); const $dropdownItems = $authorDropdown.find("[data-value]"); let selectedAuthors = new Set(); const updateDropdownHighlights = () => { $dropdownItems.each(function () { const value = $(this).data("value"); $(this).toggleClass("bg-gray-200", selectedAuthors.has(value)); }); }; const renderSelectedAuthors = () => { $selectedAuthorsContainer.children().not("#author-search-input").remove(); selectedAuthors.forEach((author) => { const $authorChip = $("", { class: "flex items-center bg-gray-200 text-gray-800 text-sm font-medium px-2.5 py-0.5 rounded-full", }); $authorChip.html(` ${author} `); $selectedAuthorsContainer.append($authorChip); }); updateDropdownHighlights(); }; $authorSearchInput.on("focus", () => { $authorDropdown.removeClass("hidden"); }); $authorSearchInput.on("input", function () { const query = $(this).val().toLowerCase(); $dropdownItems.each(function () { const text = $(this).text().toLowerCase(); $(this).css("display", text.includes(query) ? "block" : "none"); }); $authorDropdown.removeClass("hidden"); }); $(document).on("click", function (event) { if ( !$selectedAuthorsContainer.is(event.target) && !$selectedAuthorsContainer.has(event.target).length && !$authorDropdown.is(event.target) && !$authorDropdown.has(event.target).length ) { $authorDropdown.addClass("hidden"); } }); $authorDropdown.on("click", "[data-value]", function () { const selectedValue = $(this).data("value"); if (selectedAuthors.has(selectedValue)) { selectedAuthors.delete(selectedValue); } else { selectedAuthors.add(selectedValue); } $authorSearchInput.val(""); renderSelectedAuthors(); $authorSearchInput.focus(); }); $selectedAuthorsContainer.on("click", "button", function () { const authorToRemove = $(this).data("author"); selectedAuthors.delete(authorToRemove); renderSelectedAuthors(); $authorSearchInput.focus(); }); renderSelectedAuthors(); } 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(); }); } });