// Code generated by templ - DO NOT EDIT. // templ: version: v0.3.1001 package bn //lint:file-ignore SA4006 This context is only used if a nested component is present. import "github.com/a-h/templ" import templruntime "github.com/a-h/templ/runtime" import ( "strings" "time" "github.com/google/uuid" ) // resolveMediaURL converts media: prefixed URLs to proper /media/ paths // e.g., "media:abc/image.webp" → "/media/abc/image.webp" func resolveMediaURL(url string) string { if strings.HasPrefix(url, "media:") { return "/media/" + strings.TrimPrefix(url, "media:") } return url } // BrandingData contains generated favicon/icon URLs type BrandingData struct { FaviconICO string // /brand/{id}/favicon.ico Favicon16 string // /brand/{id}/favicon-16x16.png Favicon32 string // /brand/{id}/favicon-32x32.png AppleTouchIcon string // /brand/{id}/apple-touch-icon.png (180x180) Android192 string // /brand/{id}/android-chrome-192x192.png Android512 string // /brand/{id}/android-chrome-512x512.png Maskable512 string // /brand/{id}/maskable-512x512.png Master1024 string // /brand/{id}/icon-1024x1024.png ManifestURL string // /brand/{id}/site.webmanifest ThemeColor string SVG string // Optional SVG pass-through IsGenerated bool } // SiteSettingsData contains site-wide settings for head injection type SiteSettingsData struct { Title string Description string Favicon string Logo string LogoAlt string AppleTouchIcon string GoogleAnalyticsID string CustomHeadScripts string CustomBodyScripts string MetaDescription string DefaultOGImage string TwitterHandle string OGSiteName string Branding BrandingData AdminBypassMode string // "maintenance", "coming_soon", or "" if not bypassing Toolbar ToolbarData LLMsTxtEnabled bool // Whether llms.txt is enabled for AI content discovery TurnstileSiteKey string // Cloudflare Turnstile site key for bot protection RSSFeedURL string // URL to the RSS feed (e.g., "/rss"), empty to disable RSSFeedTitle string // Feed title for discovery link } // PageMeta contains page-level SEO meta data (overrides site defaults) type PageMeta struct { MetaTitle string // Custom SEO title (overrides page title) MetaDescription string // Page-specific meta description OGTitle string // Open Graph title OGDescription string // Open Graph description OGImage string // Open Graph image URL TwitterTitle string // Twitter card title TwitterDescription string // Twitter card description TwitterImage string // Twitter card image URL CanonicalURL string // Canonical URL for this page RobotsDirective string // Robots directive (e.g., "noindex, nofollow") } // HeadData contains all data needed to render the
element type HeadData struct { Title string Settings SiteSettingsData PageMeta PageMeta // Page-level SEO overrides ThemeCSS string ThemeMode string // "light", "dark", or "system" PluginStyles []string // Additional stylesheet URLs StructuredData string // JSON-LD structured data CSSHash string // Cache-busting hash for styles.css PageviewNonce string // Unique nonce for pageview deduplication EngagementConfig EngagementConfig // Engagement tracking config (for blog posts) } // ParseEngagementConfig extracts engagement config from the document map func ParseEngagementConfig(doc map[string]any) EngagementConfig { config := EngagementConfig{} engData, ok := doc["engagement_config"].(map[string]any) if !ok { return config } if v, ok := engData["page_path"].(string); ok { config.PagePath = v } if v, ok := engData["page_id"].(string); ok { config.PageID = v } if v, ok := engData["post_word_count"].(int); ok { config.PostWordCount = v } if v, ok := engData["session_id"].(string); ok { config.SessionID = v } if v, ok := engData["visitor_hash"].(string); ok { config.VisitorHash = v } if v, ok := engData["is_post"].(bool); ok { config.IsPost = v } return config } // ParseSiteSettings extracts site settings from the document map func ParseSiteSettings(doc map[string]any) SiteSettingsData { settings := SiteSettingsData{} siteData, ok := doc["site_settings"].(map[string]any) if !ok { return settings } if v, ok := siteData["title"].(string); ok { settings.Title = v } if v, ok := siteData["description"].(string); ok { settings.Description = v } if v, ok := siteData["favicon"].(string); ok { settings.Favicon = v } if v, ok := siteData["logo"].(string); ok { settings.Logo = v } if v, ok := siteData["logo_alt"].(string); ok { settings.LogoAlt = v } if v, ok := siteData["apple_touch_icon"].(string); ok { settings.AppleTouchIcon = v } // SEO defaults if seoData, ok := siteData["seo_defaults"].(map[string]any); ok { if v, ok := seoData["meta_description"].(string); ok { settings.MetaDescription = v } if v, ok := seoData["default_og_image"].(string); ok { settings.DefaultOGImage = v } if v, ok := seoData["twitter_handle"].(string); ok { settings.TwitterHandle = v } if v, ok := seoData["og_site_name"].(string); ok { settings.OGSiteName = v } } // Analytics if analyticsData, ok := siteData["analytics"].(map[string]any); ok { if v, ok := analyticsData["google_analytics_id"].(string); ok { settings.GoogleAnalyticsID = v } if v, ok := analyticsData["custom_head_scripts"].(string); ok { settings.CustomHeadScripts = v } if v, ok := analyticsData["custom_body_scripts"].(string); ok { settings.CustomBodyScripts = v } } // Branding (generated favicons) if brandingData, ok := siteData["branding"].(map[string]any); ok { if v, ok := brandingData["is_generated"].(bool); ok { settings.Branding.IsGenerated = v } if v, ok := brandingData["favicon_ico"].(string); ok { settings.Branding.FaviconICO = v } if v, ok := brandingData["favicon_16"].(string); ok { settings.Branding.Favicon16 = v } if v, ok := brandingData["favicon_32"].(string); ok { settings.Branding.Favicon32 = v } if v, ok := brandingData["apple_touch_icon"].(string); ok { settings.Branding.AppleTouchIcon = v } if v, ok := brandingData["android_192"].(string); ok { settings.Branding.Android192 = v } if v, ok := brandingData["android_512"].(string); ok { settings.Branding.Android512 = v } if v, ok := brandingData["maskable_512"].(string); ok { settings.Branding.Maskable512 = v } if v, ok := brandingData["master_1024"].(string); ok { settings.Branding.Master1024 = v } if v, ok := brandingData["manifest_url"].(string); ok { settings.Branding.ManifestURL = v } if v, ok := brandingData["theme_color"].(string); ok { settings.Branding.ThemeColor = v } if v, ok := brandingData["svg"].(string); ok { settings.Branding.SVG = v } } // Admin bypass mode (for showing banner when admin bypasses maintenance/coming_soon) if v, ok := siteData["admin_bypass_mode"].(string); ok { settings.AdminBypassMode = v } // Admin editor toolbar if toolbarData, ok := siteData["toolbar"].(map[string]any); ok { settings.Toolbar = ParseToolbarData(toolbarData) } // AI Optimization settings if aiOpt, ok := siteData["aiOptimization"].(map[string]any); ok { if v, ok := aiOpt["llmsTxtEnabled"].(bool); ok { settings.LLMsTxtEnabled = v } } // Turnstile bot protection (from site settings) if turnstileData, ok := siteData["turnstile"].(map[string]any); ok { // Only set site key if Turnstile is enabled if enabled, ok := turnstileData["enabled"].(bool); ok && enabled { if v, ok := turnstileData["site_key"].(string); ok { settings.TurnstileSiteKey = v } } } // RSS feed auto-discovery (injected by page handler from system page) if v, ok := siteData["rss_feed_url"].(string); ok { settings.RSSFeedURL = v } if v, ok := siteData["rss_feed_title"].(string); ok { settings.RSSFeedTitle = v } return settings } // ParsePageMeta extracts page-level SEO metadata from the document map // These fields override site-level defaults when set func ParsePageMeta(doc map[string]any) PageMeta { meta := PageMeta{} // Page-level SEO fields are stored directly in the document root // (matching frontend PageInfo type in web/src/store/editor.ts) if v, ok := doc["metaTitle"].(string); ok { meta.MetaTitle = v } if v, ok := doc["metaDescription"].(string); ok { meta.MetaDescription = v } if v, ok := doc["ogTitle"].(string); ok { meta.OGTitle = v } if v, ok := doc["ogDescription"].(string); ok { meta.OGDescription = v } if v, ok := doc["ogImage"].(string); ok { meta.OGImage = v } if v, ok := doc["twitterTitle"].(string); ok { meta.TwitterTitle = v } if v, ok := doc["twitterDescription"].(string); ok { meta.TwitterDescription = v } if v, ok := doc["twitterImage"].(string); ok { meta.TwitterImage = v } if v, ok := doc["canonicalUrl"].(string); ok { meta.CanonicalURL = v } if v, ok := doc["robotsDirective"].(string); ok { meta.RobotsDirective = v } return meta } // ParseToolbarData extracts toolbar data from the document map func ParseToolbarData(data map[string]any) ToolbarData { toolbar := ToolbarData{} if v, ok := data["enabled"].(bool); ok { toolbar.Enabled = v } if v, ok := data["page_id"].(string); ok { if id, err := uuid.Parse(v); err == nil { toolbar.PageID = id } } if v, ok := data["page_slug"].(string); ok { toolbar.PageSlug = v } if v, ok := data["page_title"].(string); ok { toolbar.PageTitle = v } if v, ok := data["post_type"].(string); ok { toolbar.PostType = v } if v, ok := data["status"].(string); ok { toolbar.Status = v } if v, ok := data["has_unpublished_changes"].(bool); ok { toolbar.HasUnpublishedChanges = v } if v, ok := data["preview_mode"].(string); ok { toolbar.PreviewMode = v } if v, ok := data["position"].(string); ok { toolbar.Position = v } if v, ok := data["template_name"].(string); ok { toolbar.TemplateName = v } if v, ok := data["author_name"].(string); ok { toolbar.AuthorName = v } if v, ok := data["author_slug"].(string); ok { toolbar.AuthorSlug = v } if v, ok := data["last_modified"].(time.Time); ok { toolbar.LastModified = v } if v, ok := data["edit_url"].(string); ok { toolbar.EditURL = v } if v, ok := data["settings_url"].(string); ok { toolbar.SettingsURL = v } if v, ok := data["history_url"].(string); ok { toolbar.HistoryURL = v } if v, ok := data["analytics_url"].(string); ok { toolbar.AnalyticsURL = v } // Analytics snapshot if v, ok := data["today_pageviews"].(int64); ok { toolbar.TodayPageviews = v } if v, ok := data["pageviews_trend"].(string); ok { toolbar.PageviewsTrend = v } if v, ok := data["trend_percent"].(int); ok { toolbar.TrendPercent = v } // Blog-specific fields if v, ok := data["reading_time"].(int); ok { toolbar.ReadingTime = v } if v, ok := data["word_count"].(int); ok { toolbar.WordCount = v } if v, ok := data["category_count"].(int); ok { toolbar.CategoryCount = v } return toolbar } // EffectiveTitle returns the title to use in the