Error executing template "Designs/Swift/_parsed/Swift_Page.parsed.cshtml"
System.Data.SqlClient.SqlException (0x80131904): Login failed for user 'cerama.cloud.dynamicweb-cms.com'. Reason: Server is in single user mode. Only one administrator can connect at this time.
   at System.Data.ProviderBase.DbConnectionPool.TryGetConnection(DbConnection owningObject, UInt32 waitForMultipleObjectsTimeout, Boolean allowCreate, Boolean onlyOneCheckConnection, DbConnectionOptions userOptions, DbConnectionInternal& connection)
   at System.Data.ProviderBase.DbConnectionPool.TryGetConnection(DbConnection owningObject, TaskCompletionSource`1 retry, DbConnectionOptions userOptions, DbConnectionInternal& connection)
   at System.Data.ProviderBase.DbConnectionFactory.TryGetConnection(DbConnection owningConnection, TaskCompletionSource`1 retry, DbConnectionOptions userOptions, DbConnectionInternal oldConnection, DbConnectionInternal& connection)
   at System.Data.ProviderBase.DbConnectionInternal.TryOpenConnectionInternal(DbConnection outerConnection, DbConnectionFactory connectionFactory, TaskCompletionSource`1 retry, DbConnectionOptions userOptions)
   at System.Data.SqlClient.SqlConnection.TryOpenInner(TaskCompletionSource`1 retry)
   at System.Data.SqlClient.SqlConnection.TryOpen(TaskCompletionSource`1 retry)
   at System.Data.SqlClient.SqlConnection.Open()
   at Dynamicweb.Data.DatabaseConnectionProvider.CreateConnection(Boolean open)
   at Dynamicweb.Data.Database.CreateConnection()
   at Dynamicweb.Data.Database.CreateDataReader(CommandBuilder commandBuilder, IDbConnection connection, IDbTransaction transaction, Int32 commandTimeout)
   at Dynamicweb.Ecommerce.Products.ProductRepository.GetProductById(String productId, String productVariantId, String productLanguageId)
   at Dynamicweb.Ecommerce.Products.ProductService.FetchMissingProductsInternal(IProductRepository repo, IEnumerable`1 keys)
   at Dynamicweb.Caching.ServiceCache`2.GetCache(IEnumerable`1 keys)
   at Dynamicweb.Caching.ServiceCache`2.GetCache(TKey key)
   at Dynamicweb.Ecommerce.Products.ProductService.GetProductById(String productId, String productVariantId, String productLanguageId, User user, Boolean showUntranslated)
   at Dynamicweb.Ecommerce.Products.ProductService.GetProductById(String productId, String productVariantId, String productLanguageId, Boolean useAssortments)
   at Dynamicweb.Ecommerce.Products.ProductService.GetProductById(String productId, String productVariantId, String productLanguageId)
   at CompiledRazorTemplates.Dynamic.RazorEngine_5519d13845614cdaa98c0cd07451fb66.Execute() in D:\dynamicweb.net\Solutions\cerama.cloud.dynamicweb-cms.com\Files\Templates\Designs\Swift\_parsed\Swift_Page.parsed.cshtml:line 460
   at RazorEngine.Templating.TemplateBase.RazorEngine.Templating.ITemplate.Run(ExecuteContext context, TextWriter reader)
   at RazorEngine.Templating.RazorEngineService.RunCompile(ITemplateKey key, TextWriter writer, Type modelType, Object model, DynamicViewBag viewBag)
   at RazorEngine.Templating.RazorEngineServiceExtensions.<>c__DisplayClass16_0.<RunCompile>b__0(TextWriter writer)
   at RazorEngine.Templating.RazorEngineServiceExtensions.WithWriter(Action`1 withWriter)
   at Dynamicweb.Rendering.RazorTemplateRenderingProvider.Render(Template template)
   at Dynamicweb.Rendering.TemplateRenderingService.Render(Template template)
   at Dynamicweb.Rendering.Template.RenderRazorTemplate()
ClientConnectionId:f6cad6de-b6b1-4be8-8f03-d6cc985841d2
Error Number:18461,State:1,Class:14

