UNPKG

studiocms

Version:

Astro Native CMS for AstroDB. Built from the ground up by the Astro community.

665 lines (646 loc) 23.3 kB
const PERMISSION_HIERARCHY = { owner: /* @__PURE__ */ new Set(["owner"]), admin: /* @__PURE__ */ new Set(["owner", "admin"]), editor: /* @__PURE__ */ new Set(["owner", "admin", "editor"]), visitor: /* @__PURE__ */ new Set(["owner", "admin", "editor", "visitor"]), unknown: /* @__PURE__ */ new Set(["owner", "admin", "editor", "visitor", "unknown"]) }; const KNOWN_API_ROUTES = ["/studiocms_api/", "/_studiocms-devapps/", "/_web-vitals"]; const DEFAULT_AVATAR = ""; const COMPONENT_STYLES = ` :host { --border: hsl(240 5% 17%); --background-base: hsl(0 0% 6%); --background-step-1: hsl(0 0% 8%); --background-step-2: hsl(0 0% 10%); --background-step-3: hsl(0 0% 14%); --primary-base: hsl(259 83% 73%); --success-base: hsl(142 71% 46%); --warning-base: hsl(48 96% 53%); --danger-base: hsl(339 97% 31%); --info-base: hsl(217 92% 52%); --light: 70; --threshold: 50; } [data-theme="light"] { --border: hsl(263 5% 68%); --background-base: hsl(0 0% 97%); --background-step-1: hsl(0 0% 90%); --background-step-2: hsl(0 0% 85%); --background-step-3: hsl(0 0% 80%); --primary-base: hsl(259 85% 61%); --success-base: hsl(142 59% 47%); --warning-base: hsl(48 92% 46%); --danger-base: hsl(339 97% 31%); --info-base: hsl(217 92% 52%); } .menu_overlay { position: fixed; background: rgba(0,0,0,0.4); inset: 0; z-index: 500; display: none; } .menu_overlay.menuOpened { display: block; } .cornerMenu { position: fixed; right: 25px; bottom: 25px; width: 50px; height: 50px; background: var(--background-step-1); box-shadow: 0 3px 7px rgba(0,0,0,0.3); border-radius: 50%; z-index: 600; cursor: pointer; transition: transform 0.3s ease; display: flex; align-items: center; justify-content: center; } .avatar-container { width: 100%; height: 100%; border-radius: 50%; background: var(--background-step-1); border: 1px solid var(--border); object-fit: cover; z-index: 700; transition: transform 0.3s ease; display: flex; align-items: center; justify-content: center; } .avatar { width: 100%; height: 100%; background: var(--background-step-1); border: 1px solid var(--border); border-radius: 50%; object-fit: cover; transition: transform 0.3s ease; display: flex; align-items: center; justify-content: center; } .avatar-error { width: 100%; height: auto; margin: 2.5rem; border: none; } .cornerMenu.menuOpened .avatar-container { transform: scale(1.5); border: 1px solid var(--border); } .menu { --switch: calc((var(--light) - var(--threshold)) * -100%); position: absolute; width: 32px; height: 32px; background: var(--background-step-2); box-shadow: 0 3px 7px rgba(0,0,0,0.1); border-radius: 50%; border: 1px solid var(--border); color: hsl(0, 0%, var(--switch)); display: flex; align-items: center; justify-content: center; padding: 8px; top: -5px; left: -5px; opacity: 0; z-index: 550; pointer-events: none; user-select: none; transition: all 0.4s ease-in-out; text-decoration: none; } .menu svg { width: 24px; height: 24px; } .cornerMenu.menuOpened .menu { opacity: 1; cursor: pointer; transition: transform 0.3s ease, opacity 0.3s ease, background-color 0.15s ease; } .cornerMenu.menuOpened .menu:hover { background: var(--background-step-3); } /* Click protection: Only enable pointer events when menu is ready */ .cornerMenu.menu-ready .menu { pointer-events: all; box-shadow: 0 3px 7px rgba(0,0,0,0.1), 0 0 0 1px color-mix(in hsl, var(--primary-base) 20%, transparent); } .cornerMenu.menu-ready .menu:hover { box-shadow: 0 3px 7px rgba(0,0,0,0.2), 0 0 0 2px color-mix(in hsl, var(--primary-base) 40%, transparent); } /* Visual feedback for ignored clicks */ .menu.click-ignored { animation: shake 0.3s ease-in-out; } @keyframes shake { 0%, 100% { transform: translateX(0) translateY(0); } 25% { transform: translateX(-2px) translateY(-1px); } 50% { transform: translateX(2px) translateY(1px); } 75% { transform: translateX(-1px) translateY(-2px); } } .cornerMenu.menuOpened .menu:nth-child(1) { transform: translate(-105px, 20px); transition-delay: 0s; } .cornerMenu.menuOpened .menu:nth-child(2) { transform: translate(-78px, -33px); transition-delay: 0.05s; } .cornerMenu.menuOpened .menu:nth-child(3) { transform: translate(-38px, -76px); transition-delay: 0.1s; } .cornerMenu.menuOpened .menu:nth-child(4) { transform: translate(18px, -99px); transition-delay: 0.15s; } .menu.logout { color: var(--danger-base); } .menu.profile { color: var(--primary-base); } .menu.dashboard { color: var(--success-base); } .menu.edit { color: var(--warning-base); } `; function verifyUserPermissionLevel(userLevel, requiredLevel) { return PERMISSION_HIERARCHY[requiredLevel]?.has(userLevel) ?? false; } function shouldSkipRendering(pathname) { return KNOWN_API_ROUTES.some((route) => pathname.includes(route)); } function isDashboardRoute(pathname, dashboardRoute) { return pathname.includes(dashboardRoute); } class UserQuickTools extends HTMLElement { sessionData = null; isMenuOpen = false; menuItemsReady = false; lastMenuToggleTime = 0; readyTimeout = null; themeObserver = null; cornerMenu = null; menuOverlay = null; isInitialized = false; userInteractionListeners = []; // Click protection settings CLICK_PROTECTION_DURATION = 400; // milliseconds MENU_READY_DELAY = 350; // milliseconds (after animation completes) // Static menu items configuration static MENU_ITEMS = [ { name: "Logout", svg: '<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="M8.25 9V5.25A2.25 2.25 0 0 1 10.5 3h6a2.25 2.25 0 0 1 2.25 2.25v13.5A2.25 2.25 0 0 1 16.5 21h-6a2.25 2.25 0 0 1-2.25-2.25V15m-3 0-3-3m0 0 3-3m-3 3H15" /></svg>', permission: "visitor", cssClass: "logout" }, { name: "Profile", svg: '<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="M15.75 6a3.75 3.75 0 1 1-7.5 0 3.75 3.75 0 0 1 7.5 0ZM4.501 20.118a7.5 7.5 0 0 1 14.998 0A17.933 17.933 0 0 1 12 21.75c-2.676 0-5.216-.584-7.499-1.632Z" /></svg>', permission: "visitor", cssClass: "profile" }, { name: "Dashboard", svg: '<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="M2.25 7.125C2.25 6.504 2.754 6 3.375 6h6c.621 0 1.125.504 1.125 1.125v3.75c0 .621-.504 1.125-1.125 1.125h-6a1.125 1.125 0 0 1-1.125-1.125v-3.75ZM14.25 8.625c0-.621.504-1.125 1.125-1.125h5.25c.621 0 1.125.504 1.125 1.125v8.25c0 .621-.504 1.125-1.125 1.125h-5.25a1.125 1.125 0 0 1-1.125-1.125v-8.25ZM3.75 16.125c0-.621.504-1.125 1.125-1.125h5.25c.621 0 1.125.504 1.125 1.125v2.25c0 .621-.504 1.125-1.125 1.125h-5.25a1.125 1.125 0 0 1-1.125-1.125v-2.25Z" /></svg>', permission: "editor", cssClass: "dashboard" }, { name: "Edit", svg: '<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="m16.862 4.487 1.687-1.688a1.875 1.875 0 1 1 2.652 2.652L10.582 16.07a4.5 4.5 0 0 1-1.897 1.13L6 18l.8-2.685a4.5 4.5 0 0 1 1.13-1.897l8.932-8.931Zm0 0L19.5 7.125M18 14v4.75A2.25 2.25 0 0 1 15.75 21H5.25A2.25 2.25 0 0 1 3 18.75V8.25A2.25 2.25 0 0 1 5.25 6H10" /></svg>', permission: "editor", cssClass: "edit" } ]; constructor() { super(); this.attachShadow({ mode: "open" }); } connectedCallback() { const pathname = window.location.pathname; if (shouldSkipRendering(pathname)) { return; } this.initOnUserInteraction(); } /* v8 ignore stop */ initOnUserInteraction() { const interactionEvents = [ "mouseenter", "mousemove", "touchstart", "scroll", "keydown", "click" ]; const handleUserInteraction = () => { if (this.isInitialized) return; this.removeInteractionListeners(); this.scheduleInitialization(); }; interactionEvents.forEach((eventType) => { const listener = handleUserInteraction; document.addEventListener(eventType, listener, { passive: true, once: true // Automatically removes after first trigger }); this.userInteractionListeners.push({ event: eventType, handler: listener }); }); } removeInteractionListeners() { this.userInteractionListeners.forEach(({ event, handler }) => { document.removeEventListener(event, handler); }); this.userInteractionListeners = []; } scheduleInitialization() { if (this.isInitialized) return; this.isInitialized = true; const initializeComponent = () => { const pathname = window.location.pathname; this.initializeAsync(pathname).catch((error) => { console.error("UserQuickTools initialization failed:", error); }); }; if ("requestIdleCallback" in window) { requestIdleCallback(initializeComponent, { timeout: 1e3 }); } else { setTimeout(initializeComponent, 0); } } async initializeAsync(pathname) { try { const sessionData = await this.getSession(); if (!sessionData?.isLoggedIn) { return; } if (isDashboardRoute(pathname, sessionData.routes.dashboardIndex)) { return; } this.sessionData = sessionData; this.scheduleRender(); } catch (error) { console.warn("UserQuickTools failed to initialize:", error); } } /* v8 ignore start */ scheduleRender() { const renderComponent = () => { this.render(); this.setupEventListeners(); this.setupThemeObserver(); }; requestAnimationFrame(() => { if ("requestIdleCallback" in window) { requestIdleCallback(renderComponent, { timeout: 500 }); } else { setTimeout(renderComponent, 0); } }); } disconnectedCallback() { this.cleanup(); this.removeInteractionListeners(); } async render() { if (!this.shadowRoot || !this.sessionData) return; this.menuOverlay = document.createElement("div"); this.menuOverlay.className = "menu_overlay"; this.cornerMenu = document.createElement("div"); this.cornerMenu.className = "cornerMenu"; this.cornerMenu.dataset.theme = document.documentElement.dataset.theme ?? "dark"; const styleElm = document.createElement("style"); styleElm.textContent = COMPONENT_STYLES; this.addMenuItems(); this.shadowRoot.append(styleElm, this.menuOverlay, this.cornerMenu); void this.addUserAvatar().catch((e) => console.warn("Avatar load failed:", e)); } addMenuItems() { if (!this.cornerMenu || !this.sessionData) return; const { routes, permissionLevel } = this.sessionData; const routeMap = { Logout: routes.logout, Profile: routes.userProfile, Dashboard: routes.dashboardIndex, Edit: routes.contentManagement }; UserQuickTools.MENU_ITEMS.forEach((item) => { if (verifyUserPermissionLevel(permissionLevel, item.permission)) { const menuElement = this.createMenuElement({ ...item, href: routeMap[item.name] }); this.cornerMenu.appendChild(menuElement); } }); } createMenuElement(item) { if (item.name === "Logout") { return this.createLogoutElement(item); } const element = document.createElement("a"); element.className = `menu ${item.cssClass}`; element.title = item.name; const parser = new DOMParser(); const svgDoc = parser.parseFromString(item.svg, "image/svg+xml"); const svgElement = svgDoc.documentElement; if (svgElement && svgElement.nodeName === "svg") { element.appendChild(svgElement.cloneNode(true)); } else { console.warn("Invalid SVG content for menu item:", item.name); } element.href = item.href; element.addEventListener("click", (e) => { const timeSinceToggle = Date.now() - this.lastMenuToggleTime; if (!this.menuItemsReady || timeSinceToggle < this.CLICK_PROTECTION_DURATION) { e.preventDefault(); e.stopPropagation(); element.classList.add("click-ignored"); setTimeout(() => element.classList.remove("click-ignored"), 300); return false; } }); return element; } createLogoutElement(item) { const element = document.createElement("a"); element.className = `menu ${item.cssClass}`; element.title = item.name; element.href = "#"; const parser = new DOMParser(); const svgDoc = parser.parseFromString(item.svg, "image/svg+xml"); const svgElement = svgDoc.documentElement; if (svgElement && svgElement.nodeName === "svg") { element.appendChild(svgElement.cloneNode(true)); } element.addEventListener("click", (e) => { e.preventDefault(); const timeSinceToggle = Date.now() - this.lastMenuToggleTime; if (!this.menuItemsReady || timeSinceToggle < this.CLICK_PROTECTION_DURATION) { element.classList.add("click-ignored"); setTimeout(() => element.classList.remove("click-ignored"), 300); return; } this.submitLogoutForm(item.href); }); return element; } submitLogoutForm(logoutUrl) { const form = document.createElement("form"); form.method = "POST"; form.action = logoutUrl; form.style.display = "none"; document.body.appendChild(form); form.submit(); } async testAvatarURL(url) { let urlObj; try { urlObj = new URL(url); } catch { console.warn("Invalid avatar URL:", url); return void 0; } if (urlObj.protocol !== "https:") { console.error(`Insecure avatar URL protocol: ${urlObj.protocol}`); return void 0; } const controller = new AbortController(); const timeoutId = window.setTimeout(() => controller.abort(), 4e3); try { const response = await fetch(url, { method: "HEAD", signal: controller.signal, cache: "no-cache" }); if (!response.ok) return void 0; const contentType = (response.headers.get("content-type") || "").split(";")[0].trim(); if (!contentType.startsWith("image/")) return void 0; if (contentType === "image/svg+xml") { console.error("Remote SVG avatars are disallowed for security."); return void 0; } return { type: contentType }; } catch (err) { console.warn("Avatar HEAD check failed:", err); return void 0; } finally { clearTimeout(timeoutId); } } async addUserAvatar() { if (!this.cornerMenu || !this.sessionData) return; const { user, permissionLevel } = this.sessionData; const avatarContainer = document.createElement("div"); avatarContainer.className = "avatar-container"; const newAvatar = document.createElement("img"); const avatar = await (async () => { let avatar2 = DEFAULT_AVATAR; if (user.avatar) { const result = await this.testAvatarURL(user.avatar); if (result) { avatar2 = user.avatar; } } return avatar2; })(); newAvatar.src = avatar; newAvatar.width = 64; newAvatar.height = 64; newAvatar.className = "avatar"; newAvatar.alt = `${user.name} - ${this.capitalizeFirst(permissionLevel)}`; newAvatar.loading = "lazy"; newAvatar.decoding = "async"; newAvatar.referrerPolicy = "no-referrer"; if (avatar === DEFAULT_AVATAR) { newAvatar.classList.add("avatar-error"); } newAvatar.onerror = function() { this.src = DEFAULT_AVATAR; }; newAvatar.setAttribute("aria-hidden", "true"); avatarContainer.appendChild(newAvatar); this.cornerMenu.appendChild(avatarContainer); } setupEventListeners() { if (!this.cornerMenu || !this.menuOverlay) return; this.cornerMenu.addEventListener("click", this.handleMenuToggle.bind(this)); this.menuOverlay.addEventListener("click", this.handleOverlayClick.bind(this)); } handleMenuToggle() { this.lastMenuToggleTime = Date.now(); this.isMenuOpen = !this.isMenuOpen; if (this.isMenuOpen) { this.menuItemsReady = false; this.cornerMenu?.classList.remove("menu-ready"); this.readyTimeout = window.setTimeout(() => { this.menuItemsReady = true; this.cornerMenu?.classList.add("menu-ready"); }, this.MENU_READY_DELAY); } else { this.menuItemsReady = false; this.cornerMenu?.classList.remove("menu-ready"); if (this.readyTimeout) { clearTimeout(this.readyTimeout); this.readyTimeout = null; } } this.updateMenuState(); } handleOverlayClick() { if (this.isMenuOpen) { this.isMenuOpen = false; this.menuItemsReady = false; this.cornerMenu?.classList.remove("menu-ready"); if (this.readyTimeout) { clearTimeout(this.readyTimeout); this.readyTimeout = null; } this.updateMenuState(); } } updateMenuState() { if (!this.cornerMenu || !this.menuOverlay) return; const method = this.isMenuOpen ? "add" : "remove"; this.cornerMenu.classList[method]("menuOpened"); this.menuOverlay.classList[method]("menuOpened"); } setupThemeObserver() { if (!this.cornerMenu) return; const updateTheme = () => { const theme = document.documentElement.getAttribute("data-theme"); this.cornerMenu.dataset.theme = theme === "light" ? "light" : "dark"; }; this.themeObserver = new MutationObserver(updateTheme); this.themeObserver.observe(document.documentElement, { attributes: true, attributeFilter: ["data-theme"] }); } /* v8 ignore stop */ async getSession() { try { const controller = new AbortController(); const timeoutId = setTimeout(() => controller.abort(), 5e3); const response = await fetch("/studiocms_api/dashboard/verify-session", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ originPathname: window.location.toString() }), signal: controller.signal }); clearTimeout(timeoutId); return response.ok ? await response.json() : null; } catch (error) { if (error instanceof Error && error.name === "AbortError") { console.warn("Session verification timed out"); } else { console.warn("Session verification failed:", error); } return null; } } /* v8 ignore start */ capitalizeFirst(str) { return str.charAt(0).toUpperCase() + str.slice(1); } cleanup() { this.themeObserver?.disconnect(); this.themeObserver = null; if (this.readyTimeout) { clearTimeout(this.readyTimeout); this.readyTimeout = null; } this.cornerMenu = null; this.menuOverlay = null; this.sessionData = null; this.isInitialized = false; this.menuItemsReady = false; this.lastMenuToggleTime = 0; } /* v8 ignore stop */ } class ConfigurableUserQuickTools extends UserQuickTools { config; constructor() { super(); this.config = { strategy: this.getAttribute("data-init-strategy") || "interaction", timeout: Number.parseInt(this.getAttribute("data-timeout") || "1000", 10), clickProtectionDuration: Number.parseInt( this.getAttribute("data-click-protection") || "400", 10 ), menuReadyDelay: Number.parseInt(this.getAttribute("data-menu-delay") || "350", 10) }; if (this.config.clickProtectionDuration) { this.CLICK_PROTECTION_DURATION = this.config.clickProtectionDuration; } if (this.config.menuReadyDelay) { this.MENU_READY_DELAY = this.config.menuReadyDelay; } } connectedCallback() { const pathname = window.location.pathname; if (shouldSkipRendering(pathname)) { return; } switch (this.config.strategy) { case "immediate": this.scheduleInitialization(); break; /* v8 ignore stop */ case "idle": if ("requestIdleCallback" in window) { requestIdleCallback(() => this.scheduleInitialization(), { timeout: this.config.timeout }); } else { setTimeout(() => this.scheduleInitialization(), 0); } break; /* v8 ignore stop */ case "interaction": this.initOnUserInteraction(); break; default: console.warn(`Unknown initialization strategy: ${this.config.strategy}`); this.initOnUserInteraction(); break; } } } if ("customElements" in window && !customElements.get("user-quick-tools")) { customElements.define("user-quick-tools", ConfigurableUserQuickTools); } function initializeWhenReady() { const createElement = () => { if (!document.querySelector("user-quick-tools")) { const element = document.createElement("user-quick-tools"); element.setAttribute("data-init-strategy", "idle"); element.setAttribute("data-timeout", "1000"); element.setAttribute("data-click-protection", "400"); element.setAttribute("data-menu-delay", "350"); document.body.appendChild(element); } }; if (document.readyState === "loading") { document.addEventListener("DOMContentLoaded", () => { setTimeout(createElement, 0); }); } else { setTimeout(createElement, 0); } } initializeWhenReady(); export { COMPONENT_STYLES, ConfigurableUserQuickTools, DEFAULT_AVATAR, KNOWN_API_ROUTES, PERMISSION_HIERARCHY, UserQuickTools, initializeWhenReady, isDashboardRoute, shouldSkipRendering, verifyUserPermissionLevel };