Produktet blev tilføjet favoritliste Produktet blev fjernet fra din favoritliste
Otto schachner logo
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_000fae8c6286404ab0a90673f14a8644.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 <div class="pview__cart__container d-flex align-items-stretch mb-3"> 554 <div class="col-4"> 555 <input type="number" min="@step" step="@step" value="@step" class="pview__cart__container__quantity col-4"/> 556 </div> 557 <div class="col-8"> 558 <form method="post" class="h-100"> 559 <input type="hidden" name="ProductId" value='@Model.Id'/> 560 <input type="hidden" name="VariantId" value="@Model.VariantId"/> 561 <button class="pview__cart__container__button" type="submit" name="CartCmd" value="add">Tilføj til kurv</button> 562 </form> 563 </div> 564 </div> 565 <div class="d-flex flex-row align-items-center"> 566 <div class="col-4 d-flex flex-row justify-content-between align-items-center"> 567 <div class="pview__cart__min ">Min. antal: @step</div> 568 <div class="pview__cart__status-color pview__cart__status-color--@stockLevel"> 569 </div> 570 </div> 571 <div class="pview__cart__message d-flex align-items-center"> 572 @stockLevelText 573 </div> 574 </div> 575 } 576 </div>} 577 578 @if (signedIn == false) 579 { 580 @RenderSections() 581 } 582 </div> 583 </div> 584 </div> 585 </div> 586 587 @functions{ 588 589 string StockLevelText(string stockLevel) 590 { 591 string result = ""; 592 593 if (!string.IsNullOrEmpty(stockLevel)) 594 { 595 stockLevel = stockLevel.ToLower(); 596 597 switch (stockLevel) 598 { 599 case "green": 600 result = Translate("in-stock", "På lager"); 601 break; 602 case "yellow": 603 result = Translate("low-stock", "Få på lager"); 604 break; 605 case "red": 606 result = Translate("out-of-stock", "Ikke på lager"); 607 break; 608 default: 609 result = Translate("out-of-stock", "Ikke på lager"); 610 break; 611 } 612 } 613 614 return result; 615 } 616 617 } 618 619 @if (Model.VariantCombinations().Count > 0) 620 { 621 var quickOrderVariants = product.GetVariantCombinations().Where(x => x.GetProduct(Model.LanguageId) != null && x.GetProduct(Model.LanguageId).Active).ToList(); 622 quickOrderVariants = quickOrderVariants.OrderBy(x => x.GetProduct(Model.LanguageId).Number).ToList(); 623 int count = 0; 624 625 if (quickOrderVariants.Count == 0) 626 { 627 return; 628 } 629 630 631 <div class="pview__qorder"> 632 <div class="os-container"> 633 <div class="d-flex flex-column flex-lg-row justify-content-between flex-wrap mb-3 mb-lg-5"> 634 <div class="pview__qorder__header col-12 col-lg-auto mb-3 mb-lg-0"> 635 @Translate("quick-order", "Hurtig bestilling") 636 </div> 637 638 <div class="pview__qorder__input-container d-flex flex-row align-items-center col-12 col-lg-5"> 639 <i class="bi-search me-2"></i> 640 <input class="pview__qorder__input-container__input col-5" 641 placeholder='@Translate("quick-order-search", "Søg efter varenr., DB nr., materiale, størrelse m.m.")'/> 642 </div> 643 </div> 644 <form method="post" action=""> 645 646 647 <input type="hidden" name="cartcmd" value="addmulti"/> 648 <input type="hidden" name="redirect" value="false"/> 649 650 <div class="product-table"> 651 <div class="row header @(signedIn ? "signed-in" : "") @(Model.VariantGroups().Count > 1 ? "multi" : "")"> 652 <div>@Translate("order-variant-number", "Varenr.")</div> 653 654 @{ 655 List<string> variantGroupNames = new List<string>(); 656 } 657 658 @foreach (var variantGroup in Model.VariantGroups()) 659 { 660 string _variantGroupName = Translate("VariantGroup.Name." + variantGroup.Name, variantGroup.Name); 661 variantGroupNames.Add(_variantGroupName); 662 <div>@_variantGroupName</div> 663 } 664 665 666 <div>@Translate("db-number", "DB-nummer")</div> 667 <div>@Translate("ean-upc", "EAN/UPC")</div> 668 <div>@Translate("order-quantity", "Ordrekvantum")</div> 669 <div>@Translate("packaging", "Forpakning")</div> 670 @if (signedIn) 671 { 672 <div>@Translate("stock-status", "Lagerstatus")</div> 673 } 674 </div> 675 676 <div class="row-wrapper"> 677 @foreach (VariantCombination variantComb in quickOrderVariants) 678 { 679 count++; 680 Dynamicweb.Ecommerce.Products.Product variant = variantComb.GetProduct(Model.LanguageId); 681 682 // DB-nummer (VjmDBNumber) 683 string dbNumber = 684 variant?.ProductFieldValues? 685 .GetProductFieldValue("VjmDBNumber")?.Value?.ToString() 686 ?? Translate("not-available", "N/A"); 687 688 // EAN 689 string EAN = !string.IsNullOrEmpty(variant?.EAN) 690 ? variant.EAN 691 : Translate("not-available", "N/A"); 692 693 694 string variantStockLevel = Convert.ToString(variant != null ? variant.ProductFieldValues["VjmStockLevel"]?.Value : "red"); 695 string variantStockLevelText = ""; 696 697 698 if (!string.IsNullOrEmpty(variantStockLevel)) 699 { 700 variantStockLevel = variantStockLevel.ToLower(); 701 variantStockLevelText = StockLevelText(variantStockLevel); 702 } 703 704 // Packaging (VjmUnitsConcat) 705 string packaging = 706 variant?.ProductFieldValues? 707 .GetProductFieldValue("VjmUnitsConcat")?.Value?.ToString() 708 ?? ""; 709 710 // Unit (VjmSalesUnit) 711 string unit = 712 variant?.ProductFieldValues? 713 .GetProductFieldValue("VjmSalesUnit")?.Value?.ToString() 714 ?? ""; 715 716 // Minimum order quantity 717 string minQty = variant?.PurchaseQuantityStep.ToString() ?? ""; 718 719 if (minQty == "0") 720 { 721 minQty = "1"; 722 unit = ""; 723 } 724 725 // Price 726 string bruttoPrice = $"{variant.DefaultPrice:0.00} KR"; 727 PriceContext priceContext = new PriceContext(Dynamicweb.Ecommerce.Common.Context.Currency, Dynamicweb.Ecommerce.Common.Context.Country, shop, Pageview.User, false, null); 728 string netPrice = $"{variant?.GetPrice(priceContext).PriceWithoutVATFormattedNoSymbol:00} KR"; 729 <div class="row data @(count % 2 == 0 ? "visible-index-even" : "") @(signedIn ? "signed-in" : "") @(Model.VariantGroups().Count > 1 ? "multi" : "")"> 730 <input type="hidden" name="ProductLoopCounter@(count)" id="ProductLoopCounter@(count)" value="@count"/> 731 <input type="hidden" name="ProductId@(count)" id="ProductId@(count)" value="@variant.Id"/> 732 <input type="hidden" name="VariantId@(count)" id="VariantId@(count)" value="@variantComb.VariantId"/> 733 734 <!-- Varenr. --> 735 <div class="d-flex justify-content-between d-lg-block"> 736 <span class="d-block d-lg-none">@Translate("order-variant-number", "Varenr.")</span> 737 <span class="searchable">@variant.Number</span> 738 </div> 739 <hr/> 740 <!-- Variantnavne --> 741 @{ 742 int index = 0; 743 } 744 @foreach (var optionId in variantComb.GetVariantOptionIds()) 745 { 746 <div class="d-flex justify-content-between d-lg-block"> 747 <span class="d-block d-lg-none">@variantGroupNames[index]</span> 748 <span class="searchable">@Model.VariantGroups().Find(group => group.Options.Find(option => option.Id == optionId) != null).Options.Single(option2 => option2.Id == optionId).Name</span> 749 </div> 750 <hr/> 751 index++; 752 } 753 754 755 <!-- DB nummer --> 756 <div class="d-flex justify-content-between d-lg-block"> 757 <span class="d-block d-lg-none">@Translate("db-number", "DB-nummer")</span> 758 <span class="searchable">@dbNumber</span> 759 </div> 760 761 <hr/> 762 <!-- EAN --> 763 <div class="d-flex justify-content-between d-lg-block"> 764 <span class="d-block d-lg-none">@Translate("ean-upc", "EAN/UPC")</span> 765 <span class="searchable">@EAN</span> 766 </div> 767 768 <hr/> 769 <!-- Ordrekvantum --> 770 <div class="d-flex justify-content-between d-lg-block"> 771 <span class="d-block d-lg-none">@Translate("order-quantity", "Ordrekvantum")</span> 772 <span>@minQty @unit</span> 773 </div> 774 <hr/> 775 776 <!-- Forpakning --> 777 <div class="d-flex justify-content-between d-lg-block"> 778 <span class="d-block d-lg-none">@Translate("packaging", "Forpakning")</span> 779 <span>@packaging</span> 780 </div> 781 782 @if (signedIn) 783 { 784 <hr/> 785 <!-- Lagerstatus --> 786 <div class="d-flex justify-content-between d-lg-block align-items-center"> 787 <span class="d-block d-lg-none">@Translate("stock-status", "Lagerstatus")</span> 788 <div class="d-flex flex-row align-items-center p-0"> 789 <div class="pview__cart__status-color pview__cart__status-color--@variantStockLevel"></div> 790 @variantStockLevelText 791 </div> 792 </div> 793 794 <hr/> 795 796 <!-- Pris --> 797 <div class="d-flex justify-content-between d-lg-block flex-row flex-lg-column align-items-center"> 798 <span class="d-block d-lg-none">@Translate("price", "Pris")</span> 799 <div class="d-flex flex-column"> 800 <span class="fw-bold">@bruttoPrice</span> 801 <span class="quick-order-net d-none">(@netPrice)</span> 802 <small>@Translate("quick-order-min", "Bestilles i"): @minQty @unit</small> 803 </div> 804 805 </div> 806 <hr/> 807 808 <!-- Antal vælger --> 809 <div class="d-flex justify-content-between d-lg-block align-items-center"> 810 <span class="d-block d-lg-none">@Translate("quantity", "Antal")</span> 811 <div class="pview__qorder__actions d-flex flex-row"> 812 <a class="pview__qorder__actions__decrement" href="javascript:void(0)">-</a> 813 <input class="pview__qorder__actions__input" name="Quantity@(count)" type="number" step="@minQty" min="0" value="0"> 814 <a class="pview__qorder__actions__incremenet" href="javascript:void(0)">+</a> 815 </div> 816 </div> 817 } 818 </div> 819 } 820 821 </div> 822 </div> 823 824 825 <div class="pview__qorder__no-results justify-content-center mt-3" style="display: none"> 826 @Translate("quick-order-no-results", "Ingen resultater fundet") 827 </div> 828 829 @if (signedIn) 830 { 831 <div class="d-flex justify-content-end"> 832 <button type="submit" class="pview__qorder__add-button os-button os-button--red mt-4"> 833 @Translate("add-all-to-cart", "Læg alle i kurv") 834 </button> 835 </div> 836 } 837 </form > 838 839 </div> 840 </div> 841 } 842 843 @{ 844 var productListParameters = new Dictionary<string, object>(); 845 productListParameters.Add("itemsToShow", 4); 846 productListParameters.Add("maxItems", 99); 847 848 List<dynamic> productsRelated = new List<dynamic>(); 849 foreach (var relatedGroup in Model.RelatedGroups) 850 { 851 if (relatedGroup.Id == "TILBEHØR") 852 { 853 foreach (var relatedProduct in relatedGroup.Products) 854 { 855 productsRelated.Add(ProductInfoViewModelExtensions.GetProduct(relatedProduct)); 856 } 857 } 858 } 859 860 bool slider = productsRelated.Count > 4; 861 productListParameters.Add("slider", slider); 862 productListParameters.Add("products", productsRelated); 863 } 864 865 @if (productsRelated.Count > 0) 866 { 867 <div class="os-container pview__related"> 868 <div class="pview__related__header mb-4"> 869 @Translate("product-add-ons", "Tilbehør") 870 </div> 871 <div class="@(slider ? "" : "row g-3") "> 872 @RenderPartial("/Designs/OttoSchachner/Partials/ProductList.cshtml", new ParagraphViewModel(), productListParameters) 873 </div> 874 </div> 875 } 876 877 @if (signedIn == false) 878 { 879 <div class="os-container pview__signup"> 880 @{ 881 var signupParagraphPageId = GetPageIdByNavigationTag("NotSignedInProductViewParagraph"); 882 @(new Content(Pageview).RenderExternalGrid(signupParagraphPageId, "")) 883 } 884 </div> 885 }