import { DeesElement, property, html, customElement, type TemplateResult, css, cssManager, unsafeCSS } from '@design.estate/dees-element'; import * as domtools from '@design.estate/dees-domtools'; import * as sharedStyles from '../styles/shared.styles.js'; import { demoFunc } from './upl-statuspage-header.demo.js'; declare global { interface HTMLElementTagNameMap { 'upl-statuspage-header': UplStatuspageHeader; } } @customElement('upl-statuspage-header') export class UplStatuspageHeader extends DeesElement { // STATIC public static demo = demoFunc; // INSTANCE @property({ type: String }) accessor pageTitle: string = "Statuspage Title"; @property({ type: Boolean }) accessor showReportButton: boolean = true; @property({ type: Boolean }) accessor showSubscribeButton: boolean = true; @property({ type: String }) accessor logoUrl: string = ''; @property({ type: Boolean }) accessor loading: boolean = false; constructor() { super(); } public static styles = [ domtools.elementBasic.staticStyles, css` :host { display: block; background: ${cssManager.bdTheme( 'rgba(255, 255, 255, 0.85)', 'rgba(9, 9, 11, 0.85)' )}; font-family: ${unsafeCSS(sharedStyles.fonts.base)}; color: ${sharedStyles.colors.text.primary}; border-bottom: 1px solid ${cssManager.bdTheme( 'rgba(0, 0, 0, 0.06)', 'rgba(255, 255, 255, 0.06)' )}; position: sticky; top: 0; z-index: 40; backdrop-filter: blur(16px) saturate(180%); -webkit-backdrop-filter: blur(16px) saturate(180%); } .header-container { max-width: 1200px; margin: 0 auto; padding: 0 ${unsafeCSS(sharedStyles.spacing.lg)}; } .header-nav { display: flex; align-items: center; justify-content: space-between; height: 64px; } .header-left { display: flex; align-items: center; gap: ${unsafeCSS(sharedStyles.spacing.md)}; } .header-actions { display: flex; align-items: center; gap: ${unsafeCSS(sharedStyles.spacing.sm)}; } .actionButton { font-family: ${unsafeCSS(sharedStyles.fonts.base)}; font-size: 13px; font-weight: 500; padding: 0 14px; border-radius: ${unsafeCSS(sharedStyles.borderRadius.base)}; cursor: pointer; user-select: none; transition: all ${unsafeCSS(sharedStyles.durations.fast)} ${unsafeCSS(sharedStyles.easings.default)}; display: inline-flex; align-items: center; justify-content: center; gap: 6px; height: 36px; background: transparent; border: 1px solid ${sharedStyles.colors.border.default}; color: ${sharedStyles.colors.text.primary}; letter-spacing: -0.01em; position: relative; overflow: hidden; } .actionButton::before { content: ''; position: absolute; inset: 0; background: ${cssManager.bdTheme( 'linear-gradient(135deg, rgba(0, 0, 0, 0.02) 0%, transparent 100%)', 'linear-gradient(135deg, rgba(255, 255, 255, 0.03) 0%, transparent 100%)' )}; opacity: 0; transition: opacity ${unsafeCSS(sharedStyles.durations.fast)} ${unsafeCSS(sharedStyles.easings.default)}; } .actionButton:hover { background: ${sharedStyles.colors.background.secondary}; border-color: ${sharedStyles.colors.border.muted}; box-shadow: ${unsafeCSS(sharedStyles.shadows.sm)}; transform: translateY(-1px); } .actionButton:hover::before { opacity: 1; } .actionButton:active { transform: translateY(0) scale(0.98); transition-duration: ${unsafeCSS(sharedStyles.durations.fast)}; box-shadow: none; } .actionButton:focus-visible { outline: 2px solid ${sharedStyles.colors.accent.focus}; outline-offset: 2px; } /* Button icon styles */ .actionButton svg { width: 14px; height: 14px; flex-shrink: 0; transition: transform ${unsafeCSS(sharedStyles.durations.fast)} ${unsafeCSS(sharedStyles.easings.default)}; } .actionButton:hover svg { transform: scale(1.1); } .site-title { font-size: 17px; font-weight: 600; letter-spacing: -0.02em; color: ${sharedStyles.colors.text.primary}; transition: color ${unsafeCSS(sharedStyles.durations.fast)} ${unsafeCSS(sharedStyles.easings.default)}; } .site-title:hover { color: ${sharedStyles.colors.text.secondary}; } .logo { height: 28px; width: auto; filter: ${cssManager.bdTheme('none', 'brightness(0) invert(1)')}; transition: all ${unsafeCSS(sharedStyles.durations.normal)} ${unsafeCSS(sharedStyles.easings.default)}; } .logo:hover { opacity: 0.8; transform: scale(1.02); } .page-info { padding: ${unsafeCSS(sharedStyles.spacing.lg)} 0 ${unsafeCSS(sharedStyles.spacing.xl)} 0; } .page-title { font-size: 48px; font-weight: 700; letter-spacing: -0.03em; line-height: 1.1; color: ${sharedStyles.colors.text.primary}; margin: 0 0 16px 0; } .page-subtitle { font-size: 18px; color: ${sharedStyles.colors.text.secondary}; margin: 0; line-height: 1.5; } /* Primary button variant */ .actionButton.primary { background: ${sharedStyles.colors.accent.primary}; color: ${sharedStyles.colors.background.primary}; border-color: transparent; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.1); } .actionButton.primary::before { background: linear-gradient(135deg, rgba(255, 255, 255, 0.15) 0%, transparent 100%); } .actionButton.primary:hover { background: ${sharedStyles.colors.accent.hover}; box-shadow: ${unsafeCSS(sharedStyles.shadows.md)}; } .loading-skeleton { height: 64px; background: ${sharedStyles.colors.background.secondary}; border-bottom: 1px solid ${sharedStyles.colors.border.default}; position: relative; overflow: hidden; } .loading-skeleton::after { content: ''; position: absolute; inset: 0; background: ${cssManager.bdTheme( 'linear-gradient(90deg, transparent 0%, rgba(0,0,0,0.04) 50%, transparent 100%)', 'linear-gradient(90deg, transparent 0%, rgba(255,255,255,0.04) 50%, transparent 100%)' )}; animation: shimmer 1.5s ${unsafeCSS(sharedStyles.easings.default)} infinite; } @keyframes shimmer { 0% { transform: translateX(-100%); } 100% { transform: translateX(200%); } } @media (max-width: 768px) { .header-container { padding: 0 ${unsafeCSS(sharedStyles.spacing.md)}; } .header-nav { height: 56px; } .header-left { gap: ${unsafeCSS(sharedStyles.spacing.sm)}; } .site-title { font-size: 15px; } .logo { height: 24px; } } @media (max-width: 640px) { .actionButton { font-size: 12px; padding: 0 10px; height: 32px; gap: 4px; } .actionButton svg { width: 12px; height: 12px; } .actionButton .button-text { display: none; } .page-title { font-size: 32px; } .page-subtitle { font-size: 16px; } .header-actions { gap: 6px; } } ` ] public render(): TemplateResult { if (this.loading) { return html`
`; } return html`
`; } public dispatchReportNewIncident() { this.dispatchEvent(new CustomEvent('reportNewIncident', { })) } public dispatchStatusSubscribe() { this.dispatchEvent(new CustomEvent('statusSubscribe', { })) } }