/* point.free — newsletter subscribe card.
 *
 * Floating bottom-right card injected by /static/newsletter.js. The card is
 * deliberately quiet: hairline border, subtle blurred background, monospaced
 * label and button — the same vocabulary as .home-byline, .home-section-label,
 * and .home-elsewhere so it reads as part of the site rather than a SaaS
 * widget bolted on.
 *
 * Lifecycle (state is in localStorage, see newsletter.js):
 *   first visit          → shown after dwell + scroll thresholds
 *   user dismisses       → suppressed for ~14 days, then re-shown
 *   user subscribes      → suppressed forever (double opt-in handled by Plunk)
 *
 * Markup contract — newsletter.js builds exactly this:
 *
 *   <aside id="newsletter-card" class="newsletter-card" aria-live="polite">
 *     <button class="newsletter-close" aria-label="Dismiss">×</button>
 *     <p class="newsletter-eyebrow">Newsletter</p>
 *     <h3 class="newsletter-title">Quiet notes, sent rarely.</h3>
 *     <p class="newsletter-desc">…</p>
 *     <form class="newsletter-form" novalidate>
 *       <label class="visually-hidden" for="newsletter-email">Email</label>
 *       <input id="newsletter-email" type="email" required placeholder="your@email.com" />
 *       <input id="newsletter-trap" type="text" tabindex="-1" autocomplete="off" />
 *       <button type="submit" class="newsletter-submit">Subscribe</button>
 *     </form>
 *     <p class="newsletter-status" role="status"></p>
 *   </aside>
 *
 * The honeypot input lives inside the form but is positioned off-screen
 * (not display:none, because some bots skip display:none fields). Real
 * users never see it; bots fill it and we silently swallow the submit.
 */

.newsletter-card {
	position: fixed;
	right: 1.25rem;
	bottom: 1.25rem;
	z-index: 50;
	width: min(22rem, calc(100vw - 2.5rem));
	padding: 1.25rem 1.25rem 1rem;
	background: var(--bg-color);
	border: 1px solid var(--fg-muted-2);
	border-radius: 0.6rem;
	box-shadow: 0 0.5rem 2rem rgba(0, 0, 0, 0.18);
	/* Slide-up entrance — disabled for users who asked us not to move things. */
	opacity: 0;
	transform: translateY(0.5rem);
	transition: opacity 280ms ease, transform 280ms ease;
}

.newsletter-card.is-visible {
	opacity: 1;
	transform: translateY(0);
}

@media (prefers-reduced-motion: reduce) {
	.newsletter-card {
		transition: none;
		transform: none;
	}
}

/* On very narrow viewports the card spans the full width minus the same
 * 1rem-per-side gutter the nav and main content use. Anchoring it to the
 * bottom corner on phones is awkward; centring it keeps it tidy. */
@media (max-width: 30rem) {
	.newsletter-card {
		left: 1rem;
		right: 1rem;
		bottom: 1rem;
		width: auto;
	}
}

.newsletter-eyebrow {
	font-family: monospace;
	font-size: 0.72rem;
	letter-spacing: 0.18em;
	text-transform: uppercase;
	color: var(--accent-color);
	margin: 0 0 0.5rem;
}

.newsletter-title {
	font-family: monospace;
	font-size: 1.05rem;
	letter-spacing: 0.01em;
	line-height: 1.3;
	margin: 0 0 0.5rem;
	color: var(--fg-color);
	font-weight: 600;
}

.newsletter-desc {
	font-size: 0.88rem;
	line-height: 1.5;
	color: var(--fg-muted-5);
	margin: 0 0 0.85rem;
}

.newsletter-form {
	display: flex;
	gap: 0.4rem;
	align-items: stretch;
	margin: 0;
}

.newsletter-form input[type="email"] {
	flex: 1 1 auto;
	min-width: 0;
	padding: 0.45rem 0.65rem;
	font-family: inherit;
	font-size: 0.9rem;
	color: var(--fg-color);
	background: var(--bg-color);
	border: 1px solid var(--fg-muted-2);
	border-radius: 0.35rem;
	transition: border-color 120ms ease, box-shadow 120ms ease;
}

.newsletter-form input[type="email"]:focus {
	outline: none;
	border-color: var(--accent-color);
	box-shadow: 0 0 0 2px color-mix(in srgb, var(--accent-color) 25%, transparent);
}

/* The honeypot. display:none is skipped by some bots; visually-hide it
 * the same way we do for screen-reader-only content so it's reachable
 * by automation but invisible to real users. The input must not be
 * `required` (which it isn't — set in JS) so its emptiness doesn't
 * trip browser validation. */
.newsletter-form .newsletter-trap {
	position: absolute !important;
	width: 1px !important;
	height: 1px !important;
	padding: 0 !important;
	margin: -1px !important;
	overflow: hidden !important;
	clip: rect(0, 0, 0, 0) !important;
	white-space: nowrap !important;
	border: 0 !important;
}

.newsletter-submit {
	flex: 0 0 auto;
	padding: 0.45rem 0.85rem;
	font-family: monospace;
	font-size: 0.85rem;
	letter-spacing: 0.04em;
	color: var(--bg-color);
	background: var(--accent-color);
	border: 1px solid var(--accent-color);
	border-radius: 0.35rem;
	cursor: pointer;
	transition: filter 120ms ease, transform 120ms ease;
}

.newsletter-submit:hover,
.newsletter-submit:focus-visible {
	filter: brightness(1.1);
	outline: none;
}

.newsletter-submit:active {
	transform: translateY(1px);
}

.newsletter-submit[disabled] {
	opacity: 0.55;
	cursor: progress;
}

.newsletter-status {
	min-height: 1.2em;
	margin: 0.65rem 0 0;
	font-family: monospace;
	font-size: 0.78rem;
	letter-spacing: 0.04em;
	color: var(--fg-muted-5);
}

.newsletter-status.is-error {
	color: #c93b3b;
}

.newsletter-status.is-success {
	color: var(--accent-color);
}

.newsletter-close {
	position: absolute;
	top: 0.35rem;
	right: 0.5rem;
	width: 1.75rem;
	height: 1.75rem;
	padding: 0;
	font-size: 1.1rem;
	line-height: 1;
	color: var(--fg-muted-5);
	background: transparent;
	border: 0;
	border-radius: 0.25rem;
	cursor: pointer;
	transition: color 120ms ease, background-color 120ms ease;
}

.newsletter-close:hover,
.newsletter-close:focus-visible {
	color: var(--fg-color);
	background-color: var(--fg-muted-1);
	outline: none;
}

/* Reused utility (also defined in custom.css). Kept here so this file
 * is self-contained if custom.css ever stops loading first. */
.newsletter-card .visually-hidden {
	position: absolute;
	width: 1px;
	height: 1px;
	padding: 0;
	margin: -1px;
	overflow: hidden;
	clip: rect(0, 0, 0, 0);
	white-space: nowrap;
	border: 0;
}
