@duffcloudservices/cms 0.3.15 → 0.3.17

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/composables/useTextContent.ts","../src/composables/useSEO.ts","../src/composables/useReleaseNotes.ts","../src/composables/useSiteVersion.ts","../src/composables/useMediaCarousel.ts","../src/composables/useResponsiveImage.ts","../src/composables/useReviewContent.ts"],"names":["computed","getEnvVar","ref","onMounted","CACHE_TTL"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * useTextContent Composable\n *\n * Provides text content management with build-time injection support and optional\n * runtime API overrides for DCS-managed customer sites.\n *\n * Content resolution order:\n * 1. Runtime API overrides (premium tier only, if mode is 'runtime')\n * 2. Build-time content from .dcs/content.yaml (injected via dcsContentPlugin)\n * 3. Hardcoded defaults passed to the composable\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useTextContent } from '@duffcloudservices/cms'\n *\n * const { t, isLoading, error } = useTextContent({\n * pageSlug: 'home',\n * defaults: {\n * 'hero.title': 'Welcome to Our Site',\n * 'hero.subtitle': 'Build something amazing',\n * 'cta.primary': 'Get Started'\n * }\n * })\n * </script>\n *\n * <template>\n * <h1>{{ t('hero.title') }}</h1>\n * <p>{{ t('hero.subtitle') }}</p>\n * <button>{{ t('cta.primary') }}</button>\n * </template>\n * ```\n */\n\nimport { ref, computed, readonly, onMounted, type Ref } from 'vue'\nimport type { DcsContentFile, TextContentConfig, TextContentReturn } from '../types/content'\n\n// Declare the global injected by dcsContentPlugin\ndeclare const __DCS_CONTENT__: DcsContentFile | undefined\n\n// Simple in-memory cache for runtime fetches\nconst fetchCache = new Map<string, { data: Record<string, string>; expiresAt: number }>()\n\n/**\n * Safely get build-time content configuration.\n * Returns undefined if not available (no content.yaml or plugin not configured).\n */\nfunction getBuildTimeContent(): DcsContentFile | undefined {\n try {\n if (typeof __DCS_CONTENT__ !== 'undefined' && __DCS_CONTENT__ !== null) {\n return __DCS_CONTENT__\n }\n } catch {\n // __DCS_CONTENT__ not defined - that's fine, use defaults\n }\n return undefined\n}\n\n/**\n * Get build-time content for a specific page, merging global and page-specific content.\n */\nfunction getBuildTimePageContent(pageSlug: string): Record<string, string> {\n const content = getBuildTimeContent()\n if (!content) return {}\n\n const global = content.global ?? {}\n const page = content.pages?.[pageSlug] ?? {}\n\n return { ...global, ...page }\n}\n\n/**\n * Get environment variable value, handling both Vite and process.env patterns.\n */\nfunction getEnvVar(key: string, defaultValue = ''): string {\n try {\n // Vite pattern\n if (typeof import.meta !== 'undefined' && import.meta.env) {\n const value = (import.meta.env as Record<string, string | undefined>)[key]\n if (value !== undefined) return value\n }\n } catch {\n // import.meta not available\n }\n\n try {\n // Node.js pattern\n if (typeof process !== 'undefined' && process.env) {\n const value = process.env[key]\n if (value !== undefined) return value\n }\n } catch {\n // process not available\n }\n\n return defaultValue\n}\n\n/**\n * useTextContent composable for DCS-managed text content.\n *\n * @param config - Configuration object\n * @returns Text content helpers and state\n */\nexport function useTextContent(config: TextContentConfig): TextContentReturn {\n const {\n pageSlug,\n defaults,\n fetchOnMount = true,\n cacheKey,\n cacheTtl = 60000,\n } = config\n\n // Get configuration from environment\n const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', '')\n const siteSlug = getEnvVar('VITE_SITE_SLUG', '')\n const textOverrideMode = getEnvVar('VITE_TEXT_OVERRIDE_MODE', 'commit')\n const mode: 'commit' | 'runtime' = textOverrideMode === 'runtime' ? 'runtime' : 'commit'\n\n // Get build-time content immediately (synchronous)\n const buildTimeContent = getBuildTimePageContent(pageSlug)\n const hasBuildTimeContent = Object.keys(buildTimeContent).length > 0\n\n // State - initialize with build-time content\n const overrides = ref<Record<string, string>>({ ...buildTimeContent })\n const isLoading = ref(false)\n const error = ref<string | null>(null)\n\n // Computed merged texts (defaults + overrides)\n const texts = computed(() => ({ ...defaults, ...overrides.value }))\n\n /**\n * Get text by key with optional fallback.\n * Resolution order: overrides → defaults → fallback → key\n */\n function t(key: string, fallback?: string): string {\n return overrides.value[key] ?? defaults[key] ?? fallback ?? key\n }\n\n /**\n * Check if a key has an override (from build-time or runtime).\n */\n function hasOverride(key: string): boolean {\n return key in overrides.value\n }\n\n /**\n * Get an array of objects from indexed content keys.\n * Useful for lists where keys follow the pattern: arrayKey.index.property\n * Example: features.1.title, features.1.description, features.2.title, etc.\n *\n * @param arrayKey - The base key prefix (e.g., 'features', 'items')\n * @returns Array of objects with properties extracted from matching keys, sorted by index\n *\n * @example\n * ```ts\n * // Given content keys:\n * // positions.1.title = \"Software Engineer\"\n * // positions.1.description = \"Build cool stuff\"\n * // positions.2.title = \"Designer\"\n * // positions.2.description = \"Design cool stuff\"\n *\n * const positions = getArray('positions')\n * // Returns:\n * // [\n * // { _index: 1, title: \"Software Engineer\", description: \"Build cool stuff\" },\n * // { _index: 2, title: \"Designer\", description: \"Design cool stuff\" }\n * // ]\n * ```\n */\n function getArray(arrayKey: string): Array<Record<string, unknown> & { _index: number }> {\n const items: Record<number, Record<string, unknown> & { _index: number }> = {}\n const source = { ...defaults, ...overrides.value }\n\n Object.keys(source).forEach((key) => {\n if (key.startsWith(`${arrayKey}.`)) {\n const parts = key.split('.')\n // Format: arrayKey.index.property (or arrayKey.index.nested.property)\n // Example: features.1.title -> index=1, prop=title\n if (parts.length >= 3) {\n const index = Number.parseInt(parts[1], 10)\n const prop = parts.slice(2).join('.')\n\n if (!Number.isNaN(index)) {\n if (!items[index]) items[index] = { _index: index }\n items[index][prop] = source[key]\n }\n }\n }\n })\n\n return Object.values(items).sort((a, b) => a._index - b._index)\n }\n\n /**\n * Fetch runtime overrides from the API.\n * Only runs if mode is 'runtime' and API is configured.\n */\n async function fetchOverrides(): Promise<void> {\n // Skip API fetch in commit mode - use build-time content only\n if (mode !== 'runtime') {\n return\n }\n\n // Skip if API not configured\n if (!apiBaseUrl || !siteSlug) {\n console.warn(\n '[@duffcloudservices/cms] Runtime mode enabled but VITE_API_BASE_URL or VITE_SITE_SLUG not set'\n )\n return\n }\n\n // Check cache first\n const effectiveCacheKey = cacheKey ?? `${siteSlug}:${pageSlug}`\n const cached = fetchCache.get(effectiveCacheKey)\n if (cached && cached.expiresAt > Date.now()) {\n overrides.value = { ...buildTimeContent, ...cached.data }\n return\n }\n\n isLoading.value = true\n error.value = null\n\n try {\n const url = `${apiBaseUrl}/api/v1/sites/${siteSlug}/pages/${pageSlug}/text`\n const response = await fetch(url, {\n headers: {\n Accept: 'application/json',\n },\n })\n\n if (!response.ok) {\n if (response.status === 404) {\n // No overrides for this page - that's fine\n return\n }\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const data = await response.json()\n const apiOverrides = data.overrides ?? data.texts ?? {}\n\n // Cache the result\n fetchCache.set(effectiveCacheKey, {\n data: apiOverrides,\n expiresAt: Date.now() + cacheTtl,\n })\n\n // Merge: build-time content is baseline, API overrides on top\n overrides.value = { ...buildTimeContent, ...apiOverrides }\n } catch (e) {\n error.value = e instanceof Error ? e.message : 'Failed to load text overrides'\n console.error('[@duffcloudservices/cms] Failed to fetch text overrides:', e)\n } finally {\n isLoading.value = false\n }\n }\n\n /**\n * Manually refresh overrides.\n * In commit mode, resets to build-time content.\n * In runtime mode, fetches fresh data from API.\n */\n async function refresh(): Promise<void> {\n if (mode !== 'runtime') {\n // In commit mode, just reset to build-time content\n overrides.value = { ...buildTimeContent }\n return\n }\n\n // Clear cache for this key\n const effectiveCacheKey = cacheKey ?? `${siteSlug}:${pageSlug}`\n fetchCache.delete(effectiveCacheKey)\n\n await fetchOverrides()\n }\n\n // Fetch on mount if enabled and in runtime mode (browser only)\n if (fetchOnMount && mode === 'runtime' && globalThis.window !== undefined) {\n onMounted(() => {\n fetchOverrides()\n })\n }\n\n return {\n t,\n getArray,\n texts,\n overrides,\n isLoading: readonly(isLoading) as Ref<boolean>,\n error: readonly(error) as Ref<string | null>,\n refresh,\n hasOverride,\n hasBuildTimeContent,\n mode,\n }\n}\n","/**\n * useSEO Composable\n *\n * Provides SEO configuration with build-time injection support from .dcs/seo.yaml.\n * Generates meta tags, Open Graph, Twitter Cards, and JSON-LD structured data.\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useSEO } from '@duffcloudservices/cms'\n *\n * const { applyHead, getSchema, config } = useSEO('home')\n *\n * // Apply all meta tags\n * applyHead()\n *\n * // Or customize before applying\n * applyHead({\n * title: 'Custom Override Title',\n * schemas: [...getSchema(), customSchema]\n * })\n * </script>\n * ```\n */\n\nimport { computed, type ComputedRef } from 'vue'\nimport { useHead } from '@unhead/vue'\nimport type {\n SeoConfiguration,\n GlobalSeoConfig,\n SeoOpenGraphConfig,\n SeoTwitterConfig,\n SeoSchemaConfig,\n ResolvedPageSeo,\n UseSeoReturn,\n HeadOverrides,\n} from '../types/seo'\n\n// Declare the global injected by dcsSeoPlugin\ndeclare const __DCS_SEO__: SeoConfiguration | undefined\n\n/**\n * Safely get build-time SEO configuration.\n * Returns undefined if not available (no seo.yaml or plugin not configured).\n */\nfunction getBuildTimeSeo(): SeoConfiguration | undefined {\n try {\n if (typeof __DCS_SEO__ !== 'undefined' && __DCS_SEO__ !== null) {\n return __DCS_SEO__\n }\n } catch {\n // __DCS_SEO__ not defined - that's fine, use defaults\n }\n return undefined\n}\n\n// =============================================================================\n// Meta Tag Generation Utilities\n// =============================================================================\n\ninterface HeadInput {\n title?: string\n titleTemplate?: string | ((title: string) => string)\n meta?: Array<{ name?: string; property?: string; content: string }>\n link?: Array<{ rel: string; href: string; hreflang?: string }>\n script?: Array<{ type: string; children: string }>\n}\n\n/**\n * Generate Open Graph meta tags from config\n */\nfunction generateOpenGraphMeta(\n og: SeoOpenGraphConfig,\n global: GlobalSeoConfig,\n pageTitle: string,\n pageDescription: string,\n canonical: string\n): Array<{ property: string; content: string }> {\n const tags: Array<{ property: string; content: string }> = []\n\n tags.push({ property: 'og:title', content: og.title || pageTitle })\n tags.push({ property: 'og:description', content: og.description || pageDescription })\n tags.push({ property: 'og:url', content: og.url || canonical })\n tags.push({ property: 'og:type', content: og.type || 'website' })\n\n const image = og.image || global.images?.ogDefault\n if (image) {\n tags.push({ property: 'og:image', content: image })\n if (og.imageAlt || pageTitle) {\n tags.push({ property: 'og:image:alt', content: og.imageAlt || pageTitle })\n }\n if (og.imageWidth) {\n tags.push({ property: 'og:image:width', content: String(og.imageWidth) })\n }\n if (og.imageHeight) {\n tags.push({ property: 'og:image:height', content: String(og.imageHeight) })\n }\n }\n\n if (global.siteName) {\n tags.push({ property: 'og:site_name', content: global.siteName })\n }\n\n if (global.locale) {\n tags.push({ property: 'og:locale', content: global.locale })\n }\n\n // Article-specific tags\n if (og.type === 'article') {\n if (og.publishedTime) {\n tags.push({ property: 'article:published_time', content: og.publishedTime })\n }\n if (og.modifiedTime) {\n tags.push({ property: 'article:modified_time', content: og.modifiedTime })\n }\n if (og.author) {\n tags.push({ property: 'article:author', content: og.author })\n }\n if (og.section) {\n tags.push({ property: 'article:section', content: og.section })\n }\n if (og.tags) {\n og.tags.forEach((tag) => {\n tags.push({ property: 'article:tag', content: tag })\n })\n }\n }\n\n return tags\n}\n\n/**\n * Generate Twitter Card meta tags from config\n */\nfunction generateTwitterMeta(\n twitter: SeoTwitterConfig,\n global: GlobalSeoConfig,\n pageTitle: string,\n pageDescription: string\n): Array<{ name: string; content: string }> {\n const tags: Array<{ name: string; content: string }> = []\n\n tags.push({ name: 'twitter:card', content: twitter.card || 'summary_large_image' })\n tags.push({ name: 'twitter:title', content: twitter.title || pageTitle })\n tags.push({ name: 'twitter:description', content: twitter.description || pageDescription })\n\n const image = twitter.image || global.images?.twitterDefault\n if (image) {\n tags.push({ name: 'twitter:image', content: image })\n if (twitter.imageAlt || pageTitle) {\n tags.push({ name: 'twitter:image:alt', content: twitter.imageAlt || pageTitle })\n }\n }\n\n const site = twitter.site || global.social?.twitter\n if (site) {\n tags.push({ name: 'twitter:site', content: site.startsWith('@') ? site : `@${site}` })\n }\n\n if (twitter.creator) {\n tags.push({\n name: 'twitter:creator',\n content: twitter.creator.startsWith('@') ? twitter.creator : `@${twitter.creator}`,\n })\n }\n\n return tags\n}\n\n/**\n * Generate JSON-LD script content from schemas\n */\nfunction generateJsonLd(schemas: SeoSchemaConfig[], global: GlobalSeoConfig): object[] {\n return schemas.map((schema) => {\n const base: Record<string, unknown> = {\n '@context': 'https://schema.org',\n '@type': schema.type,\n }\n\n // Merge properties\n if (schema.properties) {\n Object.assign(base, schema.properties)\n }\n\n // Auto-populate common properties from global config\n if (schema.type === 'WebSite' && global.siteUrl && !base.url) {\n base.url = global.siteUrl\n }\n if (schema.type === 'WebSite' && global.siteName && !base.name) {\n base.name = global.siteName\n }\n\n return base\n })\n}\n\n/**\n * Resolve page SEO by merging global defaults with page-specific config\n */\nfunction resolvePageSeo(\n pageSlug: string,\n pagePath: string | undefined,\n seoConfig: SeoConfiguration | undefined\n): ResolvedPageSeo {\n const global = seoConfig?.global ?? {}\n const page = seoConfig?.pages?.[pageSlug] ?? {}\n\n // Build canonical URL\n let canonical = page.canonical || ''\n if (!canonical && global.siteUrl) {\n const path = pagePath ?? (pageSlug === 'home' ? '/' : `/${pageSlug}`)\n canonical = `${global.siteUrl.replace(/\\/$/, '')}${path}`\n }\n\n // Build title\n let title = page.title || global.defaultTitle || pageSlug\n if (!page.noTitleTemplate && global.titleTemplate) {\n title = global.titleTemplate.replace('%s', title)\n }\n\n // Merge Open Graph\n const openGraph: ResolvedPageSeo['openGraph'] = {\n type: page.openGraph?.type || 'website',\n title: page.openGraph?.title || page.title || global.defaultTitle || '',\n description: page.openGraph?.description || page.description || global.defaultDescription || '',\n ...page.openGraph,\n }\n\n // Merge Twitter\n const twitter: ResolvedPageSeo['twitter'] = {\n card: page.twitter?.card || 'summary_large_image',\n ...page.twitter,\n }\n\n // Combine schemas (global + page)\n const schemas = [...(global.schemas ?? []), ...(page.schemas ?? [])]\n\n return {\n title,\n description: page.description || global.defaultDescription || '',\n canonical,\n robots: page.robots || global.robots || 'index, follow',\n openGraph,\n twitter,\n schemas,\n alternates: page.alternates ?? [],\n }\n}\n\n/**\n * useSEO composable for DCS-managed SEO configuration.\n *\n * @param pageSlug - Page slug matching entry in seo.yaml\n * @param pagePath - Optional page path for canonical URL generation\n * @returns SEO helpers and state\n */\nexport function useSEO(pageSlug: string, pagePath?: string): UseSeoReturn {\n const seoConfig = getBuildTimeSeo()\n const hasBuildTimeSeo = seoConfig !== undefined\n\n // Computed resolved config\n const config: ComputedRef<ResolvedPageSeo> = computed(() =>\n resolvePageSeo(pageSlug, pagePath, seoConfig)\n )\n\n /**\n * Get JSON-LD schema objects for the page\n */\n function getSchema(): object[] {\n return generateJsonLd(config.value.schemas, seoConfig?.global ?? {})\n }\n\n /**\n * Get canonical URL for the page\n */\n function getCanonical(): string {\n return config.value.canonical\n }\n\n /**\n * Apply all meta tags via useHead\n */\n function applyHead(overrides?: HeadOverrides): void {\n const resolved = config.value\n const global = seoConfig?.global ?? {}\n\n const title = overrides?.title ?? resolved.title\n const description = overrides?.description ?? resolved.description\n\n // Build meta tags\n const meta: HeadInput['meta'] = []\n\n // Basic meta\n meta.push({ name: 'description', content: description })\n if (resolved.robots) {\n meta.push({ name: 'robots', content: resolved.robots })\n }\n\n // Verification codes\n if (global.verification?.google) {\n meta.push({ name: 'google-site-verification', content: global.verification.google })\n }\n if (global.verification?.bing) {\n meta.push({ name: 'msvalidate.01', content: global.verification.bing })\n }\n\n // Open Graph\n const ogMeta = generateOpenGraphMeta(\n resolved.openGraph,\n global,\n title,\n description,\n resolved.canonical\n )\n meta.push(...ogMeta.map((t) => ({ property: t.property, content: t.content })))\n\n // Twitter\n const twitterMeta = generateTwitterMeta(resolved.twitter, global, title, description)\n meta.push(...twitterMeta.map((t) => ({ name: t.name, content: t.content })))\n\n // Additional overrides\n if (overrides?.meta) {\n meta.push(...overrides.meta)\n }\n\n // Build links\n const link: HeadInput['link'] = []\n\n // Canonical\n if (resolved.canonical) {\n link.push({ rel: 'canonical', href: resolved.canonical })\n }\n\n // Alternate languages\n resolved.alternates.forEach((alt) => {\n link.push({ rel: 'alternate', href: alt.href, hreflang: alt.hreflang })\n })\n\n // Build scripts (JSON-LD)\n const schemas = overrides?.schemas ?? getSchema()\n const script: HeadInput['script'] = schemas.map((schema) => ({\n type: 'application/ld+json',\n children: JSON.stringify(schema),\n }))\n\n // Apply via useHead\n useHead({\n title,\n meta,\n link,\n script,\n })\n }\n\n return {\n config,\n applyHead,\n getSchema,\n getCanonical,\n hasBuildTimeSeo,\n }\n}\n\n/**\n * Create a typed useSEO function with site-specific defaults.\n * Useful for creating a site-wide wrapper.\n *\n * @example\n * ```ts\n * // composables/useSiteSeo.ts\n * import { createSiteSEO } from '@duffcloudservices/cms'\n *\n * export const useSiteSeo = createSiteSEO({\n * siteName: 'My Site',\n * siteUrl: 'https://example.com'\n * })\n * ```\n */\nexport function createSiteSEO(\n _siteDefaults: Partial<GlobalSeoConfig>\n): (pageSlug: string, pagePath?: string) => UseSeoReturn {\n return function siteUseSEO(pageSlug: string, pagePath?: string): UseSeoReturn {\n // Note: siteDefaults would be used if we needed to override at runtime\n // but build-time injection handles this via dcsSeoPlugin\n return useSEO(pageSlug, pagePath)\n }\n}\n","/**\n * useReleaseNotes Composable\n *\n * Fetches and displays versioned release notes from the DCS Portal API.\n * Supports fetching specific versions or the latest release.\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useReleaseNotes } from '@duffcloudservices/cms'\n * import { useRoute } from 'vue-router'\n *\n * const route = useRoute()\n * const version = route.params.version as string || 'latest'\n *\n * const { releaseNote, isLoading, error } = useReleaseNotes(version)\n * </script>\n *\n * <template>\n * <div v-if=\"isLoading\">Loading...</div>\n * <div v-else-if=\"error\">{{ error }}</div>\n * <article v-else-if=\"releaseNote\">\n * <h1>{{ releaseNote.title }}</h1>\n * <p>{{ releaseNote.summary }}</p>\n * <div v-html=\"renderedMarkdown\" />\n * </article>\n * </template>\n * ```\n */\n\nimport { ref, onMounted } from 'vue'\nimport type { ReleaseNote, ReleaseNotesReturn } from '../types/release-notes'\n\n/**\n * Get environment variable value.\n */\nfunction getEnvVar(key: string, defaultValue = ''): string {\n try {\n if (typeof import.meta !== 'undefined' && import.meta.env) {\n const value = (import.meta.env as Record<string, string | undefined>)[key]\n if (value !== undefined) return value\n }\n } catch {\n // import.meta not available\n }\n return defaultValue\n}\n\n// Simple cache for release notes\nconst releaseNotesCache = new Map<string, { data: ReleaseNote; expiresAt: number }>()\nconst CACHE_TTL = 5 * 60 * 1000 // 5 minutes\n\n/**\n * useReleaseNotes composable for fetching release notes from the DCS API.\n *\n * @param version - Semantic version (e.g., \"1.2.0\") or \"latest\"\n * @param options - Optional configuration\n * @returns Release notes data and state\n */\nexport function useReleaseNotes(\n version: string,\n options: { fetchOnMount?: boolean } = {}\n): ReleaseNotesReturn {\n const { fetchOnMount = true } = options\n\n const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', 'https://portal.duffcloudservices.com')\n const siteSlug = getEnvVar('VITE_SITE_SLUG', '')\n\n const releaseNote = ref<ReleaseNote | null>(null)\n const isLoading = ref(false)\n const error = ref<string | null>(null)\n\n async function fetchReleaseNotes(): Promise<void> {\n if (!siteSlug) {\n error.value = 'VITE_SITE_SLUG environment variable not set'\n return\n }\n\n // Check cache\n const cacheKey = `${siteSlug}:${version}`\n const cached = releaseNotesCache.get(cacheKey)\n if (cached && cached.expiresAt > Date.now()) {\n releaseNote.value = cached.data\n return\n }\n\n isLoading.value = true\n error.value = null\n\n try {\n const url = `${apiBaseUrl}/api/v1/sites/${siteSlug}/release-notes/${version}`\n const response = await fetch(url, {\n headers: {\n Accept: 'application/json',\n },\n })\n\n if (!response.ok) {\n if (response.status === 404) {\n error.value = `Release notes for version ${version} not found`\n return\n }\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const data = await response.json()\n\n // Normalize the response\n const note: ReleaseNote = {\n version: data.version,\n title: data.title,\n summary: data.summary || '',\n notesMarkdown: data.notesMarkdown || data.notes || '',\n changeCount: data.changeCount || 0,\n releaseDate: data.releaseDate || data.releasedAt || '',\n }\n\n // Cache the result\n releaseNotesCache.set(cacheKey, {\n data: note,\n expiresAt: Date.now() + CACHE_TTL,\n })\n\n releaseNote.value = note\n } catch (e) {\n error.value = e instanceof Error ? e.message : 'Failed to load release notes'\n console.error('[@duffcloudservices/cms] Failed to fetch release notes:', e)\n } finally {\n isLoading.value = false\n }\n }\n\n async function refresh(): Promise<void> {\n // Clear cache\n const cacheKey = `${siteSlug}:${version}`\n releaseNotesCache.delete(cacheKey)\n await fetchReleaseNotes()\n }\n\n // Fetch on mount if enabled\n if (fetchOnMount && typeof window !== 'undefined') {\n onMounted(() => {\n fetchReleaseNotes()\n })\n }\n\n return {\n releaseNote,\n isLoading,\n error,\n refresh,\n }\n}\n","/**\n * useSiteVersion Composable\n *\n * Gets the current site version for footer badges and version displays.\n * Fetches the latest release version from the DCS Portal API.\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useSiteVersion } from '@duffcloudservices/cms'\n *\n * const { version, releaseNotesUrl } = useSiteVersion()\n * </script>\n *\n * <template>\n * <footer>\n * <a v-if=\"version\" :href=\"releaseNotesUrl\" class=\"version-badge\">\n * v{{ version }}\n * </a>\n * </footer>\n * </template>\n * ```\n */\n\nimport { ref, computed, onMounted } from 'vue'\nimport type { SiteVersionReturn } from '../types/release-notes'\n\n/**\n * Get environment variable value.\n */\nfunction getEnvVar(key: string, defaultValue = ''): string {\n try {\n if (typeof import.meta !== 'undefined' && import.meta.env) {\n const value = (import.meta.env as Record<string, string | undefined>)[key]\n if (value !== undefined) return value\n }\n } catch {\n // import.meta not available\n }\n return defaultValue\n}\n\n// Cache for site version\nlet versionCache: { version: string; expiresAt: number } | null = null\nconst CACHE_TTL = 10 * 60 * 1000 // 10 minutes\n\n/**\n * useSiteVersion composable for displaying the current site version.\n *\n * @param options - Optional configuration\n * @returns Site version data and computed URL\n */\nexport function useSiteVersion(options: { fetchOnMount?: boolean } = {}): SiteVersionReturn {\n const { fetchOnMount = true } = options\n\n const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', 'https://portal.duffcloudservices.com')\n const siteSlug = getEnvVar('VITE_SITE_SLUG', '')\n\n const version = ref<string | null>(null)\n const isLoading = ref(false)\n\n const releaseNotesUrl = computed(() => {\n if (!version.value) return '/releaseNotes/latest'\n return `/releaseNotes/${version.value}`\n })\n\n async function fetchVersion(): Promise<void> {\n if (!siteSlug) {\n return\n }\n\n // Check cache\n if (versionCache && versionCache.expiresAt > Date.now()) {\n version.value = versionCache.version\n return\n }\n\n isLoading.value = true\n\n try {\n // Fetch the latest release notes to get the version\n const url = `${apiBaseUrl}/api/v1/sites/${siteSlug}/release-notes/latest`\n const response = await fetch(url, {\n headers: {\n Accept: 'application/json',\n },\n })\n\n if (!response.ok) {\n // No release notes yet - that's fine\n return\n }\n\n const data = await response.json()\n const latestVersion = data.version\n\n // Cache the result\n versionCache = {\n version: latestVersion,\n expiresAt: Date.now() + CACHE_TTL,\n }\n\n version.value = latestVersion\n } catch (e) {\n console.error('[@duffcloudservices/cms] Failed to fetch site version:', e)\n } finally {\n isLoading.value = false\n }\n }\n\n // Fetch on mount if enabled\n if (fetchOnMount && typeof window !== 'undefined') {\n onMounted(() => {\n fetchVersion()\n })\n }\n\n return {\n version,\n isLoading,\n releaseNotesUrl,\n }\n}\n","/**\n * useMediaCarousel Composable\n *\n * Extracts media carousel items from text content keys following the pattern:\n * `{prefix}.{N}.url`, `{prefix}.{N}.type`, `{prefix}.{N}.alt`\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useTextContent, useMediaCarousel } from '@duffcloudservices/cms'\n *\n * const { t } = useTextContent({\n * pageSlug: 'bio-mackenzie',\n * defaults: {\n * 'hero.media-carousel.0.url': '/images/staff/mackenzie.webp',\n * 'hero.media-carousel.0.type': 'image',\n * 'hero.media-carousel.0.alt': 'Mackenzie Kowalick',\n * 'hero.media-carousel.1.url': '/videos/intro.mp4',\n * 'hero.media-carousel.1.type': 'video',\n * 'hero.media-carousel.1.alt': 'Introduction video',\n * }\n * })\n *\n * const { items } = useMediaCarousel({\n * prefix: 'hero.media-carousel',\n * t,\n * defaults: [\n * { url: '/images/staff/mackenzie.webp', type: 'image', alt: 'Mackenzie Kowalick' }\n * ]\n * })\n * </script>\n *\n * <template>\n * <MediaCarousel :items=\"items\" />\n * </template>\n * ```\n */\n\nimport { computed, type ComputedRef } from 'vue'\nimport { isCdnAssetUrl } from '@duffcloudservices/cms-core'\n\n/**\n * Media carousel item representing an image, video, or embed\n */\nexport interface MediaCarouselItem {\n /** URL to the image, video file, or embed URL */\n url: string\n /** Type of media: 'image', 'video' (direct file), 'youtube', or 'instagram' */\n type: 'image' | 'video' | 'youtube' | 'instagram'\n /** Accessibility alt text */\n alt?: string\n /**\n * Whether this image has responsive CDN variants available.\n * Automatically set to `true` when the URL matches the DCS CDN asset pattern.\n * Components rendering the carousel should use `<ResponsiveImage>` when this is `true`.\n */\n responsive?: boolean\n}\n\n/**\n * Configuration for useMediaCarousel composable\n */\nexport interface UseMediaCarouselConfig {\n /** Key prefix for carousel items (e.g., 'hero.media-carousel') */\n prefix: string\n /** The t() function from useTextContent */\n t: (key: string, fallback?: string) => string\n /** Default items to use if no content keys are found */\n defaults?: MediaCarouselItem[]\n /** Maximum number of items to look for (default: 10) */\n maxItems?: number\n}\n\n/**\n * Return type for useMediaCarousel composable\n */\nexport interface UseMediaCarouselReturn {\n /** Computed array of media carousel items */\n items: ComputedRef<MediaCarouselItem[]>\n /** Whether any items were found from content keys */\n hasItems: ComputedRef<boolean>\n /** Number of items in the carousel */\n count: ComputedRef<number>\n}\n\n/**\n * Extract media carousel items from text content keys.\n *\n * Looks for keys in the format:\n * - `{prefix}.{N}.url` - Required URL for the media\n * - `{prefix}.{N}.type` - Type: 'image' or 'video' (defaults to 'image')\n * - `{prefix}.{N}.alt` - Alt text for accessibility\n *\n * Items are sorted by index (0, 1, etc.) and only included if they have a valid URL.\n *\n * @param config - Configuration object\n * @returns Media carousel helpers and state\n */\nexport function useMediaCarousel(config: UseMediaCarouselConfig): UseMediaCarouselReturn {\n const {\n prefix,\n t,\n defaults = [],\n maxItems = 10,\n } = config\n\n const items = computed<MediaCarouselItem[]>(() => {\n const result: MediaCarouselItem[] = []\n\n // Look for items from 0 to maxItems\n for (let i = 0; i < maxItems; i++) {\n const urlKey = `${prefix}.${i}.url`\n const typeKey = `${prefix}.${i}.type`\n const altKey = `${prefix}.${i}.alt`\n\n // Use a sentinel value to detect if the key exists\n const url = t(urlKey, '')\n \n // Skip if no URL (key doesn't exist or is empty)\n if (!url || url === urlKey) {\n continue\n }\n\n const typeValue = t(typeKey, 'image')\n // Parse type value - support image, video, youtube, instagram\n let type: 'image' | 'video' | 'youtube' | 'instagram' = 'image'\n if (typeValue === 'video') type = 'video'\n else if (typeValue === 'youtube') type = 'youtube'\n else if (typeValue === 'instagram') type = 'instagram'\n const alt = t(altKey, '')\n\n result.push({\n url,\n type,\n alt: alt && alt !== altKey ? alt : undefined,\n // Flag CDN-hosted images as responsive so carousel components\n // can render them with <ResponsiveImage> automatically\n responsive: type === 'image' && isCdnAssetUrl(url),\n })\n }\n\n // If no items found from content keys, use defaults\n if (result.length === 0) {\n return defaults\n }\n\n return result\n })\n\n const hasItems = computed(() => items.value.length > 0)\n const count = computed(() => items.value.length)\n\n return {\n items,\n hasItems,\n count,\n }\n}\n","/**\n * Vue 3 composable that resolves responsive image variants for DCS CDN-hosted assets.\n *\n * Wraps the framework-agnostic `resolveResponsiveImage` from `@duffcloudservices/cms-core`\n * with reactive Vue refs so it can be used directly in `<script setup>` blocks.\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useResponsiveImage } from '@duffcloudservices/cms'\n *\n * const hero = useResponsiveImage({\n * src: 'https://files.duffcloudservices.com/kept/assets/hero/abc-123.jpg',\n * alt: 'Hero image',\n * context: 'hero',\n * })\n * </script>\n *\n * <template>\n * <picture v-if=\"hero.hasVariants\">\n * <source\n * v-for=\"source in hero.sources\"\n * :key=\"source.type\"\n * :srcset=\"source.srcset\"\n * :type=\"source.type\"\n * :sizes=\"source.sizes\"\n * />\n * <img v-bind=\"hero.imgProps\" class=\"w-full h-full object-cover\" />\n * </picture>\n * <img v-else v-bind=\"hero.imgProps\" class=\"w-full h-full object-cover\" />\n * </template>\n * ```\n */\n\nimport { computed, type MaybeRefOrGetter, toValue } from 'vue'\nimport {\n resolveResponsiveImage,\n type ImageContext,\n type ResponsiveImageResult,\n} from '@duffcloudservices/cms-core'\n\nexport interface UseResponsiveImageOptions {\n /** Source URL — can be a reactive ref, getter, or plain string. */\n src: MaybeRefOrGetter<string>\n /** Alt text — can be a reactive ref, getter, or plain string. */\n alt: MaybeRefOrGetter<string>\n /** Sizing context — determines which variants to include. */\n context?: MaybeRefOrGetter<ImageContext | undefined>\n /** Optional `sizes` attribute override. */\n sizes?: MaybeRefOrGetter<string | undefined>\n /** Skip variant resolution and use the original URL only. */\n original?: MaybeRefOrGetter<boolean | undefined>\n}\n\n/**\n * Reactively resolves responsive image metadata for a DCS CDN URL.\n *\n * The returned object is a computed ref that recomputes whenever any\n * of the input refs change. Spread `imgProps` onto an `<img>` or\n * combine with `sources` inside a `<picture>` element.\n */\nexport function useResponsiveImage(options: UseResponsiveImageOptions): ResponsiveImageResult {\n const result = computed(() =>\n resolveResponsiveImage({\n src: toValue(options.src),\n alt: toValue(options.alt),\n context: toValue(options.context),\n sizes: toValue(options.sizes),\n original: toValue(options.original) ?? undefined,\n }),\n )\n\n // Return a reactive proxy that delegates to the computed\n return {\n get imgProps() {\n return result.value.imgProps\n },\n get sources() {\n return result.value.sources\n },\n get hasVariants() {\n return result.value.hasVariants\n },\n }\n}\n","/**\n * Composable for reading curated review selections from DCS content.\n * Reviews are stored in content.yaml by the visual editor's ReviewPickerSheet.\n */\nimport { computed, onMounted, onUnmounted, shallowRef, type ComputedRef } from 'vue'\n\nexport interface ReviewItem {\n id: string\n platform: 'google' | 'meta' | string\n rating: number\n authorName: string\n authorPhotoUrl?: string\n text?: string\n date?: string\n replyText?: string\n locationName?: string\n sourceLocationName?: string\n sourceUrl?: string\n}\n\nfunction withoutAuthorPhotos(items: ReviewItem[]): ReviewItem[] {\n return items.map(item => ({\n ...item,\n authorPhotoUrl: undefined,\n }))\n}\n\nexport interface UseReviewContentConfig {\n /** The section key matching the data-dcs-reviews attribute value */\n sectionKey: string\n /** Page slug for page-specific content lookup (defaults to current page) */\n pageSlug?: string\n /** Fallback reviews when no content is available */\n defaults?: ReviewItem[]\n}\n\nexport interface UseReviewContentReturn {\n /** The curated review items from content */\n reviews: ComputedRef<ReviewItem[]>\n /** Whether any reviews are available */\n hasReviews: ComputedRef<boolean>\n /** Number of reviews */\n count: ComputedRef<number>\n}\n\nconst previewReviewOverrides = shallowRef<Record<string, ReviewItem[]>>({})\nlet activePreviewReviewConsumers = 0\n\n// Declare the global content variable injected by dcsContentPlugin\ndeclare const __DCS_CONTENT__: {\n global?: Record<string, unknown>\n pages?: Record<string, Record<string, unknown>>\n} | undefined\n\nfunction normalizeReviewItem(item: Record<string, unknown>): ReviewItem | null {\n const id = String(item.id ?? '').trim()\n if (!id) {\n return null\n }\n\n const rawRating = Number(item.rating ?? 5)\n const rating = Number.isFinite(rawRating)\n ? Math.min(5, Math.max(1, Math.round(rawRating)))\n : 5\n\n return {\n id,\n platform: String(item.platform ?? 'google'),\n rating,\n authorName: String(item.authorName ?? 'Anonymous'),\n authorPhotoUrl: undefined,\n text: item.text ? String(item.text) : undefined,\n date: item.date ? String(item.date) : undefined,\n replyText: item.replyText ? String(item.replyText) : undefined,\n locationName: item.locationName ? String(item.locationName) : undefined,\n sourceLocationName: item.sourceLocationName ? String(item.sourceLocationName) : undefined,\n sourceUrl: item.sourceUrl ? String(item.sourceUrl) : undefined,\n }\n}\n\nfunction normalizeReviewList(value: unknown): ReviewItem[] {\n if (!Array.isArray(value)) {\n return []\n }\n\n return value\n .filter((item): item is Record<string, unknown> => item != null && typeof item === 'object')\n .map(normalizeReviewItem)\n .filter((item): item is ReviewItem => item != null)\n}\n\nfunction handlePreviewReviewUpdate(event: Event) {\n const detail = event instanceof CustomEvent && event.detail != null && typeof event.detail === 'object'\n ? event.detail as { key?: unknown; reviews?: unknown }\n : null\n const key = typeof detail?.key === 'string' ? detail.key.trim() : ''\n if (!key) {\n return\n }\n\n const next = { ...previewReviewOverrides.value }\n if (Array.isArray(detail?.reviews)) {\n next[key] = normalizeReviewList(detail.reviews)\n } else {\n delete next[key]\n }\n previewReviewOverrides.value = next\n}\n\nexport function useReviewContent(config: UseReviewContentConfig): UseReviewContentReturn {\n const { sectionKey, pageSlug, defaults = [] } = config\n\n onMounted(() => {\n activePreviewReviewConsumers += 1\n if (activePreviewReviewConsumers === 1) {\n window.addEventListener('dcs:reviews-updated', handlePreviewReviewUpdate)\n }\n })\n\n onUnmounted(() => {\n activePreviewReviewConsumers = Math.max(0, activePreviewReviewConsumers - 1)\n if (activePreviewReviewConsumers === 0) {\n window.removeEventListener('dcs:reviews-updated', handlePreviewReviewUpdate)\n }\n })\n\n const reviews = computed<ReviewItem[]>(() => {\n if (Object.prototype.hasOwnProperty.call(previewReviewOverrides.value, sectionKey)) {\n return previewReviewOverrides.value[sectionKey] ?? []\n }\n\n if (typeof __DCS_CONTENT__ === 'undefined' || __DCS_CONTENT__ == null) {\n return withoutAuthorPhotos(defaults)\n }\n\n let reviewData: unknown = null\n\n if (pageSlug && __DCS_CONTENT__.pages?.[pageSlug]) {\n reviewData = __DCS_CONTENT__.pages[pageSlug][`reviews.${sectionKey}.items`]\n ?? __DCS_CONTENT__.pages[pageSlug][`reviews.${sectionKey}`]\n }\n\n if (!reviewData && __DCS_CONTENT__.global) {\n reviewData = __DCS_CONTENT__.global[`reviews.${sectionKey}.items`]\n ?? __DCS_CONTENT__.global[`reviews.${sectionKey}`]\n }\n\n if (!reviewData || !Array.isArray(reviewData)) {\n return withoutAuthorPhotos(defaults)\n }\n\n return normalizeReviewList(reviewData)\n })\n\n const hasReviews = computed(() => reviews.value.length > 0)\n const count = computed(() => reviews.value.length)\n\n return { reviews, hasReviews, count }\n}\n"]}
1
+ {"version":3,"sources":["../src/composables/useTextContent.ts","../src/composables/useSEO.ts","../src/composables/useReleaseNotes.ts","../src/composables/useSiteVersion.ts","../src/composables/useMediaCarousel.ts","../src/composables/useResponsiveImage.ts","../src/composables/useReviewContent.ts"],"names":["computed","getEnvVar","ref","onMounted","CACHE_TTL"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * useTextContent Composable\n *\n * Provides text content management with build-time injection support and optional\n * runtime API overrides for DCS-managed customer sites.\n *\n * Content resolution order:\n * 1. Runtime API overrides (premium tier only, if mode is 'runtime')\n * 2. Build-time content from .dcs/content.yaml (injected via dcsContentPlugin)\n * 3. Hardcoded defaults passed to the composable\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useTextContent } from '@duffcloudservices/cms'\n *\n * const { t, isLoading, error } = useTextContent({\n * pageSlug: 'home',\n * defaults: {\n * 'hero.title': 'Welcome to Our Site',\n * 'hero.subtitle': 'Build something amazing',\n * 'cta.primary': 'Get Started'\n * }\n * })\n * </script>\n *\n * <template>\n * <h1>{{ t('hero.title') }}</h1>\n * <p>{{ t('hero.subtitle') }}</p>\n * <button>{{ t('cta.primary') }}</button>\n * </template>\n * ```\n */\n\nimport { ref, computed, readonly, onMounted, type Ref } from 'vue'\nimport type { DcsContentFile, TextContentConfig, TextContentReturn } from '../types/content'\n\n// Declare the global injected by dcsContentPlugin\ndeclare const __DCS_CONTENT__: DcsContentFile | undefined\n\n// Simple in-memory cache for runtime fetches\nconst fetchCache = new Map<string, { data: Record<string, string>; expiresAt: number }>()\n\n/**\n * Safely get build-time content configuration.\n * Returns undefined if not available (no content.yaml or plugin not configured).\n */\nfunction getBuildTimeContent(): DcsContentFile | undefined {\n try {\n if (typeof __DCS_CONTENT__ !== 'undefined' && __DCS_CONTENT__ !== null) {\n return __DCS_CONTENT__\n }\n } catch {\n // __DCS_CONTENT__ not defined - that's fine, use defaults\n }\n return undefined\n}\n\n/**\n * Get build-time content for a specific page, merging global and page-specific content.\n */\nfunction getBuildTimePageContent(pageSlug: string): Record<string, string> {\n const content = getBuildTimeContent()\n if (!content) return {}\n\n const global = content.global ?? {}\n const page = content.pages?.[pageSlug] ?? {}\n\n return { ...global, ...page }\n}\n\n/**\n * Get environment variable value, handling both Vite and process.env patterns.\n */\nfunction getEnvVar(key: string, defaultValue = ''): string {\n try {\n // Vite pattern\n if (typeof import.meta !== 'undefined' && import.meta.env) {\n const value = (import.meta.env as Record<string, string | undefined>)[key]\n if (value !== undefined) return value\n }\n } catch {\n // import.meta not available\n }\n\n try {\n // Node.js pattern\n if (typeof process !== 'undefined' && process.env) {\n const value = process.env[key]\n if (value !== undefined) return value\n }\n } catch {\n // process not available\n }\n\n return defaultValue\n}\n\n/**\n * useTextContent composable for DCS-managed text content.\n *\n * @param config - Configuration object\n * @returns Text content helpers and state\n */\nexport function useTextContent(config: TextContentConfig): TextContentReturn {\n const {\n pageSlug,\n defaults,\n fetchOnMount = true,\n cacheKey,\n cacheTtl = 60000,\n } = config\n\n // Get configuration from environment.\n // @deprecated VITE_SITE_SLUG: the site is now resolved server-side from the\n // request Host or the dedicated Container App's DCS_SITE_SLUG. It is still read\n // here for cache-key continuity and source compatibility, but is no longer used\n // to build the request URL.\n const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', '')\n const siteSlug = getEnvVar('VITE_SITE_SLUG', '')\n const textOverrideMode = getEnvVar('VITE_TEXT_OVERRIDE_MODE', 'commit')\n const mode: 'commit' | 'runtime' = textOverrideMode === 'runtime' ? 'runtime' : 'commit'\n\n // Get build-time content immediately (synchronous)\n const buildTimeContent = getBuildTimePageContent(pageSlug)\n const hasBuildTimeContent = Object.keys(buildTimeContent).length > 0\n\n // State - initialize with build-time content\n const overrides = ref<Record<string, string>>({ ...buildTimeContent })\n const isLoading = ref(false)\n const error = ref<string | null>(null)\n\n // Computed merged texts (defaults + overrides)\n const texts = computed(() => ({ ...defaults, ...overrides.value }))\n\n /**\n * Get text by key with optional fallback.\n * Resolution order: overrides → defaults → fallback → key\n */\n function t(key: string, fallback?: string): string {\n return overrides.value[key] ?? defaults[key] ?? fallback ?? key\n }\n\n /**\n * Check if a key has an override (from build-time or runtime).\n */\n function hasOverride(key: string): boolean {\n return key in overrides.value\n }\n\n /**\n * Get an array of objects from indexed content keys.\n * Useful for lists where keys follow the pattern: arrayKey.index.property\n * Example: features.1.title, features.1.description, features.2.title, etc.\n *\n * @param arrayKey - The base key prefix (e.g., 'features', 'items')\n * @returns Array of objects with properties extracted from matching keys, sorted by index\n *\n * @example\n * ```ts\n * // Given content keys:\n * // positions.1.title = \"Software Engineer\"\n * // positions.1.description = \"Build cool stuff\"\n * // positions.2.title = \"Designer\"\n * // positions.2.description = \"Design cool stuff\"\n *\n * const positions = getArray('positions')\n * // Returns:\n * // [\n * // { _index: 1, title: \"Software Engineer\", description: \"Build cool stuff\" },\n * // { _index: 2, title: \"Designer\", description: \"Design cool stuff\" }\n * // ]\n * ```\n */\n function getArray(arrayKey: string): Array<Record<string, unknown> & { _index: number }> {\n const items: Record<number, Record<string, unknown> & { _index: number }> = {}\n const source = { ...defaults, ...overrides.value }\n\n Object.keys(source).forEach((key) => {\n if (key.startsWith(`${arrayKey}.`)) {\n const parts = key.split('.')\n // Format: arrayKey.index.property (or arrayKey.index.nested.property)\n // Example: features.1.title -> index=1, prop=title\n if (parts.length >= 3) {\n const index = Number.parseInt(parts[1], 10)\n const prop = parts.slice(2).join('.')\n\n if (!Number.isNaN(index)) {\n if (!items[index]) items[index] = { _index: index }\n items[index][prop] = source[key]\n }\n }\n }\n })\n\n return Object.values(items).sort((a, b) => a._index - b._index)\n }\n\n /**\n * Fetch runtime overrides from the API.\n * Only runs if mode is 'runtime' and API is configured.\n */\n async function fetchOverrides(): Promise<void> {\n // Skip API fetch in commit mode - use build-time content only\n if (mode !== 'runtime') {\n return\n }\n\n // Skip if API not configured. The site is resolved server-side (Host /\n // DCS_SITE_SLUG), so VITE_SITE_SLUG is no longer required for routing —\n // only the API base URL is needed.\n if (!apiBaseUrl) {\n console.warn(\n '[@duffcloudservices/cms] Runtime mode enabled but VITE_API_BASE_URL not set'\n )\n return\n }\n\n // Check cache first\n const effectiveCacheKey = cacheKey ?? `${siteSlug}:${pageSlug}`\n const cached = fetchCache.get(effectiveCacheKey)\n if (cached && cached.expiresAt > Date.now()) {\n overrides.value = { ...buildTimeContent, ...cached.data }\n return\n }\n\n isLoading.value = true\n error.value = null\n\n try {\n // Site is resolved server-side from the request Host or the dedicated\n // Container App's DCS_SITE_SLUG; the slug is no longer encoded in the path.\n const url = `${apiBaseUrl}/api/v1/pages/${pageSlug}/text`\n const response = await fetch(url, {\n headers: {\n Accept: 'application/json',\n },\n })\n\n if (!response.ok) {\n if (response.status === 404) {\n // No overrides for this page - that's fine\n return\n }\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const data = await response.json()\n const apiOverrides = data.overrides ?? data.texts ?? {}\n\n // Cache the result\n fetchCache.set(effectiveCacheKey, {\n data: apiOverrides,\n expiresAt: Date.now() + cacheTtl,\n })\n\n // Merge: build-time content is baseline, API overrides on top\n overrides.value = { ...buildTimeContent, ...apiOverrides }\n } catch (e) {\n error.value = e instanceof Error ? e.message : 'Failed to load text overrides'\n console.error('[@duffcloudservices/cms] Failed to fetch text overrides:', e)\n } finally {\n isLoading.value = false\n }\n }\n\n /**\n * Manually refresh overrides.\n * In commit mode, resets to build-time content.\n * In runtime mode, fetches fresh data from API.\n */\n async function refresh(): Promise<void> {\n if (mode !== 'runtime') {\n // In commit mode, just reset to build-time content\n overrides.value = { ...buildTimeContent }\n return\n }\n\n // Clear cache for this key\n const effectiveCacheKey = cacheKey ?? `${siteSlug}:${pageSlug}`\n fetchCache.delete(effectiveCacheKey)\n\n await fetchOverrides()\n }\n\n // Fetch on mount if enabled and in runtime mode (browser only)\n if (fetchOnMount && mode === 'runtime' && globalThis.window !== undefined) {\n onMounted(() => {\n fetchOverrides()\n })\n }\n\n return {\n t,\n getArray,\n texts,\n overrides,\n isLoading: readonly(isLoading) as Ref<boolean>,\n error: readonly(error) as Ref<string | null>,\n refresh,\n hasOverride,\n hasBuildTimeContent,\n mode,\n }\n}\n","/**\n * useSEO Composable\n *\n * Provides SEO configuration with build-time injection support from .dcs/seo.yaml.\n * Generates meta tags, Open Graph, Twitter Cards, and JSON-LD structured data.\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useSEO } from '@duffcloudservices/cms'\n *\n * const { applyHead, getSchema, config } = useSEO('home')\n *\n * // Apply all meta tags\n * applyHead()\n *\n * // Or customize before applying\n * applyHead({\n * title: 'Custom Override Title',\n * schemas: [...getSchema(), customSchema]\n * })\n * </script>\n * ```\n */\n\nimport { computed, type ComputedRef } from 'vue'\nimport { useHead } from '@unhead/vue'\nimport type {\n SeoConfiguration,\n GlobalSeoConfig,\n SeoOpenGraphConfig,\n SeoTwitterConfig,\n SeoSchemaConfig,\n ResolvedPageSeo,\n UseSeoReturn,\n HeadOverrides,\n} from '../types/seo'\n\n// Declare the global injected by dcsSeoPlugin\ndeclare const __DCS_SEO__: SeoConfiguration | undefined\n\n/**\n * Safely get build-time SEO configuration.\n * Returns undefined if not available (no seo.yaml or plugin not configured).\n */\nfunction getBuildTimeSeo(): SeoConfiguration | undefined {\n try {\n if (typeof __DCS_SEO__ !== 'undefined' && __DCS_SEO__ !== null) {\n return __DCS_SEO__\n }\n } catch {\n // __DCS_SEO__ not defined - that's fine, use defaults\n }\n return undefined\n}\n\n// =============================================================================\n// Meta Tag Generation Utilities\n// =============================================================================\n\ninterface HeadInput {\n title?: string\n titleTemplate?: string | ((title: string) => string)\n meta?: Array<{ name?: string; property?: string; content: string }>\n link?: Array<{ rel: string; href: string; hreflang?: string }>\n script?: Array<{ type: string; children: string }>\n}\n\n/**\n * Generate Open Graph meta tags from config\n */\nfunction generateOpenGraphMeta(\n og: SeoOpenGraphConfig,\n global: GlobalSeoConfig,\n pageTitle: string,\n pageDescription: string,\n canonical: string\n): Array<{ property: string; content: string }> {\n const tags: Array<{ property: string; content: string }> = []\n\n tags.push({ property: 'og:title', content: og.title || pageTitle })\n tags.push({ property: 'og:description', content: og.description || pageDescription })\n tags.push({ property: 'og:url', content: og.url || canonical })\n tags.push({ property: 'og:type', content: og.type || 'website' })\n\n const image = og.image || global.images?.ogDefault\n if (image) {\n tags.push({ property: 'og:image', content: image })\n if (og.imageAlt || pageTitle) {\n tags.push({ property: 'og:image:alt', content: og.imageAlt || pageTitle })\n }\n if (og.imageWidth) {\n tags.push({ property: 'og:image:width', content: String(og.imageWidth) })\n }\n if (og.imageHeight) {\n tags.push({ property: 'og:image:height', content: String(og.imageHeight) })\n }\n }\n\n if (global.siteName) {\n tags.push({ property: 'og:site_name', content: global.siteName })\n }\n\n if (global.locale) {\n tags.push({ property: 'og:locale', content: global.locale })\n }\n\n // Article-specific tags\n if (og.type === 'article') {\n if (og.publishedTime) {\n tags.push({ property: 'article:published_time', content: og.publishedTime })\n }\n if (og.modifiedTime) {\n tags.push({ property: 'article:modified_time', content: og.modifiedTime })\n }\n if (og.author) {\n tags.push({ property: 'article:author', content: og.author })\n }\n if (og.section) {\n tags.push({ property: 'article:section', content: og.section })\n }\n if (og.tags) {\n og.tags.forEach((tag) => {\n tags.push({ property: 'article:tag', content: tag })\n })\n }\n }\n\n return tags\n}\n\n/**\n * Generate Twitter Card meta tags from config\n */\nfunction generateTwitterMeta(\n twitter: SeoTwitterConfig,\n global: GlobalSeoConfig,\n pageTitle: string,\n pageDescription: string\n): Array<{ name: string; content: string }> {\n const tags: Array<{ name: string; content: string }> = []\n\n tags.push({ name: 'twitter:card', content: twitter.card || 'summary_large_image' })\n tags.push({ name: 'twitter:title', content: twitter.title || pageTitle })\n tags.push({ name: 'twitter:description', content: twitter.description || pageDescription })\n\n const image = twitter.image || global.images?.twitterDefault\n if (image) {\n tags.push({ name: 'twitter:image', content: image })\n if (twitter.imageAlt || pageTitle) {\n tags.push({ name: 'twitter:image:alt', content: twitter.imageAlt || pageTitle })\n }\n }\n\n const site = twitter.site || global.social?.twitter\n if (site) {\n tags.push({ name: 'twitter:site', content: site.startsWith('@') ? site : `@${site}` })\n }\n\n if (twitter.creator) {\n tags.push({\n name: 'twitter:creator',\n content: twitter.creator.startsWith('@') ? twitter.creator : `@${twitter.creator}`,\n })\n }\n\n return tags\n}\n\n/**\n * Generate JSON-LD script content from schemas\n */\nfunction generateJsonLd(schemas: SeoSchemaConfig[], global: GlobalSeoConfig): object[] {\n return schemas.map((schema) => {\n const base: Record<string, unknown> = {\n '@context': 'https://schema.org',\n '@type': schema.type,\n }\n\n // Merge properties\n if (schema.properties) {\n Object.assign(base, schema.properties)\n }\n\n // Auto-populate common properties from global config\n if (schema.type === 'WebSite' && global.siteUrl && !base.url) {\n base.url = global.siteUrl\n }\n if (schema.type === 'WebSite' && global.siteName && !base.name) {\n base.name = global.siteName\n }\n\n return base\n })\n}\n\n/**\n * Resolve page SEO by merging global defaults with page-specific config\n */\nfunction resolvePageSeo(\n pageSlug: string,\n pagePath: string | undefined,\n seoConfig: SeoConfiguration | undefined\n): ResolvedPageSeo {\n const global = seoConfig?.global ?? {}\n const page = seoConfig?.pages?.[pageSlug] ?? {}\n\n // Build canonical URL\n let canonical = page.canonical || ''\n if (!canonical && global.siteUrl) {\n const path = pagePath ?? (pageSlug === 'home' ? '/' : `/${pageSlug}`)\n canonical = `${global.siteUrl.replace(/\\/$/, '')}${path}`\n }\n\n // Build title\n let title = page.title || global.defaultTitle || pageSlug\n if (!page.noTitleTemplate && global.titleTemplate) {\n title = global.titleTemplate.replace('%s', title)\n }\n\n // Merge Open Graph\n const openGraph: ResolvedPageSeo['openGraph'] = {\n type: page.openGraph?.type || 'website',\n title: page.openGraph?.title || page.title || global.defaultTitle || '',\n description: page.openGraph?.description || page.description || global.defaultDescription || '',\n ...page.openGraph,\n }\n\n // Merge Twitter\n const twitter: ResolvedPageSeo['twitter'] = {\n card: page.twitter?.card || 'summary_large_image',\n ...page.twitter,\n }\n\n // Combine schemas (global + page)\n const schemas = [...(global.schemas ?? []), ...(page.schemas ?? [])]\n\n return {\n title,\n description: page.description || global.defaultDescription || '',\n canonical,\n robots: page.robots || global.robots || 'index, follow',\n openGraph,\n twitter,\n schemas,\n alternates: page.alternates ?? [],\n }\n}\n\n/**\n * useSEO composable for DCS-managed SEO configuration.\n *\n * @param pageSlug - Page slug matching entry in seo.yaml\n * @param pagePath - Optional page path for canonical URL generation\n * @returns SEO helpers and state\n */\nexport function useSEO(pageSlug: string, pagePath?: string): UseSeoReturn {\n const seoConfig = getBuildTimeSeo()\n const hasBuildTimeSeo = seoConfig !== undefined\n\n // Computed resolved config\n const config: ComputedRef<ResolvedPageSeo> = computed(() =>\n resolvePageSeo(pageSlug, pagePath, seoConfig)\n )\n\n /**\n * Get JSON-LD schema objects for the page\n */\n function getSchema(): object[] {\n return generateJsonLd(config.value.schemas, seoConfig?.global ?? {})\n }\n\n /**\n * Get canonical URL for the page\n */\n function getCanonical(): string {\n return config.value.canonical\n }\n\n /**\n * Apply all meta tags via useHead\n */\n function applyHead(overrides?: HeadOverrides): void {\n const resolved = config.value\n const global = seoConfig?.global ?? {}\n\n const title = overrides?.title ?? resolved.title\n const description = overrides?.description ?? resolved.description\n\n // Build meta tags\n const meta: HeadInput['meta'] = []\n\n // Basic meta\n meta.push({ name: 'description', content: description })\n if (resolved.robots) {\n meta.push({ name: 'robots', content: resolved.robots })\n }\n\n // Verification codes\n if (global.verification?.google) {\n meta.push({ name: 'google-site-verification', content: global.verification.google })\n }\n if (global.verification?.bing) {\n meta.push({ name: 'msvalidate.01', content: global.verification.bing })\n }\n\n // Open Graph\n const ogMeta = generateOpenGraphMeta(\n resolved.openGraph,\n global,\n title,\n description,\n resolved.canonical\n )\n meta.push(...ogMeta.map((t) => ({ property: t.property, content: t.content })))\n\n // Twitter\n const twitterMeta = generateTwitterMeta(resolved.twitter, global, title, description)\n meta.push(...twitterMeta.map((t) => ({ name: t.name, content: t.content })))\n\n // Additional overrides\n if (overrides?.meta) {\n meta.push(...overrides.meta)\n }\n\n // Build links\n const link: HeadInput['link'] = []\n\n // Canonical\n if (resolved.canonical) {\n link.push({ rel: 'canonical', href: resolved.canonical })\n }\n\n // Alternate languages\n resolved.alternates.forEach((alt) => {\n link.push({ rel: 'alternate', href: alt.href, hreflang: alt.hreflang })\n })\n\n // Build scripts (JSON-LD)\n const schemas = overrides?.schemas ?? getSchema()\n const script: HeadInput['script'] = schemas.map((schema) => ({\n type: 'application/ld+json',\n children: JSON.stringify(schema),\n }))\n\n // Apply via useHead\n useHead({\n title,\n meta,\n link,\n script,\n })\n }\n\n return {\n config,\n applyHead,\n getSchema,\n getCanonical,\n hasBuildTimeSeo,\n }\n}\n\n/**\n * Create a typed useSEO function with site-specific defaults.\n * Useful for creating a site-wide wrapper.\n *\n * @example\n * ```ts\n * // composables/useSiteSeo.ts\n * import { createSiteSEO } from '@duffcloudservices/cms'\n *\n * export const useSiteSeo = createSiteSEO({\n * siteName: 'My Site',\n * siteUrl: 'https://example.com'\n * })\n * ```\n */\nexport function createSiteSEO(\n _siteDefaults: Partial<GlobalSeoConfig>\n): (pageSlug: string, pagePath?: string) => UseSeoReturn {\n return function siteUseSEO(pageSlug: string, pagePath?: string): UseSeoReturn {\n // Note: siteDefaults would be used if we needed to override at runtime\n // but build-time injection handles this via dcsSeoPlugin\n return useSEO(pageSlug, pagePath)\n }\n}\n","/**\n * useReleaseNotes Composable\n *\n * Fetches and displays versioned release notes from the DCS Portal API.\n * Supports fetching specific versions or the latest release.\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useReleaseNotes } from '@duffcloudservices/cms'\n * import { useRoute } from 'vue-router'\n *\n * const route = useRoute()\n * const version = route.params.version as string || 'latest'\n *\n * const { releaseNote, isLoading, error } = useReleaseNotes(version)\n * </script>\n *\n * <template>\n * <div v-if=\"isLoading\">Loading...</div>\n * <div v-else-if=\"error\">{{ error }}</div>\n * <article v-else-if=\"releaseNote\">\n * <h1>{{ releaseNote.title }}</h1>\n * <p>{{ releaseNote.summary }}</p>\n * <div v-html=\"renderedMarkdown\" />\n * </article>\n * </template>\n * ```\n */\n\nimport { ref, onMounted } from 'vue'\nimport type { ReleaseNote, ReleaseNotesReturn } from '../types/release-notes'\n\n/**\n * Get environment variable value.\n */\nfunction getEnvVar(key: string, defaultValue = ''): string {\n try {\n if (typeof import.meta !== 'undefined' && import.meta.env) {\n const value = (import.meta.env as Record<string, string | undefined>)[key]\n if (value !== undefined) return value\n }\n } catch {\n // import.meta not available\n }\n return defaultValue\n}\n\n// Simple cache for release notes\nconst releaseNotesCache = new Map<string, { data: ReleaseNote; expiresAt: number }>()\nconst CACHE_TTL = 5 * 60 * 1000 // 5 minutes\n\n/**\n * useReleaseNotes composable for fetching release notes from the DCS API.\n *\n * @param version - Semantic version (e.g., \"1.2.0\") or \"latest\"\n * @param options - Optional configuration\n * @returns Release notes data and state\n */\nexport function useReleaseNotes(\n version: string,\n options: { fetchOnMount?: boolean } = {}\n): ReleaseNotesReturn {\n const { fetchOnMount = true } = options\n\n const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', 'https://portal.duffcloudservices.com')\n // @deprecated VITE_SITE_SLUG: the site is now resolved server-side from the\n // request Host or the dedicated Container App's DCS_SITE_SLUG. It is retained\n // only for cache-key continuity and source compatibility, not for routing.\n const siteSlug = getEnvVar('VITE_SITE_SLUG', '')\n\n const releaseNote = ref<ReleaseNote | null>(null)\n const isLoading = ref(false)\n const error = ref<string | null>(null)\n\n async function fetchReleaseNotes(): Promise<void> {\n // Check cache. The slug is no longer required for routing (server resolves\n // the site from Host / DCS_SITE_SLUG); it is only part of the cache key.\n const cacheKey = `${siteSlug}:${version}`\n const cached = releaseNotesCache.get(cacheKey)\n if (cached && cached.expiresAt > Date.now()) {\n releaseNote.value = cached.data\n return\n }\n\n isLoading.value = true\n error.value = null\n\n try {\n const url = `${apiBaseUrl}/api/v1/release-notes/${version}`\n const response = await fetch(url, {\n headers: {\n Accept: 'application/json',\n },\n })\n\n if (!response.ok) {\n if (response.status === 404) {\n error.value = `Release notes for version ${version} not found`\n return\n }\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const data = await response.json()\n\n // Normalize the response\n const note: ReleaseNote = {\n version: data.version,\n title: data.title,\n summary: data.summary || '',\n notesMarkdown: data.notesMarkdown || data.notes || '',\n changeCount: data.changeCount || 0,\n releaseDate: data.releaseDate || data.releasedAt || '',\n }\n\n // Cache the result\n releaseNotesCache.set(cacheKey, {\n data: note,\n expiresAt: Date.now() + CACHE_TTL,\n })\n\n releaseNote.value = note\n } catch (e) {\n error.value = e instanceof Error ? e.message : 'Failed to load release notes'\n console.error('[@duffcloudservices/cms] Failed to fetch release notes:', e)\n } finally {\n isLoading.value = false\n }\n }\n\n async function refresh(): Promise<void> {\n // Clear cache\n const cacheKey = `${siteSlug}:${version}`\n releaseNotesCache.delete(cacheKey)\n await fetchReleaseNotes()\n }\n\n // Fetch on mount if enabled\n if (fetchOnMount && typeof window !== 'undefined') {\n onMounted(() => {\n fetchReleaseNotes()\n })\n }\n\n return {\n releaseNote,\n isLoading,\n error,\n refresh,\n }\n}\n","/**\n * useSiteVersion Composable\n *\n * Gets the current site version for footer badges and version displays.\n * Fetches the latest release version from the DCS Portal API.\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useSiteVersion } from '@duffcloudservices/cms'\n *\n * const { version, releaseNotesUrl } = useSiteVersion()\n * </script>\n *\n * <template>\n * <footer>\n * <a v-if=\"version\" :href=\"releaseNotesUrl\" class=\"version-badge\">\n * v{{ version }}\n * </a>\n * </footer>\n * </template>\n * ```\n */\n\nimport { ref, computed, onMounted } from 'vue'\nimport type { SiteVersionReturn } from '../types/release-notes'\n\n/**\n * Get environment variable value.\n */\nfunction getEnvVar(key: string, defaultValue = ''): string {\n try {\n if (typeof import.meta !== 'undefined' && import.meta.env) {\n const value = (import.meta.env as Record<string, string | undefined>)[key]\n if (value !== undefined) return value\n }\n } catch {\n // import.meta not available\n }\n return defaultValue\n}\n\n// Cache for site version\nlet versionCache: { version: string; expiresAt: number } | null = null\nconst CACHE_TTL = 10 * 60 * 1000 // 10 minutes\n\n/**\n * useSiteVersion composable for displaying the current site version.\n *\n * @param options - Optional configuration\n * @returns Site version data and computed URL\n */\nexport function useSiteVersion(options: { fetchOnMount?: boolean } = {}): SiteVersionReturn {\n const { fetchOnMount = true } = options\n\n const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', 'https://portal.duffcloudservices.com')\n // @deprecated VITE_SITE_SLUG: the site is now resolved server-side from the\n // request Host or the dedicated Container App's DCS_SITE_SLUG. It is no longer\n // used for routing; retained only for source compatibility.\n void getEnvVar('VITE_SITE_SLUG', '')\n\n const version = ref<string | null>(null)\n const isLoading = ref(false)\n\n const releaseNotesUrl = computed(() => {\n if (!version.value) return '/releaseNotes/latest'\n return `/releaseNotes/${version.value}`\n })\n\n async function fetchVersion(): Promise<void> {\n // Check cache\n if (versionCache && versionCache.expiresAt > Date.now()) {\n version.value = versionCache.version\n return\n }\n\n isLoading.value = true\n\n try {\n // Fetch the latest release notes to get the version\n const url = `${apiBaseUrl}/api/v1/release-notes/latest`\n const response = await fetch(url, {\n headers: {\n Accept: 'application/json',\n },\n })\n\n if (!response.ok) {\n // No release notes yet - that's fine\n return\n }\n\n const data = await response.json()\n const latestVersion = data.version\n\n // Cache the result\n versionCache = {\n version: latestVersion,\n expiresAt: Date.now() + CACHE_TTL,\n }\n\n version.value = latestVersion\n } catch (e) {\n console.error('[@duffcloudservices/cms] Failed to fetch site version:', e)\n } finally {\n isLoading.value = false\n }\n }\n\n // Fetch on mount if enabled\n if (fetchOnMount && typeof window !== 'undefined') {\n onMounted(() => {\n fetchVersion()\n })\n }\n\n return {\n version,\n isLoading,\n releaseNotesUrl,\n }\n}\n","/**\n * useMediaCarousel Composable\n *\n * Extracts media carousel items from text content keys following the pattern:\n * `{prefix}.{N}.url`, `{prefix}.{N}.type`, `{prefix}.{N}.alt`\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useTextContent, useMediaCarousel } from '@duffcloudservices/cms'\n *\n * const { t } = useTextContent({\n * pageSlug: 'bio-mackenzie',\n * defaults: {\n * 'hero.media-carousel.0.url': '/images/staff/mackenzie.webp',\n * 'hero.media-carousel.0.type': 'image',\n * 'hero.media-carousel.0.alt': 'Mackenzie Kowalick',\n * 'hero.media-carousel.1.url': '/videos/intro.mp4',\n * 'hero.media-carousel.1.type': 'video',\n * 'hero.media-carousel.1.alt': 'Introduction video',\n * }\n * })\n *\n * const { items } = useMediaCarousel({\n * prefix: 'hero.media-carousel',\n * t,\n * defaults: [\n * { url: '/images/staff/mackenzie.webp', type: 'image', alt: 'Mackenzie Kowalick' }\n * ]\n * })\n * </script>\n *\n * <template>\n * <MediaCarousel :items=\"items\" />\n * </template>\n * ```\n */\n\nimport { computed, type ComputedRef } from 'vue'\nimport { isCdnAssetUrl } from '@duffcloudservices/cms-core'\n\n/**\n * Media carousel item representing an image, video, or embed\n */\nexport interface MediaCarouselItem {\n /** URL to the image, video file, or embed URL */\n url: string\n /** Type of media: 'image', 'video' (direct file), 'youtube', or 'instagram' */\n type: 'image' | 'video' | 'youtube' | 'instagram'\n /** Accessibility alt text */\n alt?: string\n /**\n * Whether this image has responsive CDN variants available.\n * Automatically set to `true` when the URL matches the DCS CDN asset pattern.\n * Components rendering the carousel should use `<ResponsiveImage>` when this is `true`.\n */\n responsive?: boolean\n}\n\n/**\n * Configuration for useMediaCarousel composable\n */\nexport interface UseMediaCarouselConfig {\n /** Key prefix for carousel items (e.g., 'hero.media-carousel') */\n prefix: string\n /** The t() function from useTextContent */\n t: (key: string, fallback?: string) => string\n /** Default items to use if no content keys are found */\n defaults?: MediaCarouselItem[]\n /** Maximum number of items to look for (default: 10) */\n maxItems?: number\n}\n\n/**\n * Return type for useMediaCarousel composable\n */\nexport interface UseMediaCarouselReturn {\n /** Computed array of media carousel items */\n items: ComputedRef<MediaCarouselItem[]>\n /** Whether any items were found from content keys */\n hasItems: ComputedRef<boolean>\n /** Number of items in the carousel */\n count: ComputedRef<number>\n}\n\n/**\n * Extract media carousel items from text content keys.\n *\n * Looks for keys in the format:\n * - `{prefix}.{N}.url` - Required URL for the media\n * - `{prefix}.{N}.type` - Type: 'image' or 'video' (defaults to 'image')\n * - `{prefix}.{N}.alt` - Alt text for accessibility\n *\n * Items are sorted by index (0, 1, etc.) and only included if they have a valid URL.\n *\n * @param config - Configuration object\n * @returns Media carousel helpers and state\n */\nexport function useMediaCarousel(config: UseMediaCarouselConfig): UseMediaCarouselReturn {\n const {\n prefix,\n t,\n defaults = [],\n maxItems = 10,\n } = config\n\n const items = computed<MediaCarouselItem[]>(() => {\n const result: MediaCarouselItem[] = []\n\n // Look for items from 0 to maxItems\n for (let i = 0; i < maxItems; i++) {\n const urlKey = `${prefix}.${i}.url`\n const typeKey = `${prefix}.${i}.type`\n const altKey = `${prefix}.${i}.alt`\n\n // Use a sentinel value to detect if the key exists\n const url = t(urlKey, '')\n \n // Skip if no URL (key doesn't exist or is empty)\n if (!url || url === urlKey) {\n continue\n }\n\n const typeValue = t(typeKey, 'image')\n // Parse type value - support image, video, youtube, instagram\n let type: 'image' | 'video' | 'youtube' | 'instagram' = 'image'\n if (typeValue === 'video') type = 'video'\n else if (typeValue === 'youtube') type = 'youtube'\n else if (typeValue === 'instagram') type = 'instagram'\n const alt = t(altKey, '')\n\n result.push({\n url,\n type,\n alt: alt && alt !== altKey ? alt : undefined,\n // Flag CDN-hosted images as responsive so carousel components\n // can render them with <ResponsiveImage> automatically\n responsive: type === 'image' && isCdnAssetUrl(url),\n })\n }\n\n // If no items found from content keys, use defaults\n if (result.length === 0) {\n return defaults\n }\n\n return result\n })\n\n const hasItems = computed(() => items.value.length > 0)\n const count = computed(() => items.value.length)\n\n return {\n items,\n hasItems,\n count,\n }\n}\n","/**\n * Vue 3 composable that resolves responsive image variants for DCS CDN-hosted assets.\n *\n * Wraps the framework-agnostic `resolveResponsiveImage` from `@duffcloudservices/cms-core`\n * with reactive Vue refs so it can be used directly in `<script setup>` blocks.\n *\n * @example\n * ```vue\n * <script setup lang=\"ts\">\n * import { useResponsiveImage } from '@duffcloudservices/cms'\n *\n * const hero = useResponsiveImage({\n * src: 'https://files.duffcloudservices.com/kept/assets/hero/abc-123.jpg',\n * alt: 'Hero image',\n * context: 'hero',\n * })\n * </script>\n *\n * <template>\n * <picture v-if=\"hero.hasVariants\">\n * <source\n * v-for=\"source in hero.sources\"\n * :key=\"source.type\"\n * :srcset=\"source.srcset\"\n * :type=\"source.type\"\n * :sizes=\"source.sizes\"\n * />\n * <img v-bind=\"hero.imgProps\" class=\"w-full h-full object-cover\" />\n * </picture>\n * <img v-else v-bind=\"hero.imgProps\" class=\"w-full h-full object-cover\" />\n * </template>\n * ```\n */\n\nimport { computed, type MaybeRefOrGetter, toValue } from 'vue'\nimport {\n resolveResponsiveImage,\n type ImageContext,\n type ResponsiveImageResult,\n} from '@duffcloudservices/cms-core'\n\nexport interface UseResponsiveImageOptions {\n /** Source URL — can be a reactive ref, getter, or plain string. */\n src: MaybeRefOrGetter<string>\n /** Alt text — can be a reactive ref, getter, or plain string. */\n alt: MaybeRefOrGetter<string>\n /** Sizing context — determines which variants to include. */\n context?: MaybeRefOrGetter<ImageContext | undefined>\n /** Optional `sizes` attribute override. */\n sizes?: MaybeRefOrGetter<string | undefined>\n /** Skip variant resolution and use the original URL only. */\n original?: MaybeRefOrGetter<boolean | undefined>\n}\n\n/**\n * Reactively resolves responsive image metadata for a DCS CDN URL.\n *\n * The returned object is a computed ref that recomputes whenever any\n * of the input refs change. Spread `imgProps` onto an `<img>` or\n * combine with `sources` inside a `<picture>` element.\n */\nexport function useResponsiveImage(options: UseResponsiveImageOptions): ResponsiveImageResult {\n const result = computed(() =>\n resolveResponsiveImage({\n src: toValue(options.src),\n alt: toValue(options.alt),\n context: toValue(options.context),\n sizes: toValue(options.sizes),\n original: toValue(options.original) ?? undefined,\n }),\n )\n\n // Return a reactive proxy that delegates to the computed\n return {\n get imgProps() {\n return result.value.imgProps\n },\n get sources() {\n return result.value.sources\n },\n get hasVariants() {\n return result.value.hasVariants\n },\n }\n}\n","/**\n * Composable for reading curated review selections from DCS content.\n * Reviews are stored in content.yaml by the visual editor's ReviewPickerSheet.\n */\nimport { computed, onMounted, onUnmounted, shallowRef, type ComputedRef } from 'vue'\n\nexport interface ReviewItem {\n id: string\n platform: 'google' | 'meta' | string\n rating: number\n authorName: string\n authorPhotoUrl?: string\n text?: string\n date?: string\n replyText?: string\n locationName?: string\n sourceLocationName?: string\n sourceUrl?: string\n}\n\nfunction withoutAuthorPhotos(items: ReviewItem[]): ReviewItem[] {\n return items.map(item => ({\n ...item,\n authorPhotoUrl: undefined,\n }))\n}\n\nexport interface UseReviewContentConfig {\n /** The section key matching the data-dcs-reviews attribute value */\n sectionKey: string\n /** Page slug for page-specific content lookup (defaults to current page) */\n pageSlug?: string\n /** Fallback reviews when no content is available */\n defaults?: ReviewItem[]\n}\n\nexport interface UseReviewContentReturn {\n /** The curated review items from content */\n reviews: ComputedRef<ReviewItem[]>\n /** Whether any reviews are available */\n hasReviews: ComputedRef<boolean>\n /** Number of reviews */\n count: ComputedRef<number>\n}\n\nconst previewReviewOverrides = shallowRef<Record<string, ReviewItem[]>>({})\nlet activePreviewReviewConsumers = 0\n\n// Declare the global content variable injected by dcsContentPlugin\ndeclare const __DCS_CONTENT__: {\n global?: Record<string, unknown>\n pages?: Record<string, Record<string, unknown>>\n} | undefined\n\nfunction normalizeReviewItem(item: Record<string, unknown>): ReviewItem | null {\n const id = String(item.id ?? '').trim()\n if (!id) {\n return null\n }\n\n const rawRating = Number(item.rating ?? 5)\n const rating = Number.isFinite(rawRating)\n ? Math.min(5, Math.max(1, Math.round(rawRating)))\n : 5\n\n return {\n id,\n platform: String(item.platform ?? 'google'),\n rating,\n authorName: String(item.authorName ?? 'Anonymous'),\n authorPhotoUrl: undefined,\n text: item.text ? String(item.text) : undefined,\n date: item.date ? String(item.date) : undefined,\n replyText: item.replyText ? String(item.replyText) : undefined,\n locationName: item.locationName ? String(item.locationName) : undefined,\n sourceLocationName: item.sourceLocationName ? String(item.sourceLocationName) : undefined,\n sourceUrl: item.sourceUrl ? String(item.sourceUrl) : undefined,\n }\n}\n\nfunction normalizeReviewList(value: unknown): ReviewItem[] {\n if (!Array.isArray(value)) {\n return []\n }\n\n return value\n .filter((item): item is Record<string, unknown> => item != null && typeof item === 'object')\n .map(normalizeReviewItem)\n .filter((item): item is ReviewItem => item != null)\n}\n\nfunction handlePreviewReviewUpdate(event: Event) {\n const detail = event instanceof CustomEvent && event.detail != null && typeof event.detail === 'object'\n ? event.detail as { key?: unknown; reviews?: unknown }\n : null\n const key = typeof detail?.key === 'string' ? detail.key.trim() : ''\n if (!key) {\n return\n }\n\n const next = { ...previewReviewOverrides.value }\n if (Array.isArray(detail?.reviews)) {\n next[key] = normalizeReviewList(detail.reviews)\n } else {\n delete next[key]\n }\n previewReviewOverrides.value = next\n}\n\nexport function useReviewContent(config: UseReviewContentConfig): UseReviewContentReturn {\n const { sectionKey, pageSlug, defaults = [] } = config\n\n onMounted(() => {\n activePreviewReviewConsumers += 1\n if (activePreviewReviewConsumers === 1) {\n window.addEventListener('dcs:reviews-updated', handlePreviewReviewUpdate)\n }\n })\n\n onUnmounted(() => {\n activePreviewReviewConsumers = Math.max(0, activePreviewReviewConsumers - 1)\n if (activePreviewReviewConsumers === 0) {\n window.removeEventListener('dcs:reviews-updated', handlePreviewReviewUpdate)\n }\n })\n\n const reviews = computed<ReviewItem[]>(() => {\n if (Object.prototype.hasOwnProperty.call(previewReviewOverrides.value, sectionKey)) {\n return previewReviewOverrides.value[sectionKey] ?? []\n }\n\n if (typeof __DCS_CONTENT__ === 'undefined' || __DCS_CONTENT__ == null) {\n return withoutAuthorPhotos(defaults)\n }\n\n let reviewData: unknown = null\n\n if (pageSlug && __DCS_CONTENT__.pages?.[pageSlug]) {\n reviewData = __DCS_CONTENT__.pages[pageSlug][`reviews.${sectionKey}.items`]\n ?? __DCS_CONTENT__.pages[pageSlug][`reviews.${sectionKey}`]\n }\n\n if (!reviewData && __DCS_CONTENT__.global) {\n reviewData = __DCS_CONTENT__.global[`reviews.${sectionKey}.items`]\n ?? __DCS_CONTENT__.global[`reviews.${sectionKey}`]\n }\n\n if (!reviewData || !Array.isArray(reviewData)) {\n return withoutAuthorPhotos(defaults)\n }\n\n return normalizeReviewList(reviewData)\n })\n\n const hasReviews = computed(() => reviews.value.length > 0)\n const count = computed(() => reviews.value.length)\n\n return { reviews, hasReviews, count }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@duffcloudservices/cms",
3
- "version": "0.3.15",
3
+ "version": "0.3.17",
4
4
  "description": "Vue 3 composables and Vite plugins for DCS CMS integration",
5
5
  "type": "module",
6
6
  "exports": {
@@ -22,6 +22,9 @@
22
22
  "./responsive-image": {
23
23
  "import": "./src/components/ResponsiveImage.vue"
24
24
  },
25
+ "./managed-image": {
26
+ "import": "./src/components/ManagedImage.vue"
27
+ },
25
28
  "./review-showcase": {
26
29
  "import": "./src/components/DcsReviewShowcase.vue"
27
30
  }
@@ -0,0 +1,66 @@
1
+ <script setup lang="ts">
2
+ import { computed, useAttrs } from 'vue'
3
+ import { useResponsiveImage } from '../composables/useResponsiveImage'
4
+ import { useTextContent } from '../composables/useTextContent'
5
+ import type { ImageContext } from '@duffcloudservices/cms-core'
6
+
7
+ defineOptions({ inheritAttrs: false })
8
+
9
+ const props = withDefaults(defineProps<{
10
+ pageSlug: string
11
+ imageKey: string
12
+ fallbackSrc: string
13
+ altKey?: string
14
+ fallbackAlt?: string
15
+ context?: ImageContext
16
+ sizes?: string
17
+ original?: boolean
18
+ }>(), {
19
+ altKey: '',
20
+ fallbackAlt: '',
21
+ context: 'content',
22
+ sizes: undefined,
23
+ original: false,
24
+ })
25
+
26
+ const attrs = useAttrs()
27
+ const { t } = useTextContent({
28
+ pageSlug: props.pageSlug,
29
+ defaults: {
30
+ [props.imageKey]: props.fallbackSrc,
31
+ ...(props.altKey ? { [props.altKey]: props.fallbackAlt } : {}),
32
+ },
33
+ })
34
+
35
+ const imageSrc = computed(() => t(props.imageKey, props.fallbackSrc))
36
+ const imageAlt = computed(() => (props.altKey ? t(props.altKey, props.fallbackAlt) : props.fallbackAlt))
37
+ const image = useResponsiveImage({
38
+ src: imageSrc,
39
+ alt: imageAlt,
40
+ context: () => props.context,
41
+ sizes: () => props.sizes,
42
+ original: () => props.original,
43
+ })
44
+
45
+ const imgAttrs = computed(() => ({
46
+ ...image.imgProps,
47
+ ...attrs,
48
+ 'data-dcs-image-key': props.imageKey,
49
+ 'data-dcs-image-url': imageSrc.value,
50
+ 'data-dcs-image-alt': imageAlt.value,
51
+ }))
52
+ </script>
53
+
54
+ <template>
55
+ <picture v-if="image.hasVariants && !original">
56
+ <source
57
+ v-for="source in image.sources"
58
+ :key="source.type"
59
+ :srcset="source.srcset"
60
+ :type="source.type"
61
+ :sizes="source.sizes"
62
+ />
63
+ <img v-bind="imgAttrs" />
64
+ </picture>
65
+ <img v-else v-bind="imgAttrs" />
66
+ </template>
@@ -228,15 +228,14 @@ onMounted(async () => {
228
228
  }
229
229
 
230
230
  try {
231
- const slug =
232
- (import.meta.env as Record<string, string | undefined>).VITE_SITE_SLUG ??
233
- ''
234
- if (!slug) return
231
+ // The site is resolved server-side (request Host / DCS_SITE_SLUG); the slug
232
+ // is no longer encoded in the path. VITE_SITE_SLUG is retained for source
233
+ // compatibility but is not required here.
235
234
  const base =
236
235
  (import.meta.env as Record<string, string | undefined>)
237
236
  .VITE_API_BASE_URL ?? 'https://portal.duffcloudservices.com'
238
237
  const res = await fetch(
239
- `${base}/api/v1/sites/${slug}/release-notes/latest`,
238
+ `${base}/api/v1/release-notes/latest`,
240
239
  { headers: { Accept: 'application/json' } },
241
240
  )
242
241
  if (res.ok) {
@@ -13,9 +13,12 @@
13
13
  * />
14
14
  * ```
15
15
  */
16
+ import { computed, useAttrs } from 'vue'
16
17
  import { useResponsiveImage } from '../composables/useResponsiveImage'
17
18
  import type { ImageContext } from '@duffcloudservices/cms-core'
18
19
 
20
+ defineOptions({ inheritAttrs: false })
21
+
19
22
  const props = defineProps<{
20
23
  /** Image source URL (original CDN URL or local path). */
21
24
  src: string
@@ -38,6 +41,12 @@ const image = useResponsiveImage({
38
41
  sizes: () => props.sizes,
39
42
  original: () => props.original,
40
43
  })
44
+
45
+ const attrs = useAttrs()
46
+ const imgAttrs = computed(() => ({
47
+ ...image.imgProps,
48
+ ...attrs,
49
+ }))
41
50
  </script>
42
51
 
43
52
  <template>
@@ -49,7 +58,7 @@ const image = useResponsiveImage({
49
58
  :type="source.type"
50
59
  :sizes="source.sizes"
51
60
  />
52
- <img v-bind="image.imgProps" :class="props.class" />
61
+ <img v-bind="imgAttrs" :class="props.class" />
53
62
  </picture>
54
- <img v-else v-bind="image.imgProps" :class="props.class" />
63
+ <img v-else v-bind="imgAttrs" :class="props.class" />
55
64
  </template>
@@ -64,6 +64,9 @@ export function useReleaseNotes(
64
64
  const { fetchOnMount = true } = options
65
65
 
66
66
  const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', 'https://portal.duffcloudservices.com')
67
+ // @deprecated VITE_SITE_SLUG: the site is now resolved server-side from the
68
+ // request Host or the dedicated Container App's DCS_SITE_SLUG. It is retained
69
+ // only for cache-key continuity and source compatibility, not for routing.
67
70
  const siteSlug = getEnvVar('VITE_SITE_SLUG', '')
68
71
 
69
72
  const releaseNote = ref<ReleaseNote | null>(null)
@@ -71,12 +74,8 @@ export function useReleaseNotes(
71
74
  const error = ref<string | null>(null)
72
75
 
73
76
  async function fetchReleaseNotes(): Promise<void> {
74
- if (!siteSlug) {
75
- error.value = 'VITE_SITE_SLUG environment variable not set'
76
- return
77
- }
78
-
79
- // Check cache
77
+ // Check cache. The slug is no longer required for routing (server resolves
78
+ // the site from Host / DCS_SITE_SLUG); it is only part of the cache key.
80
79
  const cacheKey = `${siteSlug}:${version}`
81
80
  const cached = releaseNotesCache.get(cacheKey)
82
81
  if (cached && cached.expiresAt > Date.now()) {
@@ -88,7 +87,7 @@ export function useReleaseNotes(
88
87
  error.value = null
89
88
 
90
89
  try {
91
- const url = `${apiBaseUrl}/api/v1/sites/${siteSlug}/release-notes/${version}`
90
+ const url = `${apiBaseUrl}/api/v1/release-notes/${version}`
92
91
  const response = await fetch(url, {
93
92
  headers: {
94
93
  Accept: 'application/json',
@@ -54,7 +54,10 @@ export function useSiteVersion(options: { fetchOnMount?: boolean } = {}): SiteVe
54
54
  const { fetchOnMount = true } = options
55
55
 
56
56
  const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', 'https://portal.duffcloudservices.com')
57
- const siteSlug = getEnvVar('VITE_SITE_SLUG', '')
57
+ // @deprecated VITE_SITE_SLUG: the site is now resolved server-side from the
58
+ // request Host or the dedicated Container App's DCS_SITE_SLUG. It is no longer
59
+ // used for routing; retained only for source compatibility.
60
+ void getEnvVar('VITE_SITE_SLUG', '')
58
61
 
59
62
  const version = ref<string | null>(null)
60
63
  const isLoading = ref(false)
@@ -65,10 +68,6 @@ export function useSiteVersion(options: { fetchOnMount?: boolean } = {}): SiteVe
65
68
  })
66
69
 
67
70
  async function fetchVersion(): Promise<void> {
68
- if (!siteSlug) {
69
- return
70
- }
71
-
72
71
  // Check cache
73
72
  if (versionCache && versionCache.expiresAt > Date.now()) {
74
73
  version.value = versionCache.version
@@ -79,7 +78,7 @@ export function useSiteVersion(options: { fetchOnMount?: boolean } = {}): SiteVe
79
78
 
80
79
  try {
81
80
  // Fetch the latest release notes to get the version
82
- const url = `${apiBaseUrl}/api/v1/sites/${siteSlug}/release-notes/latest`
81
+ const url = `${apiBaseUrl}/api/v1/release-notes/latest`
83
82
  const response = await fetch(url, {
84
83
  headers: {
85
84
  Accept: 'application/json',
@@ -0,0 +1,46 @@
1
+ import { describe, it, expect, vi, afterEach, beforeEach } from 'vitest'
2
+ import { useTextContent } from './useTextContent'
3
+
4
+ /**
5
+ * Pins the slug-free URL shape for runtime text-override fetches.
6
+ *
7
+ * The site is resolved server-side (request Host or the dedicated Container
8
+ * App's DCS_SITE_SLUG), so VITE_SITE_SLUG must NOT appear in the request path.
9
+ */
10
+ describe('useTextContent runtime URL (slug-free)', () => {
11
+ beforeEach(() => {
12
+ vi.stubEnv('VITE_API_BASE_URL', 'https://api.example.com')
13
+ vi.stubEnv('VITE_TEXT_OVERRIDE_MODE', 'runtime')
14
+ // Still set for source compatibility — must be ignored for routing.
15
+ vi.stubEnv('VITE_SITE_SLUG', 'kept')
16
+ })
17
+
18
+ afterEach(() => {
19
+ vi.unstubAllEnvs()
20
+ vi.restoreAllMocks()
21
+ })
22
+
23
+ it('fetches /api/v1/pages/{pageSlug}/text without the site slug in the path', async () => {
24
+ const fetchMock = vi.fn().mockResolvedValue({
25
+ ok: true,
26
+ status: 200,
27
+ json: async () => ({ overrides: { 'hero.title': 'Hi' } }),
28
+ })
29
+ vi.stubGlobal('fetch', fetchMock)
30
+
31
+ // fetchOnMount: false avoids registering onMounted outside a component
32
+ // setup context; refresh() drives the same fetch path directly.
33
+ const { refresh } = useTextContent({
34
+ pageSlug: 'home',
35
+ defaults: {},
36
+ fetchOnMount: false,
37
+ })
38
+ await refresh()
39
+
40
+ expect(fetchMock).toHaveBeenCalledTimes(1)
41
+ const [url] = fetchMock.mock.calls[0]
42
+ expect(url).toBe('https://api.example.com/api/v1/pages/home/text')
43
+ expect(url).not.toContain('/sites/')
44
+ expect(url).not.toContain('kept')
45
+ })
46
+ })
@@ -111,7 +111,11 @@ export function useTextContent(config: TextContentConfig): TextContentReturn {
111
111
  cacheTtl = 60000,
112
112
  } = config
113
113
 
114
- // Get configuration from environment
114
+ // Get configuration from environment.
115
+ // @deprecated VITE_SITE_SLUG: the site is now resolved server-side from the
116
+ // request Host or the dedicated Container App's DCS_SITE_SLUG. It is still read
117
+ // here for cache-key continuity and source compatibility, but is no longer used
118
+ // to build the request URL.
115
119
  const apiBaseUrl = getEnvVar('VITE_API_BASE_URL', '')
116
120
  const siteSlug = getEnvVar('VITE_SITE_SLUG', '')
117
121
  const textOverrideMode = getEnvVar('VITE_TEXT_OVERRIDE_MODE', 'commit')
@@ -202,10 +206,12 @@ export function useTextContent(config: TextContentConfig): TextContentReturn {
202
206
  return
203
207
  }
204
208
 
205
- // Skip if API not configured
206
- if (!apiBaseUrl || !siteSlug) {
209
+ // Skip if API not configured. The site is resolved server-side (Host /
210
+ // DCS_SITE_SLUG), so VITE_SITE_SLUG is no longer required for routing —
211
+ // only the API base URL is needed.
212
+ if (!apiBaseUrl) {
207
213
  console.warn(
208
- '[@duffcloudservices/cms] Runtime mode enabled but VITE_API_BASE_URL or VITE_SITE_SLUG not set'
214
+ '[@duffcloudservices/cms] Runtime mode enabled but VITE_API_BASE_URL not set'
209
215
  )
210
216
  return
211
217
  }
@@ -222,7 +228,9 @@ export function useTextContent(config: TextContentConfig): TextContentReturn {
222
228
  error.value = null
223
229
 
224
230
  try {
225
- const url = `${apiBaseUrl}/api/v1/sites/${siteSlug}/pages/${pageSlug}/text`
231
+ // Site is resolved server-side from the request Host or the dedicated
232
+ // Container App's DCS_SITE_SLUG; the slug is no longer encoded in the path.
233
+ const url = `${apiBaseUrl}/api/v1/pages/${pageSlug}/text`
226
234
  const response = await fetch(url, {
227
235
  headers: {
228
236
  Accept: 'application/json',