Error executing template "Designs/OttoSchachner/eCom/ProductCatalog/ProductView.cshtml" System.ArgumentOutOfRangeException: Index was out of range. Must be non-negative and less than the size of the collection. (Parameter 'index') at System.Collections.Generic.List`1.get_Item(Int32 index) at CompiledRazorTemplates.Dynamic.RazorEngine_15d500046b19464fa8b3b4e308e5b3ce.ExecuteAsync() at RazorEngine.Templating.TemplateBase.Run(ExecuteContext context, TextWriter reader) at RazorEngine.Templating.RazorEngineCore.RunTemplate(ICompiledTemplate template, TextWriter writer, Object model, DynamicViewBag viewBag) at RazorEngine.Templating.RazorEngineService.Run(ITemplateKey key, TextWriter writer, Type modelType, Object model, DynamicViewBag viewBag) at RazorEngine.Templating.DynamicWrapperService.Run(ITemplateKey key, TextWriter writer, Type modelType, Object model, DynamicViewBag viewBag) at RazorEngine.Templating.RazorEngineServiceExtensions.Run(IRazorEngineService service, String name, TextWriter writer, Type modelType, Object model, DynamicViewBag viewBag) at RazorEngine.Templating.RazorEngineServiceExtensions.<>c__DisplayClass23_0.<Run>b__0(TextWriter writer) at RazorEngine.Templating.RazorEngineServiceExtensions.WithWriter(Action`1 withWriter) at RazorEngine.Templating.RazorEngineServiceExtensions.Run(IRazorEngineService service, String name, Type modelType, Object model, DynamicViewBag viewBag) at Dynamicweb.Rendering.RazorTemplateRenderingProvider.Render(Template template) at Dynamicweb.Rendering.TemplateRenderingService.Render(Template template) at Dynamicweb.Rendering.Template.RenderRazorTemplate()
1 @inherits Dynamicweb.Rendering.ViewModelTemplate<Dynamicweb.Ecommerce.ProductCatalog.ProductViewModel> 2 @using System.Diagnostics.SymbolStore 3 @using Dynamicweb.Content 4 @using Dynamicweb.Content.Data 5 @using Dynamicweb.Ecommerce 6 @using Dynamicweb.Ecommerce.CustomerExperienceCenter.Favorites 7 @using Dynamicweb.Ecommerce.Prices 8 @using Dynamicweb.Frontend 9 @using Dynamicweb.Frontend.Devices 10 @using Dynamicweb.Frontend.Navigation 11 @using Dynamicweb.Ecommerce.ProductCatalog 12 @using Dynamicweb.Ecommerce.Products 13 @using Dynamicweb.Ecommerce.Shops 14 @using Dynamicweb.Ecommerce.Variants 15 @using Dynamicweb.Rendering 16 @using OttoSchachner.CustomCode.Helpers 17 18 @{ 19 var isCsv = string.Equals(Dynamicweb.Context.Current.Request.QueryString["download"], "csv", StringComparison.OrdinalIgnoreCase); 20 if (isCsv) 21 { 22 var fileVirtual = OttoSchachner.CustomCode.Helpers.ProductCsvExport.CreateTempCsvFile( 23 Model.Id, 24 Model.VariantId, 25 Model.LanguageId, 26 Model, 27 Pageview.User 28 ); 29 30 Dynamicweb.Context.Current.Response.Redirect(fileVirtual); 31 return; 32 } 33 34 var isZip = string.Equals(Dynamicweb.Context.Current.Request.QueryString["download"], "imageszip", StringComparison.OrdinalIgnoreCase); 35 if (isZip) 36 { 37 var zipVirtual = OttoSchachner.CustomCode.Helpers.ProductImagesZipExport.CreateTempImagesZipFile( 38 Model.Id, 39 Model.VariantId, 40 Model.LanguageId, 41 Model 42 ); 43 44 Dynamicweb.Context.Current.Response.Redirect(zipVirtual); 45 return; 46 } 47 } 48 49 @{ 50 string RenderPriceMatrix() 51 { 52 Dynamicweb.Ecommerce.Products.ProductService productService = new Dynamicweb.Ecommerce.Products.ProductService(); 53 ProductQuantityPrices matrixProvider = new ProductQuantityPrices(); 54 var productPriceMatrix = matrixProvider.GetProductQuantityPrices(productService.GetProductById(Model.Id, Model.VariantId, Model.LanguageId)); 55 var uniquePrices = productPriceMatrix 56 .GroupBy(p => new { p.Quantity, Unit = Model.ProductFields["VjmSalesUnit"].Value, Amount = Dynamicweb.Core.Converter.ToDouble(p.Amount) }) 57 .Select(g => g.First()) 58 .ToList(); 59 <div class="pview__matrix mb-4"> 60 @if (uniquePrices.Count != 0) 61 { 62 <div class="d-flex flex-row justify-content-between mb-2"> 63 <div class="fw-bold">@Translate("price-matrix-amount", "Antal")</div> 64 <div class="fw-bold">@Translate("price-matrix-price", "Pris")</div> 65 </div> 66 67 68 bool notLast = true; 69 int index = 0; 70 71 72 foreach (var price in uniquePrices) 73 { 74 <div class="d-flex flex-row justify-content-between"> 75 <div>@price.Quantity @price.UnitId</div> 76 <div>@Dynamicweb.Core.Converter.ToDouble(price.Amount).ToString("F2")</div> 77 </div> 78 79 if (index < uniquePrices.Count - 1) 80 { 81 <hr></hr> 82 } 83 84 index++; 85 } 86 } 87 else 88 { 89 <div class="fw-bold"> 90 @Translate("price-matrix-no-prices", "Ingen mængdepriser fundet") 91 </div> 92 } 93 </div> 94 95 return ""; 96 } 97 98 string RenderSections() 99 { 100 <div class="mb-4"> 101 <!-- Specifikationer - skjult for nu --> 102 103 @* <div class="pview__section"> *@ 104 @* <a href="javascript:void(0)" data-id="specifikationer" class="pview__section__name-container d-flex align-items-center justify-content-between"> *@ 105 @* *@ 106 @* <div class="d-flex flex-row"> *@ 107 @* <i class="pview__section__name-container__icon fa-solid fa-gears"></i> *@ 108 @* *@ 109 @* <div class="pview__section__name-container__name"> *@ 110 @* @Translate("product-specifications", "Specifikationer") *@ 111 @* </div> *@ 112 @* </div> *@ 113 @* <i class="bi-chevron-down"></i> *@ 114 @* <i class="bi-chevron-up d-none"></i> *@ 115 @* </a> *@ 116 @* *@ 117 @* <div data-id="specifikationer" class="pview__section__content"> *@ 118 @* Indhold *@ 119 @* </div> *@ 120 @* *@ 121 @* </div> *@ 122 123 @{ 124 var download = Dynamicweb.Core.Converter.ToString(Model.ProductFields.Where(x => x.Key == "VjmDocumentLibraryHtml").FirstOrDefault().Value.Value); 125 bool hasDocDownloads = !string.IsNullOrEmpty(download); 126 127 var current = Dynamicweb.Context.Current.Request.RawUrl ?? ""; 128 var csvUrl = current + (current.Contains("?") ? "&" : "?") + "download=csv"; 129 var zipUrl = (Dynamicweb.Context.Current.Request.RawUrl ?? "/") + 130 ((Dynamicweb.Context.Current.Request.RawUrl ?? "/").Contains("?") ? "&" : "?") + 131 "download=imageszip"; 132 } 133 134 <div class="pview__section"> 135 <a href="javascript:void(0)" data-id="Downloads" class="pview__section__name-container d-flex align-items-center justify-content-between"> 136 <div class="d-flex flex-row"> 137 <i class="pview__section__name-container__icon fa-solid fa-file-pdf"></i> 138 <div class="pview__section__name-container__name"> 139 @(Translate("product-downloads", "Downloads")) 140 </div> 141 </div> 142 <i class="bi-chevron-down"></i> 143 <i class="bi-chevron-up d-none"></i> 144 </a> 145 146 <div data-id="Downloads" class="pview__section__content"> 147 <div class="d-flex flex-column"> 148 <a class="os-button os-button--red mb-2" href="@csvUrl"> 149 @Translate("download-csv", "Download CSV") 150 </a> 151 152 <a class="os-button os-button--red mb-2" href="@zipUrl"> 153 @Translate("download-images-zip", "Download billeder ZIP") 154 </a> 155 156 @if (hasDocDownloads) 157 { 158 @download 159 } 160 </div> 161 </div> 162 </div> 163 </div> 164 165 return ""; 166 } 167 168 bool signedIn = PageView.Current().User != null; 169 bool purchasable = false; 170 171 if (signedIn) 172 { 173 if (Model.VariantCombinations().Count > 1 && string.IsNullOrEmpty(Model.VariantId)) 174 { 175 purchasable = false; 176 } 177 else 178 { 179 purchasable = true; 180 } 181 } 182 183 List<BreadcrumbItem> items = new List<BreadcrumbItem>(); 184 bool isMobile = PageView.Current().Device == DeviceType.Mobile; 185 186 var groupService = new GroupService(); 187 var shop = new ShopService().GetShop("SHOP1"); 188 var groups = groupService.FindPath(shop, groupService.GetGroup(Model.PrimaryOrDefaultGroup.Id)); 189 var shopPageId = GetPageIdByNavigationTag("Shop"); 190 var product = new ProductService().GetProductById(Model.Id, Model.VariantId, Model.LanguageId); 191 var productImageService = new ProductImageService(); 192 var images = productImageService.GetImagesFromPatterns(product, shop).ToList(); 193 bool variantSelected = !string.IsNullOrEmpty(Model.VariantId); 194 var pageService = new PageService(); 195 196 if (variantSelected) 197 { 198 images.Clear(); 199 images.Add(Model.DefaultImage.Value); 200 } 201 202 foreach (var group in groups) 203 { 204 items.Add(new BreadcrumbItem 205 { 206 Title = group.Name, 207 Url = "/Default.aspx?ID=" + shopPageId + "&groupid=" + group.Id 208 }); 209 } 210 211 var parameters = new Dictionary<string, object>(); 212 parameters.Add("items", items); 213 214 var _navigationSettings = new Dynamicweb.Frontend.Navigation.NavigationSettings() 215 { 216 Parameters = parameters 217 }; 218 219 var _navigationTemplate = "../Partials/Breadcrumb.cshtml"; 220 221 var alternativeImageObject = Model.ProductFields.Where(x => x.Key == "VjmpAlternativeManufacturerBrandLogo").FirstOrDefault(); 222 var alternativeImage = Dynamicweb.Core.Converter.ToString(alternativeImageObject.Value.Value); 223 } 224 <div class="pview os-container "> 225 226 <div class="pview__top position-relative d-flex flex-row justify-content-between align-items-center"> 227 228 <div class="pview__top__breadcrumb"> 229 @Navigation.RenderNavigation(_navigationTemplate, _navigationSettings) 230 </div> 231 232 <img class="pview__top__brand" src="@alternativeImage"/> 233 </div> 234 235 <div class="d-flex flew-row flex-wrap"> 236 237 238 <div class="pview__row row z-0"> 239 240 <div class="col-12 col-lg-6"> 241 <div class="product-view-gallery-wrapper position-relative"> 242 <div class="swiper product-view-gallery"> 243 <div class="swiper-wrapper"> 244 245 @foreach (var image in images) 246 { 247 var imageSrc = ""; 248 if (!string.IsNullOrEmpty(image)) 249 { 250 Dynamicweb.VestjyskMarketing.Models.ResizeImageSettings imageSettings = new Dynamicweb.VestjyskMarketing.Models.ResizeImageSettings 251 { 252 Width = 1500, 253 Height = 1500, 254 Crop = "5", 255 Quality = 90, 256 Image = image 257 }; 258 259 imageSrc = Dynamicweb.VestjyskMarketing.Helpers.ImageHelper.ResizeImage(imageSettings); 260 } 261 else 262 { 263 imageSrc = "/Files/Images/missing_image.jpg"; 264 } 265 266 <div class="swiper-slide"> 267 <img alt="Produkt billede 0" class="pview__image-container__image" src="@imageSrc"> 268 </div> 269 } 270 271 @if (images.Count() == 0) 272 { 273 <div class="swiper-slide"> 274 <img alt="Produkt billede 0" class="pview__image-container__image" src="/Files/Images/missing_image.jpg"> 275 </div> 276 } 277 278 </div> 279 </div> 280 281 @{ 282 if (signedIn && purchasable) 283 { 284 string favoriteLink = "/Default.aspx?ID=" + pageService.GetPageByNavigationTag(PageView.Current().AreaID, "FavoriteService").ID + "&ProductID=" + Model.Id + "&ProductVariantId=" + Model.VariantId + "&UserID=" + PageView.Current().User.ID + "&ReloadPage=false"; 285 bool isInFavoriteList = false; 286 var favoriteLists = Pageview.User.GetFavoriteLists(); 287 int favoriteListContainingProductId = 0; 288 string command = "add"; 289 new FavoriteListService().ClearCache(); 290 291 foreach (var favoriteList in favoriteLists) 292 { 293 isInFavoriteList = Pageview.User.IsProductInFavoriteList(favoriteList.ListId, product.Id, Model.VariantId); 294 295 if (isInFavoriteList) 296 { 297 favoriteListContainingProductId = favoriteList.ListId; 298 command = "remove"; 299 break; 300 } 301 } 302 303 <a data-command="@command" data-in-this-list="@favoriteListContainingProductId" data-url="@favoriteLink" href="javascript:void(0)" title="@Translate("add-or-remove-favorites")" class="product-list__favorite z-2"> 304 305 @if (isInFavoriteList) 306 { 307 <i class="fa-sharp fa-solid fa-heart"></i> 308 } 309 else 310 { 311 <i class="fa-regular fa-heart"></i> 312 } 313 314 </a> 315 } 316 } 317 <div class="product-view-gallery__navigation position-absolute top-50 translate-middle-y z-1 justify-content-between d-flex "> 318 <i class="os-chevron os-chevron--prev bi-chevron-left product-view-prev"></i > 319 <i class="os-chevron bi-chevron-right product-view-next"></i > 320 </div> 321 </div> 322 323 <div class="pview__thumbnails product-view-thumbs row g-3 mt-2 "> 324 @if (images.Count > 1) 325 { 326 var imagesArray = images.ToArray(); 327 328 for (int i = 0; 329 i < images.Count(); 330 i++) 331 { 332 Dynamicweb.VestjyskMarketing.Models.ResizeImageSettings imageSettings = new Dynamicweb.VestjyskMarketing.Models.ResizeImageSettings 333 { 334 Width = 750, 335 Height = 750, 336 Crop = "5", 337 Quality = 90, 338 Image = imagesArray[i] 339 }; 340 341 var imageSrc = Dynamicweb.VestjyskMarketing.Helpers.ImageHelper.ResizeImage(imageSettings); 342 343 int number = i + 1; 344 345 <a title="@Translate("go-to-image", "Gå til billede ") @number" data-slide="@i" class="pview__thumbnails__item col-2 "> 346 <img alt="@(Translate("product-image-alt", "Produkt billede ")) @i" loading="lazy" src="@imageSrc"/> 347 </a> 348 } 349 } 350 351 </div> 352 353 @if (signedIn && isMobile == false) 354 { 355 @RenderSections() 356 } 357 358 </div> 359 <div class="col-12 col-lg-6"> 360 <h1 class="pview__name"> 361 @Model.Name 362 </h1 > 363 <div class="pview__productid "> 364 @Translate("product-number", "Varenr: ") @Model.Number 365 </div> 366 367 368 @if (purchasable) 369 { 370 <div class="pview__price d-flex flex-row align-items-center"> 371 <div class="pview__price__text me-3"> 372 <span class="pview__price__text"> 373 @Translate("recommended-price", "Vejl. pris") 374 </span> 375 </div> 376 <span class="pview__price__amount">@product.DefaultPrice.ToString("F2") KR. <span class="pview__price__amount__netto d-none">(@Translate("net-price", "netto") @Model.Price.PriceWithoutVatFormatted )</span></span> 377 378 </div> 379 380 <div class="pview__price-toggler d-flex flex-row mb-3"> 381 382 383 <div class="pview__price-toggler__container d-flex flex-row align-items-center"> 384 <input class="me-3" type="checkbox" id="pview__netprice" name="horns"/> 385 <label for="pview__netprice"> 386 @Translate("show-net-price", "Vis nettopris") 387 </label> 388 </div> 389 390 391 </div> 392 } 393 else if (signedIn == false) 394 { 395 <div class="pview__signin-message "> 396 @Translate("sign-in-to-see-prices", "LOG IND FOR AT SE PRISER OG KØBE PRODUKTET") 397 </div> 398 } 399 400 @if (purchasable) 401 { 402 @RenderPriceMatrix() 403 } 404 405 <div class="pview__description"> 406 @Model.ShortDescription 407 @Model.LongDescription 408 </div> 409 410 @{ 411 string pictogramHtml = Model.ProductFields["VjmPictogramHtml"]?.Value.ToString(); 412 if (!string.IsNullOrEmpty(pictogramHtml)) 413 { 414 <div class="pictogram__section d-flex"> 415 <div class="d-flex "> 416 @pictogramHtml 417 </div> 418 </div> 419 } 420 } 421 422 @if (Model.VariantCombinations().Count > 1 && string.IsNullOrEmpty(Model.VariantId) && signedIn) 423 { 424 <div class="pview__variant-selector__message mb-3 fw-bold"> 425 @Translate("select-variant-message", "Du skal vælge variant før du kan tilføje til kurv") 426 </div> 427 } 428 @{ 429 string variantGroupName = ""; 430 bool colorGroup = Model.VariantInfo.VariantInfoGroupName == "Farver"; 431 432 @if (Model.VariantGroups().Count > 0) 433 { 434 if (Model.VariantGroups().Count == 1) 435 { 436 variantGroupName = Model.VariantGroups().First().Name; 437 variantGroupName = Translate("VariantGroup.Name." + variantGroupName, variantGroupName); 438 439 <div class="pview__cart__variant mb-3"> 440 <span class="fw-bold">@variantGroupName@(colorGroup ? ": " : "")</span> 441 @if (colorGroup) 442 { 443 <span>@Model.VariantName</span> 444 } 445 </div> 446 } 447 } 448 } 449 450 @if (signedIn && isMobile) 451 { 452 @RenderSections() 453 } 454 455 @{ 456 var step = Model.PurchaseQuantityStep; 457 if (step == null || step == 0) 458 { 459 step = 1; 460 } 461 462 string stockLevel = Convert.ToString(product != null ? product.ProductFieldValues["VjmStockLevel"]?.Value : Model.ProductFields["VjmStockLevel"]?.Value); 463 string stockLevelText = ""; 464 465 if (!string.IsNullOrEmpty(stockLevel)) 466 { 467 stockLevel = stockLevel.ToLower(); 468 stockLevelText = StockLevelText(stockLevel); 469 } 470 471 472 <div class="pview__cart"> 473 <div class="pview__variant-selector row g-2 mb-2 mb-lg-4 d-flex flex-row flex-wrap"> 474 475 @if (Model.VariantGroups().Count == 1) 476 { 477 @foreach (var variantGroup in Model.VariantGroups()) 478 { 479 foreach (var option in variantGroup.Options) 480 { 481 string variantUrl = "/Default.aspx?ID=" + Pageview.ID + "&groupid=" + Model.PrimaryOrDefaultGroup.Id + "&productid=" + Model.Id + "&variantid=" + option.Id; 482 var variantProduct = new ProductService().GetProductById(Model.Id, option.Id, Model.LanguageId); 483 484 @if (Model.VariantInfo.VariantInfoGroupName == "Farver") 485 { 486 var variantImagePath = productImageService.GetImagePath(variantProduct); 487 488 Dynamicweb.VestjyskMarketing.Models.ResizeImageSettings imageSettings = new Dynamicweb.VestjyskMarketing.Models.ResizeImageSettings 489 { 490 Width = 1500, 491 Height = 1500, 492 Crop = "5", 493 Quality = 90, 494 Image = variantImagePath 495 }; 496 497 string imageSrc = Dynamicweb.VestjyskMarketing.Helpers.ImageHelper.ResizeImage(imageSettings); 498 499 <div class="w-auto"> 500 <div class="pview__variant-selector__item"> 501 <a href="@variantUrl" title="@Translate("go-to", "Gå til") @variantGroupName.ToLower() @option.Name.ToLower()"> 502 <img class="w-100" alt="Variant @option.Name" src="@imageSrc"/> 503 </a> 504 </div> 505 </div> 506 } 507 else 508 { 509 <div class="w-auto"> 510 <div class="pview__variant-selector__item number-selector @(Model.VariantId == option.Id ? "active" : "")"> 511 <a href="@variantUrl" title="@Translate("go-to", "Gå til") @variantGroupName.ToLower() @option.Name.ToLower()" class="d-flex h-100 w-100 align-items-center justify-content-center text-decoration-none"> 512 @option.Name 513 </a> 514 </div> 515 </div> 516 } 517 } 518 } 519 } 520 //multi variant håndtering 521 else 522 { 523 <div hidden="" id="variant-combinations"> 524 @string.Join(",", Model.VariantCombinations()) 525 </div> 526 @foreach (var variantGroup in Model.VariantGroups()) 527 { 528 string variantGroupNameMulti = Translate("VariantGroup.Name." + variantGroup.Name, variantGroup.Name); 529 <div class="pview__cart__variant fw-bold"> 530 @variantGroupNameMulti 531 </div> 532 <div class="d-flex mb-1 flex-wrap"> 533 @{ 534 foreach (var option in variantGroup.Options) 535 { 536 string variantUrl = "/Default.aspx?ID=" + Pageview.ID + "&groupid=" + Model.PrimaryOrDefaultGroup.Id + "&productid=" + Model.Id + "&variantid=" + option.Id; 537 bool active = Model.VariantId.Contains(option.Id); 538 539 <div class="me-2 mb-2"> 540 <a data-group="@variantGroup.Name" data-id="@option.Id" data-url="@variantUrl" href="javascript:void(0)" title="@Translate("select", "vælg") @option.Name" class="os-button os-button--no-hover @(active ? "" : "active os-button--transparent") d-flex h-100 w-100 align-items-center justify-content-center text-decoration-none"> 541 @option.Name 542 </a> 543 </div> 544 } 545 } 546 </div> 547 } 548 } 549 </div> 550 551 @if (purchasable) 552 { 553 <form method="post" class="h-100"> 554 <div class="pview__cart__container d-flex align-items-stretch mb-3"> 555 <div class="col-4"> 556 <input name="Quantity" type="number" min="@step" step="@step" value="@step" class="pview__cart__container__quantity col-4"/> 557 </div> 558 <div class="col-8"> 559 560 <input type="hidden" name="ProductId" value='@Model.Id'/> 561 <input type="hidden" name="VariantId" value="@Model.VariantId"/> 562 <button class="pview__cart__container__button" type="submit" name="CartCmd" value="add">Tilføj til kurv</button> 563 564 </div> 565 </div> 566 </form> 567 <div class="d-flex flex-row align-items-center"> 568 <div class="col-4 d-flex flex-row justify-content-between align-items-center"> 569 <div class="pview__cart__min ">Min. antal: @step</div> 570 <div class="pview__cart__status-color pview__cart__status-color--@stockLevel"> 571 </div> 572 </div> 573 <div class="pview__cart__message d-flex align-items-center"> 574 @stockLevelText 575 </div> 576 </div> 577 } 578 </div>} 579 580 @if (signedIn == false) 581 { 582 @RenderSections() 583 } 584 </div> 585 </div> 586 </div> 587 </div> 588 589 @functions{ 590 591 string StockLevelText(string stockLevel) 592 { 593 string result = ""; 594 595 if (!string.IsNullOrEmpty(stockLevel)) 596 { 597 stockLevel = stockLevel.ToLower(); 598 599 switch (stockLevel) 600 { 601 case "green": 602 result = Translate("in-stock", "På lager"); 603 break; 604 case "yellow": 605 result = Translate("low-stock", "Få på lager"); 606 break; 607 case "red": 608 result = Translate("out-of-stock", "Ikke på lager"); 609 break; 610 default: 611 result = Translate("out-of-stock", "Ikke på lager"); 612 break; 613 } 614 } 615 616 return result; 617 } 618 619 } 620 621 @if (Model.VariantCombinations().Count > 0) 622 { 623 var quickOrderVariants = product.GetVariantCombinations().Where(x => x.GetProduct(Model.LanguageId) != null && x.GetProduct(Model.LanguageId).Active).ToList(); 624 quickOrderVariants = quickOrderVariants.OrderBy(x => x.GetProduct(Model.LanguageId).Number).ToList(); 625 int count = 0; 626 <div class="pview__qorder"> 627 <div class="os-container"> 628 <div class="d-flex flex-column flex-lg-row justify-content-between flex-wrap mb-3 mb-lg-5"> 629 <div class="pview__qorder__header col-12 col-lg-auto mb-3 mb-lg-0"> 630 @Translate("quick-order", "Hurtig bestilling") 631 </div> 632 633 <div class="pview__qorder__input-container d-flex flex-row align-items-center col-12 col-lg-5"> 634 <i class="bi-search me-2"></i> 635 <input class="pview__qorder__input-container__input col-5" 636 placeholder='@Translate("quick-order-search", "Søg efter varenr., DB nr., materiale, størrelse m.m.")'/> 637 </div> 638 </div> 639 <form method="post" action=""> 640 641 642 <input type="hidden" name="cartcmd" value="addmulti"/> 643 <input type="hidden" name="redirect" value="false"/> 644 645 <div class="product-table"> 646 <div class="row header @(signedIn ? "signed-in" : "") @(Model.VariantGroups().Count > 1 ? "multi" : "")"> 647 <div>@Translate("order-variant-number", "Varenr.")</div> 648 649 @{ 650 List<string> variantGroupNames = new List<string>(); 651 } 652 653 @foreach (var variantGroup in Model.VariantGroups()) 654 { 655 string _variantGroupName = Translate("VariantGroup.Name." + variantGroup.Name, variantGroup.Name); 656 variantGroupNames.Add(_variantGroupName); 657 <div>@_variantGroupName</div> 658 } 659 660 661 <div>@Translate("db-number", "DB-nummer")</div> 662 <div>@Translate("ean-upc", "EAN/UPC")</div> 663 <div>@Translate("special-order-item", "Skaffevare")</div> 664 <div>@Translate("order-quantity", "Ordrekvantum")</div> 665 <div>@Translate("packaging", "Forpakning")</div> 666 @if (signedIn) 667 { 668 <div>@Translate("stock-status", "Lagerstatus")</div> 669 } 670 </div> 671 672 @{ 673 var matrixItems = new List<(Dynamicweb.Ecommerce.Products.Product Product, VariantCombination VariantComb, bool IsVariant)>(); 674 675 //hvis ingen varianter - hvis hovedproduktet i tabellen 676 if (quickOrderVariants.Count == 0) 677 { 678 matrixItems.Add((product, null, false)); 679 } 680 681 // Tilføj alle variants bagefter 682 foreach (var vc in quickOrderVariants) 683 { 684 matrixItems.Add((vc.GetProduct(Model.LanguageId), vc, true)); 685 } 686 } 687 688 <div class="row-wrapper"> 689 @foreach (var item in matrixItems) 690 { 691 count++; 692 693 694 var variant = item.Product; 695 var variantComb = item.VariantComb; 696 bool isVariant = item.IsVariant; 697 // DB-nummer (VjmDBNumber) 698 string dbNumber = 699 variant?.ProductFieldValues? 700 .GetProductFieldValue("VjmDBNumber")?.Value?.ToString() 701 ?? Translate("not-available", "N/A"); 702 703 // EAN 704 string EAN = !string.IsNullOrEmpty(variant?.EAN) 705 ? variant.EAN 706 : Translate("not-available", "N/A"); 707 708 string spStatus = variant?.ProductFieldValues.GetProductFieldValue("SpStatus")?.Value?.ToString().ToLower(); 709 710 if (spStatus == "skaffevare" || spStatus == "pris på forespørgsel") 711 { 712 spStatus = Translate("Yes", "Ja"); 713 } 714 else 715 { 716 spStatus = Translate("No", "Nej"); 717 } 718 719 string variantStockLevel = Convert.ToString(variant != null ? variant.ProductFieldValues["VjmStockLevel"]?.Value : "red"); 720 string variantStockLevelText = ""; 721 722 723 if (!string.IsNullOrEmpty(variantStockLevel)) 724 { 725 variantStockLevel = variantStockLevel.ToLower(); 726 variantStockLevelText = StockLevelText(variantStockLevel); 727 } 728 729 // Packaging (VjmUnitsConcat) 730 string packaging = 731 variant?.ProductFieldValues? 732 .GetProductFieldValue("VjmUnitsConcat")?.Value?.ToString() 733 ?? ""; 734 735 // Unit (VjmSalesUnit) 736 string unit = 737 variant?.ProductFieldValues? 738 .GetProductFieldValue("VjmSalesUnit")?.Value?.ToString() 739 ?? ""; 740 741 // Minimum order quantity 742 string minQty = variant?.PurchaseQuantityStep.ToString() ?? ""; 743 744 if (minQty == "0") 745 { 746 minQty = "1"; 747 unit = ""; 748 } 749 750 // Price 751 string bruttoPrice = $"{variant.DefaultPrice:0.00} KR"; 752 PriceContext priceContext = new PriceContext(Dynamicweb.Ecommerce.Common.Context.Currency, Dynamicweb.Ecommerce.Common.Context.Country, shop, Pageview.User, false, null); 753 string netPrice = $"{variant?.GetPrice(priceContext).PriceWithoutVATFormattedNoSymbol:00} KR"; 754 755 <div class="row data @(count % 2 == 0 ? "visible-index-even" : "") @(signedIn ? "signed-in" : "") @(Model.VariantGroups().Count > 1 ? "multi" : "")"> 756 <input type="hidden" name="ProductLoopCounter@(count)" id="ProductLoopCounter@(count)" value="@count"/> 757 <input type="hidden" name="ProductId@(count)" id="ProductId@(count)" value="@variant.Id"/> 758 @if (isVariant) 759 { 760 <input type="hidden" name="VariantId@(count)" value="@variantComb.VariantId"/> 761 } 762 763 <!-- Varenr. --> 764 <div class="d-flex justify-content-between d-lg-block"> 765 <span class="d-block d-lg-none">@Translate("order-variant-number", "Varenr.")</span> 766 <span class="searchable">@variant.Number</span> 767 </div> 768 <hr/> 769 <!-- Variantnavne --> 770 @if (isVariant) 771 { 772 int index = 0; 773 774 foreach (var optionId in variantComb.GetVariantOptionIds()) 775 { 776 <div class="d-flex justify-content-between d-lg-block"> 777 <span class="d-block d-lg-none">@variantGroupNames[index]</span> 778 <span class="searchable"> 779 @Model.VariantGroups().Find(group => group.Options.Find(option => option.Id == optionId) != null).Options.Single(option2 => option2.Id == optionId).Name 780 </span> 781 </div> 782 <hr/> 783 index++; 784 } 785 } 786 787 <!-- DB nummer --> 788 <div class="d-flex justify-content-between d-lg-block"> 789 <span class="d-block d-lg-none">@Translate("db-number", "DB-nummer")</span> 790 <span class="searchable">@dbNumber</span> 791 </div> 792 793 <hr/> 794 <!-- EAN --> 795 <div class="d-flex justify-content-between d-lg-block"> 796 <span class="d-block d-lg-none">@Translate("ean-upc", "EAN/UPC")</span> 797 <span class="searchable">@EAN</span> 798 </div> 799 <hr/> 800 801 <!-- Skaffevare --> 802 <div class="d-flex justify-content-between d-lg-block"> 803 <span class="d-block d-lg-none">@Translate("special-order-item", "Skaffevare")</span> 804 <span class="searchable">@spStatus</span> 805 </div> 806 807 <hr/> 808 <!-- Ordrekvantum --> 809 <div class="d-flex justify-content-between d-lg-block"> 810 <span class="d-block d-lg-none">@Translate("order-quantity", "Ordrekvantum")</span> 811 <span>@minQty @unit</span> 812 </div> 813 <hr/> 814 815 <!-- Forpakning --> 816 <div class="d-flex justify-content-between d-lg-block"> 817 <span class="d-block d-lg-none">@Translate("packaging", "Forpakning")</span> 818 <span>@packaging</span> 819 </div> 820 821 @if (signedIn) 822 { 823 <hr/> 824 <!-- Lagerstatus --> 825 <div class="d-flex justify-content-between d-lg-block align-items-center"> 826 <span class="d-block d-lg-none">@Translate("stock-status", "Lagerstatus")</span> 827 <div class="d-flex flex-row align-items-center p-0"> 828 <div class="pview__cart__status-color pview__cart__status-color--@variantStockLevel"></div> 829 @variantStockLevelText 830 </div> 831 </div> 832 833 <hr/> 834 835 <!-- Pris --> 836 <div class="d-flex justify-content-between d-lg-block flex-row flex-lg-column align-items-center"> 837 <span class="d-block d-lg-none">@Translate("price", "Pris")</span> 838 <div class="d-flex flex-column"> 839 <span class="fw-bold">@bruttoPrice</span> 840 <span class="quick-order-net d-none">(@netPrice)</span> 841 <small>@Translate("quick-order-min", "Bestilles i"): @minQty @unit</small> 842 </div> 843 844 </div> 845 <hr/> 846 847 <!-- Antal vælger --> 848 <div class="d-flex justify-content-between d-lg-block align-items-center"> 849 <span class="d-block d-lg-none">@Translate("quantity", "Antal")</span> 850 <div class="pview__qorder__actions d-flex flex-row"> 851 <a class="pview__qorder__actions__decrement" href="javascript:void(0)">-</a> 852 <input class="pview__qorder__actions__input" name="Quantity@(count)" type="number" step="@minQty" min="0" value="0"> 853 <a class="pview__qorder__actions__incremenet" href="javascript:void(0)">+</a> 854 </div> 855 </div> 856 } 857 </div> 858 } 859 860 </div> 861 </div> 862 863 864 <div class="pview__qorder__no-results justify-content-center mt-3" style="display: none"> 865 @Translate("quick-order-no-results", "Ingen resultater fundet") 866 </div> 867 868 @if (signedIn) 869 { 870 <div class="d-flex justify-content-end"> 871 <button type="submit" class="pview__qorder__add-button os-button os-button--red mt-4"> 872 @Translate("add-all-to-cart", "Læg alle i kurv") 873 </button> 874 </div> 875 } 876 </form > 877 878 </div> 879 </div> 880 } 881 882 @{ 883 var productListParameters = new Dictionary<string, object>(); 884 productListParameters.Add("itemsToShow", 4); 885 productListParameters.Add("maxItems", 99); 886 887 List<dynamic> productsRelated = new List<dynamic>(); 888 foreach (var relatedGroup in Model.RelatedGroups) 889 { 890 if (relatedGroup.Id == "TILBEHØR") 891 { 892 foreach (var relatedProduct in relatedGroup.Products) 893 { 894 productsRelated.Add(ProductInfoViewModelExtensions.GetProduct(relatedProduct)); 895 } 896 } 897 } 898 899 bool slider = productsRelated.Count > 4; 900 productListParameters.Add("slider", slider); 901 productListParameters.Add("products", productsRelated); 902 } 903 904 @if (productsRelated.Count > 0) 905 { 906 <div class="os-container pview__related"> 907 <div class="pview__related__header mb-4"> 908 @Translate("product-add-ons", "Tilbehør") 909 </div> 910 <div class="@(slider ? "" : "row g-3") "> 911 @RenderPartial("/Designs/OttoSchachner/Partials/ProductList.cshtml", new ParagraphViewModel(), productListParameters) 912 </div> 913 </div> 914 } 915 916 @if (signedIn == false) 917 { 918 <div class="os-container pview__signup"> 919 @{ 920 var signupParagraphPageId = GetPageIdByNavigationTag("NotSignedInProductViewParagraph"); 921 @(new Content(Pageview).RenderExternalGrid(signupParagraphPageId, "")) 922 } 923 </div> 924 }