1 @inherits Dynamicweb.Rendering.ViewModelTemplate<Dynamicweb.Frontend.PageViewModel> 2 @using System 3 @using Dynamicweb 4 @using Dynamicweb.Core.Encoders 5 @using Dynamicweb.Environment 6 @using Dynamicweb.Frontend 7 8 @{ 9 var brandingPageId = Model.Area.Item?.GetInt32("BrandingPage") ?? 0; 10 var themePageId = Model.Area.Item?.GetInt32("ThemesPage") ?? 0; 11 var cssPageId = Model.Area.Item?.GetInt32("CssPage") ?? 0; 12 var brandingPage = brandingPageId != 0 ? Dynamicweb.Content.Services.Pages?.GetPage(brandingPageId) ?? null : null; 13 var themesParagraphs = themePageId != 0 ? Dynamicweb.Content.Services.Paragraphs?.GetParagraphsByPageId(themePageId) ?? null : null; 14 var cssParagraphs = cssPageId != 0 ? Dynamicweb.Content.Services.Paragraphs?.GetParagraphsByPageId(cssPageId) ?? null : null; 15 } 16 17 @if (themesParagraphs != null || brandingPage != null) 18 { 19 string swiftVersion = ReadFile("/Files/Templates/Designs/Swift/swift_version.txt"); 20 bool renderAsResponsive = Model.Area.Item.GetString("DeviceRendering", "responsive").Equals("responsive", StringComparison.OrdinalIgnoreCase); 21 bool renderMobile = Pageview.Device == Dynamicweb.Frontend.Devices.DeviceType.Mobile || Pageview.Device == Dynamicweb.Frontend.Devices.DeviceType.Tablet; 22 string responsiveClassDesktop = string.Empty; 23 string responsiveClassMobile = string.Empty; 24 if (renderAsResponsive) 25 { 26 responsiveClassDesktop = " d-none d-xl-block"; 27 responsiveClassMobile = " d-block d-xl-none"; 28 } 29 30 var headerDesktopLink = Model.Area.Item?.GetLink("HeaderDesktop") ?? null; 31 var headerMobileLink = Model.Area.Item?.GetLink("HeaderMobile") ?? null; 32 33 var footerDesktopLink = Model.Area.Item?.GetLink("FooterDesktop") ?? null; 34 var footerMobileLink = Model.Area.Item?.GetLink("FooterMobile") ?? null; 35 36 var disableWideBreakpoints = Model.Area?.Item?.GetRawValueString("DisableWideBreakpoints", "default"); 37 38 string customHeaderInclude = !string.IsNullOrEmpty(Model.Area.Item.GetRawValueString("CustomHeaderInclude")) ? Model.Area.Item.GetFile("CustomHeaderInclude").Name : string.Empty; 39 40 var themesParagraphLastChanged = Dynamicweb.Content.Services.Paragraphs.GetParagraphsByPageId(themePageId).OrderByDescending(p => p.Audit.LastModifiedAt).FirstOrDefault(); 41 var cssLastModified = brandingPage.Audit.LastModifiedAt > themesParagraphLastChanged.Audit.LastModifiedAt ? brandingPage.Audit.LastModifiedAt : themesParagraphLastChanged.Audit.LastModifiedAt; 42 43 var cssThemeAndBrandingStyleFileInfo = new System.IO.FileInfo(Dynamicweb.Core.SystemInformation.MapPath($"/Files/Templates/Designs/Swift/_parsed/Swift_css/Swift_styles_{Model.Area.ID}.min.css")); 44 45 46 if (cssPageId != 0) 47 { 48 var cssFileInfo = new System.IO.FileInfo(Dynamicweb.Core.SystemInformation.MapPath($"/Files/Templates/Designs/Swift/_parsed/Swift_css/Swift_css_styles_{Model.Area.ID}.css")); 49 var cssParagraphLastChanged = Dynamicweb.Content.Services.Paragraphs.GetParagraphsByPageId(cssPageId).OrderByDescending(p => p.Audit.LastModifiedAt).FirstOrDefault(); 50 if (!cssThemeAndBrandingStyleFileInfo.Exists || cssThemeAndBrandingStyleFileInfo.LastWriteTime < cssParagraphLastChanged.Audit.LastModifiedAt) 51 { 52 var cssPageview = Dynamicweb.Frontend.PageView.GetPageviewByPageID(cssPageId); 53 cssPageview.Redirect = false; 54 cssPageview.Output(); 55 } 56 } 57 58 if (!cssThemeAndBrandingStyleFileInfo.Exists || cssThemeAndBrandingStyleFileInfo.LastWriteTime < brandingPage.Audit.LastModifiedAt) 59 { 60 //Branding page has been saved or the file is missing. Rewrite the file to disc. 61 if (brandingPageId > 0) 62 { 63 var brandingPageview = Dynamicweb.Frontend.PageView.GetPageviewByPageID(brandingPageId); 64 brandingPageview.Redirect = false; 65 brandingPageview.Output(); 66 } 67 } 68 69 if (!cssThemeAndBrandingStyleFileInfo.Exists || cssThemeAndBrandingStyleFileInfo.LastWriteTime < themesParagraphLastChanged.Audit.LastModifiedAt) 70 { 71 //Branding page has been saved or the file is missing. Rewrite the file to disc. 72 if (themePageId > 0) 73 { 74 var themePageview = Dynamicweb.Frontend.PageView.GetPageviewByPageID(themePageId); 75 themePageview.Redirect = false; 76 themePageview.Output(); 77 } 78 } 79 80 // Schema.org details for PDP 81 bool isProductDetailsPage = Dynamicweb.Context.Current.Request.QueryString.AllKeys.Contains("ProductID"); 82 bool isArticlePage = Model.ItemType == "Swift_Article"; 83 string schemaOrgType = string.Empty; 84 85 if (isProductDetailsPage) 86 { 87 schemaOrgType = "itemscope=\"\" itemtype=\"https://schema.org/Product\""; 88 } 89 90 if (isArticlePage) 91 { 92 schemaOrgType = "itemscope=\"\" itemtype=\"https://schema.org/Article\""; 93 } 94 95 96 var cssStyleFileInfo = new System.IO.FileInfo(Dynamicweb.Core.SystemInformation.MapPath("/Files/Templates/Designs/Swift/Assets/css/styles.css")); 97 var jsFileInfo = new System.IO.FileInfo(Dynamicweb.Core.SystemInformation.MapPath("/Files/Templates/Designs/Swift/Assets/js/scripts.js")); 98 99 string masterTheme = !string.IsNullOrWhiteSpace(Model.Area.Item.GetRawValueString("Theme")) ? " theme " + Model.Area.Item.GetRawValueString("Theme").Replace(" ", "").Trim().ToLower() : ""; 100 101 string favicon = Model.Area.Item.GetRawValueString("Favicon", "/Files/Templates/Designs/Swift/Assets/Images/favicon.png"); 102 103 string headerCssClass = "sticky-top"; 104 bool movePageBehind = false; 105 106 if (Model.PropertyItem != null) 107 { 108 headerCssClass = Model.PropertyItem.GetRawValueString("MoveThisPageBehindTheHeader", "sticky-top"); 109 movePageBehind = headerCssClass == "fixed-top" && !Pageview.IsVisualEditorMode ? true : false; 110 } 111 112 headerCssClass = headerCssClass == "" ? "sticky-top" : headerCssClass; 113 headerCssClass = Pageview.IsVisualEditorMode ? "" : headerCssClass; 114 115 string googleTagManagerID = Model.Area.Item.GetString("GoogleTagManagerID"); 116 string googleAnalyticsMeasurementID = Model.Area.Item.GetString("GoogleAnalyticsMeasurementID"); 117 118 bool allowTracking = true; 119 if (CookieManager.IsCookieManagementActive) 120 { 121 var cookieOptInLevel = CookieManager.GetCookieOptInLevel(); 122 allowTracking = cookieOptInLevel == CookieOptInLevel.All || (cookieOptInLevel == CookieOptInLevel.Functional && CookieManager.GetCookieOptInCategories().Contains("Statistical")); 123 } 124 125 Dynamicweb.Context.Current.Response.AddHeader("link", $"</Files/Templates/Designs/Swift/Assets/css/styles.css?{cssStyleFileInfo.LastWriteTime.Ticks}>; rel=preload; as=style;"); 126 Dynamicweb.Context.Current.Response.AddHeader("link", $"</Files/Templates/Designs/Swift/_parsed/Swift_css/Swift_styles_{Model.Area.ID}.min.css?{cssLastModified.Ticks}>; rel=preload; as=style;"); 127 Dynamicweb.Context.Current.Response.AddHeader("link", $"</Files/Templates/Designs/Swift/Assets/js/scripts.js?{jsFileInfo.LastWriteTime.Ticks}>; rel=preload; as=script;"); 128 129 130 SetMetaTags(); 131 132 List<Dynamicweb.Content.Page> languages = new List<Dynamicweb.Content.Page>(); 133 134 var masterPage = Pageview.Area.IsMaster ? Pageview.Page : Pageview.Page.MasterPage; 135 languages.Add(masterPage); 136 if (masterPage?.Languages != null) 137 { 138 foreach (var language in masterPage.Languages) 139 { 140 languages.Add(language); 141 } 142 } 143 144 Uri url = Dynamicweb.Context.Current.Request.Url; 145 string hostName = url.Host; 146 147 <!doctype html> 148 <html lang="@Pageview.Area.CultureInfo.TwoLetterISOLanguageName"> 149 <head> 150 <!-- @swiftVersion --> 151 @* Required meta tags *@ 152 <meta charset="utf-8"> 153 <meta name="viewport" content="height=device-height, width=device-width, initial-scale=1.0"> 154 <link rel="shortcut icon" href="@favicon"> 155 <link rel="apple-touch-icon" href="/Files/Templates/Designs/Swift/Assets/Images/logo_transparent.png"> 156 157 @Model.MetaTags 158 159 @{ 160 var alreadyWrittenTwoletterIsos = new List<string>(); 161 @* Languages meta data *@ 162 foreach (var language in languages) 163 { 164 hostName = url.Host; 165 if (language?.Area != null) 166 { 167 if (language.Area?.MasterArea != null && !string.IsNullOrEmpty(language.Area.MasterArea.DomainLock)) 168 { 169 hostName = language.Area.MasterArea.DomainLock; //dk.domain.com or dk-domain.dk 170 } 171 if (language != null && language.Published && language.Area.Active && language.Area.Published) 172 { 173 if (!string.IsNullOrEmpty(language.Area.DomainLock)) 174 { 175 hostName = language.Area.DomainLock; //dk.domain.com or dk-domain.dk 176 } 177 string querystring = $"Default.aspx?ID={language.ID}"; 178 if (!string.IsNullOrEmpty(Dynamicweb.Context.Current.Request.QueryString["GroupID"])) 179 { 180 querystring += $"&GroupID={Dynamicweb.Context.Current.Request.QueryString["GroupID"]}"; 181 } 182 if (!string.IsNullOrEmpty(Dynamicweb.Context.Current.Request.QueryString["ProductID"])) 183 { 184 querystring += $"&ProductID={Dynamicweb.Context.Current.Request.QueryString["ProductID"]}"; 185 } 186 if (!string.IsNullOrEmpty(Dynamicweb.Context.Current.Request.QueryString["VariantID"])) 187 { 188 querystring += $"&VariantID={Dynamicweb.Context.Current.Request.QueryString["VariantID"]}"; 189 } 190 191 string friendlyUrl = Dynamicweb.Frontend.SearchEngineFriendlyURLs.GetFriendlyUrl(querystring); 192 if (language.Area.RedirectFirstPage && language.ParentPageId == 0 && language.Sort == 1) 193 { 194 friendlyUrl = "/"; 195 } 196 string href = $"{url.Scheme}://{hostName}{friendlyUrl}"; 197 198 199 <link rel="alternate" hreflang="@language.Area.CultureInfo.Name.ToLower()" href="@href"> 200 if (!alreadyWrittenTwoletterIsos.Contains(language.Area.CultureInfo.TwoLetterISOLanguageName)) 201 { 202 <link rel="alternate" hreflang="@language.Area.CultureInfo.TwoLetterISOLanguageName.ToLower()" href="@href"> 203 } 204 } 205 } 206 } 207 } 208 209 <title>@Model.Title</title> 210 @* Bootstrap + Swift stylesheet *@ 211 <link href="/Files/Templates/Designs/Swift/Assets/css/styles.css?@cssStyleFileInfo.LastWriteTime.Ticks" rel="stylesheet" media="all" type="text/css"> 212 213 @if (disableWideBreakpoints != "disableBoth") 214 { 215 <style> 216 @@media ( min-width: 1600px ) { 217 .container-xxl, 218 .container-xl, 219 .container-lg, 220 .container-md, 221 .container-sm, 222 .container { 223 max-width: 1520px; 224 } 225 } 226 </style> 227 228 229 230 if (disableWideBreakpoints != "disableUltraWideOnly") 231 { 232 <style> 233 @@media ( min-width: 1920px ) { 234 .container-xxl, 235 .container-xl, 236 .container-lg, 237 .container-md, 238 .container-sm, 239 .container { 240 max-width: 1820px; 241 } 242 } 243 </style> 244 } 245 } 246 247 @* Branding and Themes min stylesheet *@ 248 <link href="/Files/Templates/Designs/Swift/_parsed/Swift_css/Swift_styles_@(Model.Area.ID).min.css?@cssLastModified.Ticks" rel="stylesheet" media="all" type="text/css" data-last-modified-content="@cssLastModified"> 249 <script src="/Files/Templates/Designs/Swift/Assets/js/scripts.js?@jsFileInfo.LastWriteTime.Ticks" defer></script> 250 251 <script type="module"> 252 swift.Scroll.hideHeadersOnScroll(); 253 swift.Scroll.handleAlternativeTheme(); 254 255 window.addEventListener('load', () => { 256 const aosColumns = document.querySelectorAll('[data-aos]'); 257 if (aosColumns.length > 0) { 258 swift.AssetLoader.Load('/Files/Templates/Designs/Swift/Assets/js/aos.js?@jsFileInfo.LastWriteTime.Ticks', 'js'); 259 document.addEventListener('load.swift.assetloader', function () { 260 AOS.init({ duration: 400, delay: 100, easing: 'ease-in-out', mirror: false, disable: window.matchMedia('(prefers-reduced-motion: reduce)') }); 261 }); 262 } 263 }) 264 </script> 265 266 @* Google tag manager *@ 267 @if (!string.IsNullOrWhiteSpace(googleTagManagerID) && allowTracking) 268 { 269 <script> 270 (function(w,d,s,l,i){w[l]=w[l]||[];w[l].push({'gtm.start': 271 new Date().getTime(),event:'gtm.js'});var f=d.getElementsByTagName(s)[0], 272 j=d.createElement(s),dl=l!='dataLayer'?'&l='+l:'';j.async=true;j.src= 273 'https://www.googletagmanager.com/gtm.js?id='+i+dl;f.parentNode.insertBefore(j,f); 274 })(window, document, 'script', 'dataLayer', '@(googleTagManagerID)'); 275 276 function gtag() { dataLayer.push(arguments); } 277 </script> 278 } 279 280 @if (!string.IsNullOrWhiteSpace(googleAnalyticsMeasurementID) && allowTracking) 281 { 282 var GoogleAnalyticsDebugMode = ""; 283 284 if (Model.Area.Item.GetBoolean("EnableGoogleAnalyticsDebugMode")) 285 { 286 GoogleAnalyticsDebugMode = ", {'debug_mode': true}"; 287 } 288 289 <script async src="https://www.googletagmanager.com/gtag/js?id=@googleAnalyticsMeasurementID"></script> 290 <script> 291 window.dataLayer = window.dataLayer || []; 292 function gtag() { dataLayer.push(arguments); } 293 gtag('js', new Date()); 294 gtag('config', '@googleAnalyticsMeasurementID'@GoogleAnalyticsDebugMode); 295 </script> 296 } 297 298 @if (!string.IsNullOrWhiteSpace(customHeaderInclude)) 299 { 300 @RenderPartial($"Components/Custom/{customHeaderInclude}") 301 } 302 </head> 303 <body class="brand @(masterTheme)" id="page@(Model.ID)"> 304 @{ 305 var organizationItem = Model.Area.Item?.GetItem("Custom")?.GetItem("Custom_SchemaOrg_Organization"); 306 if (organizationItem != null) 307 { 308 string orgName = organizationItem.GetString("Name"); 309 if (!string.IsNullOrWhiteSpace(orgName)) 310 { 311 string orgLegalName = organizationItem.GetString("LegalName"); 312 string orgUrl = organizationItem.GetString("Url"); 313 if (string.IsNullOrWhiteSpace(orgUrl)) 314 { 315 orgUrl = $"{Dynamicweb.Context.Current.Request.Url.Scheme}://{Dynamicweb.Context.Current.Request.Url.Host}/"; 316 } 317 var orgLogoFile = organizationItem.GetFile("Logo"); 318 string orgLogoUrl = orgLogoFile != null 319 ? $"{Dynamicweb.Context.Current.Request.Url.Scheme}://{Dynamicweb.Context.Current.Request.Url.Host}{orgLogoFile.Path}" 320 : string.Empty; 321 string orgTelephone = organizationItem.GetString("Telephone"); 322 string orgEmail = organizationItem.GetString("Email"); 323 string orgStreet = organizationItem.GetString("StreetAddress"); 324 string orgPostal = organizationItem.GetString("PostalCode"); 325 string orgCity = organizationItem.GetString("AddressLocality"); 326 string orgRegion = organizationItem.GetString("AddressRegion"); 327 string orgCountry = organizationItem.GetString("AddressCountry"); 328 string orgVatID = organizationItem.GetString("VatID"); 329 330 var sameAsUrls = new List<string>(); 331 foreach (var key in new[] { "SameAsFacebook", "SameAsLinkedIn", "SameAsInstagram", "SameAsYouTube", "SameAsTwitter" }) 332 { 333 var v = organizationItem.GetString(key); 334 if (!string.IsNullOrWhiteSpace(v)) { sameAsUrls.Add(v.Trim()); } 335 } 336 var otherSameAs = organizationItem.GetString("SameAsOther"); 337 if (!string.IsNullOrWhiteSpace(otherSameAs)) 338 { 339 foreach (var line in otherSameAs.Split(new[] { '\n', '\r' }, StringSplitOptions.RemoveEmptyEntries)) 340 { 341 var trimmed = line.Trim(); 342 if (trimmed.Length > 0) { sameAsUrls.Add(trimmed); } 343 } 344 } 345 346 bool hasAddress = !string.IsNullOrWhiteSpace(orgStreet) 347 || !string.IsNullOrWhiteSpace(orgPostal) 348 || !string.IsNullOrWhiteSpace(orgCity) 349 || !string.IsNullOrWhiteSpace(orgRegion) 350 || !string.IsNullOrWhiteSpace(orgCountry); 351 352 <div itemscope itemtype="https://schema.org/Organization" hidden> 353 <meta itemprop="name" content="@HtmlEncoder.HtmlAttributeEncode(orgName)"> 354 @if (!string.IsNullOrWhiteSpace(orgLegalName)) 355 { 356 <meta itemprop="legalName" content="@HtmlEncoder.HtmlAttributeEncode(orgLegalName)"> 357 } 358 <link itemprop="url" href="@HtmlEncoder.HtmlAttributeEncode(orgUrl)"> 359 @if (!string.IsNullOrWhiteSpace(orgLogoUrl)) 360 { 361 <link itemprop="logo" href="@HtmlEncoder.HtmlAttributeEncode(orgLogoUrl)"> 362 } 363 @if (!string.IsNullOrWhiteSpace(orgTelephone)) 364 { 365 <meta itemprop="telephone" content="@HtmlEncoder.HtmlAttributeEncode(orgTelephone)"> 366 } 367 @if (!string.IsNullOrWhiteSpace(orgEmail)) 368 { 369 <meta itemprop="email" content="@HtmlEncoder.HtmlAttributeEncode(orgEmail)"> 370 } 371 @if (hasAddress) 372 { 373 <div itemprop="address" itemscope itemtype="https://schema.org/PostalAddress"> 374 @if (!string.IsNullOrWhiteSpace(orgStreet)) { <meta itemprop="streetAddress" content="@HtmlEncoder.HtmlAttributeEncode(orgStreet)"> } 375 @if (!string.IsNullOrWhiteSpace(orgPostal)) { <meta itemprop="postalCode" content="@HtmlEncoder.HtmlAttributeEncode(orgPostal)"> } 376 @if (!string.IsNullOrWhiteSpace(orgCity)) { <meta itemprop="addressLocality" content="@HtmlEncoder.HtmlAttributeEncode(orgCity)"> } 377 @if (!string.IsNullOrWhiteSpace(orgRegion)) { <meta itemprop="addressRegion" content="@HtmlEncoder.HtmlAttributeEncode(orgRegion)"> } 378 @if (!string.IsNullOrWhiteSpace(orgCountry)) { <meta itemprop="addressCountry" content="@HtmlEncoder.HtmlAttributeEncode(orgCountry)"> } 379 </div> 380 } 381 @if (!string.IsNullOrWhiteSpace(orgVatID)) 382 { 383 <meta itemprop="vatID" content="@HtmlEncoder.HtmlAttributeEncode(orgVatID)"> 384 } 385 @foreach (var sameAsUrl in sameAsUrls) 386 { 387 <link itemprop="sameAs" href="@HtmlEncoder.HtmlAttributeEncode(sameAsUrl)"> 388 } 389 </div> 390 } 391 } 392 } 393 394 @RenderAnnouncementBarCustom() 395 396 @* Google tag manager *@ 397 @if (!string.IsNullOrWhiteSpace(googleTagManagerID) && allowTracking) 398 { 399 <noscript> 400 <iframe src="https://www.googletagmanager.com/ns.html?id=@(googleTagManagerID)" 401 height="0" width="0" style="display:none;visibility:hidden"></iframe> 402 </noscript> 403 } 404 405 @if (renderAsResponsive || !renderMobile) 406 { 407 <header class="js-page-header-custom page-header @headerCssClass @(responsiveClassDesktop)" id="page-header-desktop"> 408 @if (headerDesktopLink != null) 409 { 410 @RenderGrid(headerDesktopLink.PageId) 411 } 412 </header> 413 } 414 415 @if ((renderAsResponsive || renderMobile)) 416 { 417 <header class="js-page-header-custom page-header @headerCssClass @(responsiveClassMobile)" id="page-header-mobile"> 418 @if (headerMobileLink != null) 419 { 420 @RenderGrid(headerMobileLink.PageId) 421 } 422 </header> 423 } 424 425 <main id="content" @(schemaOrgType)> 426 <div data-intersect></div> 427 @inherits Dynamicweb.Rendering.ViewModelTemplate<Dynamicweb.Frontend.PageViewModel> 428 @using System 429 @using Dynamicweb.Ecommerce.ProductCatalog 430 431 432 @{ 433 string productIdFromUrl = !string.IsNullOrEmpty(Dynamicweb.Context.Current.Request.QueryString.Get("ProductID")) ? Dynamicweb.Context.Current.Request.QueryString.Get("ProductID") : string.Empty; 434 bool isProductDetail = !string.IsNullOrEmpty(productIdFromUrl) && Pageview.Page.NavigationTag.ToLower() == "shop"; 435 436 bool isArticlePagePage = Model.ItemType == "Swift_Article"; 437 bool isArticleListPage = Model.ItemType == "Swift_ArticleListPage"; 438 string schemaOrgProp = string.Empty; 439 if(isArticlePagePage) 440 { 441 schemaOrgProp = "itemprop=\"articleBody\""; 442 } 443 444 string theme = ""; 445 string gridContent = ""; 446 447 if (Model.PropertyItem != null) 448 { 449 theme = !string.IsNullOrWhiteSpace(Model.PropertyItem.GetRawValueString("Theme")) ? "theme " + Model.PropertyItem.GetRawValueString("Theme").Replace(" ", "").Trim().ToLower() : ""; 450 } 451 452 if (Model.Item != null || Pageview.IsVisualEditorMode) 453 { 454 if (!isProductDetail) 455 { 456 gridContent = Model.Grid("Grid", "Grid", "default:true;sort:1", "Page"); 457 } 458 else 459 { 460 var productObject = Dynamicweb.Ecommerce.Services.Products.GetProductById(productIdFromUrl, "", Pageview.Area.EcomLanguageId); 461 var detailPage = Dynamicweb.Ecommerce.Services.ProductGroups.GetGroup(productObject.PrimaryGroupId)?.Meta.PrimaryPage ?? string.Empty; 462 var detailPageId = detailPage != string.Empty ? Convert.ToInt16(detailPage.Substring(detailPage.LastIndexOf('=') + 1)) : GetPageIdByNavigationTag("ProductDetailPage"); 463 464 @RenderGrid(detailPageId) 465 } 466 } 467 468 bool doNotRenderPage = false; 469 470 //Check if we are on the poduct detail page, and if there is data to render 471 ProductViewModel product = new ProductViewModel(); 472 if (Dynamicweb.Context.Current.Items.Contains("ProductDetails")) 473 { 474 product = (ProductViewModel)Dynamicweb.Context.Current.Items["ProductDetails"]; 475 if (string.IsNullOrEmpty(product.Id)) { 476 doNotRenderPage = true; 477 } 478 } 479 480 //Render the page 481 if (!doNotRenderPage) { 482 string itemIdentifier = Model?.Item?.SystemName != null ? "item_" + Model.Item.SystemName.ToLower() : "item_Swift_Page"; 483 484 485 <div class="@theme @itemIdentifier" @schemaOrgProp> 486 @if (isArticleListPage) 487 { 488 var hx = $"hx-get=\"{Dynamicweb.Frontend.SearchEngineFriendlyURLs.GetFriendlyUrl(Model.ID)}\" hx-select=\"#content\" hx-target=\"#content\" hx-swap=\"outerHTML\" hx-trigger=\"change\" hx-headers='{{\"feed\": \"true\"}}' hx-push-url=\"true\" hx-indicator=\"#ArticleFacetForm\""; 489 490 <form @hx id="ArticleFacetForm"> 491 @gridContent 492 </form> 493 <script type="module" src="/Files/Templates/Designs/Swift/Assets/js/htmx.js"></script> 494 <script type="module"> 495 document.addEventListener('htmx:confirm', (event) => { 496 let filters = event.detail.elt.querySelectorAll('select'); 497 for (var i = 0; i < filters.length; i++) { 498 let input = filters[i]; 499 if (input.name && !input.value) { 500 input.name = ''; 501 } 502 } 503 }); 504 505 document.addEventListener('htmx:beforeOnLoad', (event) => { 506 swift.Scroll.stopIntersectionObserver(); 507 }); 508 509 document.addEventListener('htmx:afterOnLoad', () => { 510 swift.Scroll.hideHeadersOnScroll(); 511 swift.Scroll.handleAlternativeTheme(); 512 }); 513 </script> 514 } 515 else 516 { 517 @gridContent 518 } 519 </div> 520 521 } else { 522 <div class="container"> 523 <div class="alert alert-info" role="alert">@Translate("Sorry. There is nothing to view here")</div> 524 </div> 525 } 526 527 if (!Model.IsCurrentUserAllowed) 528 { 529 int signInPage = GetPageIdByNavigationTag("SignInPage"); 530 int dashboardPage = GetPageIdByNavigationTag("MyAccountDashboardPage"); 531 532 if (!Pageview.IsVisualEditorMode) 533 { 534 if (signInPage != 0) 535 { 536 if (signInPage != Model.ID) { 537 Dynamicweb.Context.Current.Response.Redirect("/Default.aspx?ID=" + signInPage); 538 } else { 539 if (dashboardPage != 0) { 540 Dynamicweb.Context.Current.Response.Redirect("/Default.aspx?ID=" + dashboardPage); 541 } else { 542 Dynamicweb.Context.Current.Response.Redirect("/"); 543 } 544 } 545 } 546 else 547 { 548 <div class="alert alert-dark m-0" role="alert"> 549 <span>@Translate("You do not have access to this page")</span> 550 </div> 551 } 552 } 553 else 554 { 555 <div class="alert alert-dark m-0" role="alert"> 556 <span>@Translate("To work on this page, you must be signed in, in the frontend")</span> 557 </div> 558 } 559 } 560 } 561 562 </main> 563 564 @if (renderAsResponsive || !renderMobile) 565 { 566 <footer class="page-footer@(responsiveClassDesktop)" id="page-footer-desktop"> 567 @if (footerDesktopLink != null) 568 { 569 @RenderGrid(footerDesktopLink.PageId) 570 } 571 </footer> 572 } 573 574 @if (renderAsResponsive || renderMobile) 575 { 576 <footer class="page-footer@(responsiveClassMobile)" id="page-footer-mobile"> 577 @if (footerMobileLink != null) 578 { 579 @RenderGrid(footerMobileLink.PageId) 580 } 581 </footer> 582 } 583 584 @* Render any offcanvas menu here *@ 585 @RenderSnippet("offcanvas") 586 587 @{ 588 bool isErpConnectionDown = !Dynamicweb.Core.Converter.ToBoolean(Context.Current.Items["IsWebServiceConnectionAvailable"]); 589 } 590 591 @* Language selector modal *@ 592 <div class="modal fade" id="PreferencesModal" tabindex="-1" aria-hidden="true"> 593 <div class="modal-dialog modal-dialog-centered modal-sm" id="PreferencesModalContent"> 594 @* The content here comes from an external request *@ 595 </div> 596 </div> 597 598 @* Favorite toast *@ 599 <div aria-live="polite" aria-atomic="true"> 600 <div class="position-fixed bottom-0 end-0 p-3" style="z-index: 11"> 601 <div id="favoriteNotificationToast" class="toast" role="alert" aria-live="assertive" aria-atomic="true"> 602 <div class="toast-header"> 603 <strong class="me-auto">@Translate("Favorite list updated")</strong> 604 <button type="button" class="btn-close" data-bs-dismiss="toast" aria-label="Close"></button> 605 </div> 606 <div class="toast-body d-flex gap-3"> 607 <div id="favoriteNotificationToast_Image"></div> 608 <div id="favoriteNotificationToast_Text"></div> 609 </div> 610 </div> 611 </div> 612 </div> 613 614 @* Modal for dynamic content *@ 615 <div class="modal fade js-product" id="DynamicModal" tabindex="-1" aria-hidden="true"> 616 <div class="modal-dialog modal-dialog-centered modal-md"> 617 <div class="modal-content theme light" id="DynamicModalContent"> 618 @* The content here comes from an external request *@ 619 </div> 620 </div> 621 </div> 622 623 @* Offcanvas for dynamic content *@ 624 <div class="offcanvas offcanvas-end theme light" tabindex="-1" id="DynamicOffcanvas" style="width: 30rem"> 625 @* The content here comes from an external request *@ 626 </div> 627 628 @if (Model.Area.Item.GetBoolean("ShowErpDownMessage") && !Dynamicweb.Core.Converter.ToBoolean(Context.Current.Items["IsWebServiceConnectionAvailable"])) 629 { 630 string erpDownMessageTheme = !string.IsNullOrWhiteSpace(Model.Area.Item.GetRawValueString("ErpDownMessageTheme")) ? " theme " + Model.Area.Item.GetRawValueString("ErpDownMessageTheme").Replace(" ", "").Trim().ToLower() : "theme light"; 631 632 <div class="position-fixed bottom-0 end-0 p-3" style="z-index: 1040"> 633 <div class="toast fade show border-0 @erpDownMessageTheme" role="alert" aria-live="assertive" aria-atomic="true"> 634 <div class="toast-header"> 635 <strong class="me-auto">@Translate("Connection down")</strong> 636 <button type="button" class="btn-close" data-bs-dismiss="toast" aria-label="Close"></button> 637 </div> 638 <div class="toast-body"> 639 @Translate("We are experiencing some connectivity issues. Not all features may be available to you.") 640 </div> 641 </div> 642 </div> 643 } 644 </body> 645 </html> 646 } 647 else if (Pageview.IsVisualEditorMode) 648 { 649 <head> 650 <title>@Model.Title</title> 651 @* Bootstrap + Swift stylesheet *@ 652 <link href="/Files/Templates/Designs/Swift/Assets/css/styles.css" rel="stylesheet" media="all" type="text/css"> 653 </head> 654 <body class="p-3"> 655 <div class="alert alert-danger" role="alert"> 656 @Translate("Basic Swift setup is needed!") 657 </div> 658 659 @if (brandingPage == null) 660 { 661 <div class="alert alert-warning" role="alert"> 662 @Translate("Please add a Branding page and reference it in website settings") 663 </div> 664 } 665 666 @if (themesParagraphs == null) 667 { 668 <div class="alert alert-warning" role="alert"> 669 @Translate("Please add a Themes collection page and reference it in website settings") 670 </div> 671 } 672 </body> 673 674 } 675 676 677 @functions { 678 void SetMetaTags() 679 { 680 //Verification Tokens 681 string siteVerificationGoogle = Model.Area.Item.GetString("Google_Site_Verification") != null ? Model.Area.Item.GetString("Google_Site_Verification") : ""; 682 683 //Generic Site Values 684 string openGraphFacebookAppID = Model.Area.Item.GetString("Fb_app_id") != null ? Model.Area.Item.GetString("Fb_app_id") : ""; 685 string openGraphType = Model.Area.Item.GetString("Open_Graph_Type") != null ? Model.Area.Item.GetString("Open_Graph_Type") : ""; 686 string openGraphSiteName = Model.Area.Item.GetString("Open_Graph_Site_Name") != null ? Model.Area.Item.GetString("Open_Graph_Site_Name") : ""; 687 688 string twitterCardSite = Model.Area.Item.GetString("Twitter_Site") != null ? Model.Area.Item.GetString("Twitter_Site") : ""; 689 690 //Page specific values 691 string openGraphSiteTitle = Model.Area.Item.GetString("Open_Graph_Title") != null ? Model.Area.Item.GetString("Open_Graph_Title") : ""; 692 FileViewModel openGraphImage = Model.Area.Item.GetFile("Open_Graph_Image"); 693 string openGraphImageALT = Model.Area.Item.GetString("Open_Graph_Image_ALT") != null ? Model.Area.Item.GetString("Open_Graph_Image_ALT") : ""; 694 string openGraphDescription = Model.Area.Item.GetString("Open_Graph_Description") != null ? Model.Area.Item.GetString("Open_Graph_Description") : ""; 695 696 string twitterCardURL = Model.Area.Item.GetString("Twitter_URL") != null ? Model.Area.Item.GetString("Twitter_URL") : ""; 697 string twitterCardTitle = Model.Area.Item.GetString("Twitter_Title") != null ? Model.Area.Item.GetString("Twitter_Title") : ""; 698 string twitterCardDescription = Model.Area.Item.GetString("Twitter_Description") != null ? Model.Area.Item.GetString("Twitter_Description") : ""; 699 FileViewModel twitterCardImage = Model.Area.Item.GetFile("Twitter_Image"); 700 string twitterCardImageALT = Model.Area.Item.GetString("Twitter_Image_ALT") != null ? Model.Area.Item.GetString("Twitter_Image_ALT") : ""; 701 702 if (string.IsNullOrEmpty(Dynamicweb.Context.Current.Request.QueryString["ProductID"])) 703 { 704 if (!string.IsNullOrEmpty(Model.Description)) 705 { 706 Pageview.Meta.AddTag($"<meta property=\"og:description\" content=\"{Model.Description}\">"); 707 } 708 else 709 { 710 Pageview.Meta.AddTag($"<meta property=\"og:description\" content=\"{openGraphDescription}\">"); 711 } 712 713 if (!string.IsNullOrEmpty(Pageview.Page.TopImage)) 714 { 715 Pageview.Meta.AddTag($"<meta property=\"og:image\" content=\"{Dynamicweb.Context.Current.Request.Url.Scheme}://{Dynamicweb.Context.Current.Request.Url.Host}/Files{Pageview.Page.TopImage}\">"); 716 Pageview.Meta.AddTag($"<meta property=\"og:image:secure_url\" content=\"{Dynamicweb.Context.Current.Request.Url.Scheme}://{Dynamicweb.Context.Current.Request.Url.Host}/Files{Pageview.Page.TopImage}\">"); 717 } 718 else if (openGraphImage != null) 719 { 720 Pageview.Meta.AddTag($"<meta property=\"og:image\" content=\"{Dynamicweb.Context.Current.Request.Url.Scheme}://{Dynamicweb.Context.Current.Request.Url.Host}{openGraphImage.Path}\">"); 721 Pageview.Meta.AddTag($"<meta property=\"og:image:secure_url\" content=\"{Dynamicweb.Context.Current.Request.Url.Scheme}://{Dynamicweb.Context.Current.Request.Url.Host}{openGraphImage.Path}\">"); 722 } 723 724 if (!string.IsNullOrEmpty(openGraphImageALT)) 725 { 726 Pageview.Meta.AddTag($"<meta property=\"og:image:alt\" content=\"{openGraphImageALT}\">"); 727 } 728 if (!string.IsNullOrEmpty(twitterCardDescription)) 729 { 730 Pageview.Meta.AddTag("twitter:description", twitterCardDescription); 731 } 732 733 if (!string.IsNullOrEmpty(Pageview.Page.TopImage)) 734 { 735 Pageview.Meta.AddTag("twitter:image", $"{Dynamicweb.Context.Current.Request.Url.Scheme}://{Dynamicweb.Context.Current.Request.Url.Host}/Files{Pageview.Page.TopImage}"); 736 } 737 else if (twitterCardImage != null) 738 { 739 Pageview.Meta.AddTag("twitter:image", $"{Dynamicweb.Context.Current.Request.Url.Scheme}://{Dynamicweb.Context.Current.Request.Url.Host}{openGraphImage.Path}"); 740 } 741 742 if (!string.IsNullOrEmpty(twitterCardImageALT)) 743 { 744 Pageview.Meta.AddTag("twitter:image:alt", twitterCardImageALT); 745 } 746 } 747 748 if (!string.IsNullOrEmpty(siteVerificationGoogle)) 749 { 750 Pageview.Meta.AddTag("google-site-verification", siteVerificationGoogle); 751 } 752 753 if (!string.IsNullOrEmpty(openGraphFacebookAppID)) 754 { 755 Pageview.Meta.AddTag($"<meta property=\"fb:app_id\" content=\"{openGraphFacebookAppID}\">"); 756 } 757 758 if (!string.IsNullOrEmpty(openGraphType)) 759 { 760 Pageview.Meta.AddTag($"<meta property=\"og:type\" content=\"{openGraphType}\">"); 761 } 762 763 if (!string.IsNullOrEmpty(openGraphSiteName)) 764 { 765 Pageview.Meta.AddTag($"<meta property=\"og:url\" content=\"{Pageview.SearchFriendlyUrl}\">"); 766 } 767 768 if (!string.IsNullOrEmpty(openGraphSiteName)) 769 { 770 Pageview.Meta.AddTag($"<meta property=\"og:site_name\" content=\"{openGraphSiteName}\">"); 771 } 772 773 if (!string.IsNullOrEmpty(Model.Title)) 774 { 775 Pageview.Meta.AddTag($"<meta property=\"og:title\" content=\"{Model.Title}\">"); 776 } 777 else 778 { 779 Pageview.Meta.AddTag($"<meta property=\"og:title\" content=\"{openGraphSiteTitle}\">"); 780 } 781 782 if (!string.IsNullOrEmpty(twitterCardSite)) 783 { 784 Pageview.Meta.AddTag("twitter:site", twitterCardSite); 785 } 786 787 if (!string.IsNullOrEmpty(twitterCardURL)) 788 { 789 Pageview.Meta.AddTag("twitter:url", twitterCardURL); 790 } 791 792 if (!string.IsNullOrEmpty(twitterCardTitle)) 793 { 794 Pageview.Meta.AddTag("twitter:title", twitterCardTitle); 795 } 796 } 797 } 798 799 @* TASK #27824 - Announcement bar 800 Rendered outside & above <header> to make the announcement dissappear on scroll 801 *@ 802 @helper RenderAnnouncementBarCustom() 803 { 804 var customItem = Model.Area.Item?.GetItem("Custom") ?? null; 805 string heightCss = "0;"; 806 807 if (customItem != null) 808 { 809 bool enableAnnouncementBar = customItem.GetBoolean("Custom_EnableAnnouncementBar"); 810 bool isMobile = Pageview.Device == Dynamicweb.Frontend.Devices.DeviceType.Mobile || Pageview.Device == Dynamicweb.Frontend.Devices.DeviceType.Tablet; 811 bool enableAnnouncementBarMobile = customItem.GetBoolean("Custom_EnableAnnouncementBarForMobile"); 812 813 if ((enableAnnouncementBar && !isMobile) || (enableAnnouncementBarMobile && isMobile)) 814 { 815 bool enableAnnouncementBarOnlyOnFrontpage = customItem.GetBoolean("Custom_EnableAnnouncementBarOnlyOnFrontpage"); 816 int frontPageId = Dynamicweb.Content.Services.Pages.GetFirstPageForArea(Pageview.AreaID).ID; 817 int currentPageId = Pageview.Page.ID; 818 bool isFrontPage = frontPageId == currentPageId; 819 820 if ((enableAnnouncementBarOnlyOnFrontpage && isFrontPage) || !enableAnnouncementBarOnlyOnFrontpage) 821 { 822 string horizontalAlign = !string.IsNullOrEmpty(customItem.GetRawValueString("Custom_HorizontalAlignment")) ? "justify-content-" + customItem.GetRawValueString("Custom_HorizontalAlignment") : ""; 823 string contentPadding = customItem.GetRawValueString("Custom_ContentPadding", "px-3 py-2"); 824 contentPadding = contentPadding == "none" ? "p-0" : contentPadding; 825 contentPadding = contentPadding == "small" ? "px-3 py-0" : contentPadding; 826 contentPadding = contentPadding == "large" ? "px-4 py-0" : contentPadding; 827 828 string theme = !string.IsNullOrWhiteSpace(customItem.GetRawValueString("Custom_Theme")) ? " theme " + customItem.GetRawValueString("Custom_Theme").Replace(" ", "").Trim().ToLower() : ""; 829 int initialAnnouncementInterval = customItem.GetInt32("Custom_InitialAnnouncementInterval"); 830 int intervalBetweenEachAnnouncement = customItem.GetInt32("Custom_IntervalBetweenEachAnnouncement"); 831 832 var announcementTexts = customItem.GetItems("Custom_AnnouncementTexts"); 833 834 int mobileHeight = customItem.GetInt32("Custom_HeightMobile") > 0 ? customItem.GetInt32("Custom_HeightMobile") : 30; 835 int desktopHeight = customItem.GetInt32("Custom_Height") > 0 ? customItem.GetInt32("Custom_Height") : 30; 836 int heightToUse = isMobile ? mobileHeight : desktopHeight; 837 heightCss = heightToUse + "px;"; 838 839 <div class="m-0 d-flex @(theme) item_@Model.Item.SystemName.ToLower() @contentPadding" style="height: @heightCss"> 840 <div class="announcement-bar js-announcement-bar" data-initial-interval-speed="@initialAnnouncementInterval" data-interval-speed="@intervalBetweenEachAnnouncement" data-height="@heightToUse"> 841 842 <div class="announcement-bar-slider-container text-animation-slider__container"> 843 <div class="announcement-bar-slider text-animation-slider__slider @(horizontalAlign)"> 844 @for (int i = 0; i < announcementTexts.Count; i++) 845 { 846 ItemViewModel announcementText = announcementTexts[i]; 847 string fieldValue = announcementText.Fields.FirstOrDefault()?.Value.ToString(); 848 849 if (string.IsNullOrEmpty(fieldValue)) 850 { 851 continue; 852 } 853 854 <div class="announcement-bar-slider-item text-animation-slider__item @(i == 0 ? "animate-in" : "")"> 855 <span>@fieldValue</span> 856 </div> 857 } 858 </div> 859 </div> 860 861 </div> 862 </div> 863 } 864 } 865 } 866 867 // Value is set in JS, but avoid CLS if JS executes slowly 868 <style> 869 :root { 870 --topValueForHeader: @heightCss 871 } 872 </style> 873 } 874