// Silktide Consent Manager - https://silktide.com/consent-manager/ class SilktideCookieBanner { constructor(config) { this.config = config; this.wrapper = null; this.banner = null; this.modal = null; this.cookieIcon = null; this.backdrop = null; this.createWrapper(); if (this.shouldShowBackdrop()) { this.createBackdrop(); } this.createCookieIcon(); this.createModal(); if (this.shouldShowBanner()) { this.createBanner(); this.showBackdrop(); // while banner is visible -> icon must stay hidden this.hideCookieIcon(); } else { // ✅ NEW: icon follows the same "once per day" rule as the banner if (this.shouldShowCookieIcon()) { this.showCookieIcon(); } else { this.hideCookieIcon(); } } this.setupEventListeners(); if (this.hasSetInitialCookieChoices()) { this.loadRequiredCookies(); this.runAcceptedCookieCallbacks(); } } destroyCookieBanner() { if (this.wrapper && this.wrapper.parentNode) { this.wrapper.parentNode.removeChild(this.wrapper); } this.allowBodyScroll(); this.wrapper = null; this.banner = null; this.modal = null; this.cookieIcon = null; this.backdrop = null; } // ---------------------------------------------------------------- // Daily cooldown // ---------------------------------------------------------------- getCooldownKey() { return `silktideCookieBanner_CooldownUntil${this.getBannerSuffix()}`; } setCooldownOneDay() { const until = Date.now() + 24 * 60 * 60 * 1000; localStorage.setItem(this.getCooldownKey(), String(until)); } isInCooldown() { const raw = localStorage.getItem(this.getCooldownKey()); const until = raw ? parseInt(raw, 10) : 0; return Number.isFinite(until) && until > Date.now(); } // ✅ NEW: icon should be shown only when banner is allowed to show (same daily logic) shouldShowCookieIcon() { if (this.config.showCookieIcon === false) return false; // First visit: banner shows -> icon stays hidden if (!this.hasSetInitialCookieChoices()) return false; // If in cooldown -> hide icon as well if (this.isInCooldown()) return false; // Otherwise (next day) -> allow icon return true; } // ---------------------------------------------------------------- // Wrapper // ---------------------------------------------------------------- createWrapper() { this.wrapper = document.createElement('div'); this.wrapper.id = 'silktide-wrapper'; document.body.insertBefore(this.wrapper, document.body.firstChild); } // ---------------------------------------------------------------- // Wrapper Child Generator // ---------------------------------------------------------------- createWrapperChild(htmlContent, id) { const child = document.createElement('div'); child.id = id; if (htmlContent != null) child.innerHTML = htmlContent; if (!this.wrapper || !document.body.contains(this.wrapper)) { this.createWrapper(); } this.wrapper.appendChild(child); return child; } // ---------------------------------------------------------------- // Backdrop // ---------------------------------------------------------------- createBackdrop() { this.backdrop = this.createWrapperChild(null, 'silktide-backdrop'); // prevent Lenis from hijacking wheel/touch on backdrop (Safari-friendly) this.backdrop.setAttribute('data-lenis-prevent', ''); this.backdrop.setAttribute('data-lenis-prevent-wheel', ''); this.backdrop.setAttribute('data-lenis-prevent-touch', ''); } showBackdrop() { if (this.backdrop) this.backdrop.style.display = 'block'; if (typeof this.config.onBackdropOpen === 'function') { this.config.onBackdropOpen(); } } hideBackdrop() { if (this.backdrop) this.backdrop.style.display = 'none'; if (typeof this.config.onBackdropClose === 'function') { this.config.onBackdropClose(); } } shouldShowBackdrop() { return this.config?.background?.showBackground || false; } // ---------------------------------------------------------------- // Storage + checkbox sync // ---------------------------------------------------------------- updateCheckboxState(saveToStorage = false) { if (!this.modal) return; const preferencesSection = this.modal.querySelector('#cookie-preferences'); if (!preferencesSection) return; const checkboxes = preferencesSection.querySelectorAll('input[type="checkbox"]'); checkboxes.forEach((checkbox) => { const [, cookieId] = checkbox.id.split('cookies-'); const cookieType = this.config.cookieTypes.find((type) => type.id === cookieId); if (!cookieType) return; if (saveToStorage) { const currentState = checkbox.checked; if (cookieType.required) { localStorage.setItem(`silktideCookieChoice_${cookieId}${this.getBannerSuffix()}`, 'true'); } else { localStorage.setItem( `silktideCookieChoice_${cookieId}${this.getBannerSuffix()}`, currentState.toString(), ); if (currentState && typeof cookieType.onAccept === 'function') { cookieType.onAccept(); } else if (!currentState && typeof cookieType.onReject === 'function') { cookieType.onReject(); } } } else { if (cookieType.required) { checkbox.checked = true; checkbox.disabled = true; } else { const storedValue = localStorage.getItem( `silktideCookieChoice_${cookieId}${this.getBannerSuffix()}`, ); if (storedValue !== null) { checkbox.checked = storedValue === 'true'; } else { checkbox.checked = !!cookieType.defaultValue; } } } }); } setInitialCookieChoiceMade() { window.localStorage.setItem(`silktideCookieBanner_InitialChoice${this.getBannerSuffix()}`, 1); } hasSetInitialCookieChoices() { return !!localStorage.getItem(`silktideCookieBanner_InitialChoice${this.getBannerSuffix()}`); } // ---------------------------------------------------------------- // Consent Handling // ---------------------------------------------------------------- handleCookieChoice(accepted) { this.setInitialCookieChoiceMade(); this.setCooldownOneDay(); this.removeBanner(); this.hideBackdrop(); if (this.modal) { this.modal.style.display = 'none'; } // keep cookie icon hidden after accept/reject this.hideCookieIcon(); this.allowBodyScroll(); this.config.cookieTypes.forEach((type) => { if (type.required === true) { localStorage.setItem(`silktideCookieChoice_${type.id}${this.getBannerSuffix()}`, 'true'); if (typeof type.onAccept === 'function') type.onAccept(); return; } localStorage.setItem( `silktideCookieChoice_${type.id}${this.getBannerSuffix()}`, accepted.toString(), ); if (accepted) { if (typeof type.onAccept === 'function') type.onAccept(); } else { if (typeof type.onReject === 'function') type.onReject(); } }); if (accepted && typeof this.config.onAcceptAll === 'function') { this.config.onAcceptAll(); } else if (!accepted && typeof this.config.onRejectAll === 'function') { this.config.onRejectAll(); } this.updateCheckboxState(false); } getAcceptedCookies() { return (this.config.cookieTypes || []).reduce((acc, cookieType) => { acc[cookieType.id] = localStorage.getItem(`silktideCookieChoice_${cookieType.id}${this.getBannerSuffix()}`) === 'true'; return acc; }, {}); } runAcceptedCookieCallbacks() { if (!this.config.cookieTypes) return; const acceptedCookies = this.getAcceptedCookies(); this.config.cookieTypes.forEach((type) => { if (type.required) return; if (acceptedCookies[type.id] && typeof type.onAccept === 'function') { type.onAccept(); } }); } runStoredCookiePreferenceCallbacks() { if (!this.config.cookieTypes) return; this.config.cookieTypes.forEach((type) => { const accepted = localStorage.getItem(`silktideCookieChoice_${type.id}${this.getBannerSuffix()}`) === 'true'; if (accepted) { if (typeof type.onAccept === 'function') type.onAccept(); } else { if (typeof type.onReject === 'function') type.onReject(); } }); } loadRequiredCookies() { if (!this.config.cookieTypes) return; this.config.cookieTypes.forEach((cookie) => { if (cookie.required && typeof cookie.onAccept === 'function') { cookie.onAccept(); } }); } // ---------------------------------------------------------------- // Banner // ---------------------------------------------------------------- getBannerContent() { const bannerDescription = this.config.text?.banner?.description || "
We use cookies on our site to enhance your user experience, provide personalized content, and analyze our traffic.
"; const acceptAllButtonText = this.config.text?.banner?.acceptAllButtonText || 'Accept all'; const acceptAllButtonLabel = this.config.text?.banner?.acceptAllButtonAccessibleLabel; const acceptAllButton = ``; const rejectNonEssentialButtonText = this.config.text?.banner?.rejectNonEssentialButtonText || 'Reject non-essential'; const rejectNonEssentialButtonLabel = this.config.text?.banner?.rejectNonEssentialButtonAccessibleLabel; const rejectNonEssentialButton = ``; const preferencesButtonText = this.config.text?.banner?.preferencesButtonText || 'Preferences'; const preferencesButtonLabel = this.config.text?.banner?.preferencesButtonAccessibleLabel; const preferencesButton = ``; const silktideLogo = ` `; return ` ${bannerDescription}We respect your right to privacy. You can choose not to allow some types of cookies. Your cookie preferences will apply across our website.
"; const preferencesButtonLabel = this.config.text?.banner?.preferencesButtonAccessibleLabel; const closeModalButton = ``; const cookieTypes = this.config.cookieTypes || []; const acceptedCookieMap = this.getAcceptedCookies(); const acceptAllButtonText = this.config.text?.banner?.acceptAllButtonText || 'Accept all'; const acceptAllButtonLabel = this.config.text?.banner?.acceptAllButtonAccessibleLabel; const acceptAllButton = ``; const rejectNonEssentialButtonText = this.config.text?.banner?.rejectNonEssentialButtonText || 'Reject non-essential'; const rejectNonEssentialButtonLabel = this.config.text?.banner?.rejectNonEssentialButtonAccessibleLabel; const rejectNonEssentialButton = ``; const creditLinkText = this.config.text?.preferences?.creditLinkText || 'Get this banner for free'; const creditLinkAccessibleLabel = this.config.text?.preferences?.creditLinkAccessibleLabel; const creditLink = creditLinkText ? `${creditLinkText}` : ''; const switchOnText = this.config.text?.switch?.on || 'On'; const switchOffText = this.config.text?.switch?.off || 'Off'; return `