{"version":3,"file":"product-0509b1de.min.js","sources":["../src/js/product/ProductViewModelFactories.js","../src/js/product/ProductEditor.js","../src/js/product/ProductInteraction.js","../src/js/product/DoorFrontInteraction.js","../src/js/product/Components.js","../src/js/product/ProductDrawingViewModel.js","../src/js/product/MultiStepProductEditor.js","../src/js/product/MultiStepDoorFrontEditor.js","../src/js/product/DoorFrontEditor.js","../src/js/product/InteriorInteraction.js","../src/js/product/InteriorEditor.js","../src/js/product/VariantProductEditor.js","../src/js/product/Components-Flexi.js","../src/js/product/ProductViewModelSpecificFactories.js","../src/js/product/DrawingEngine.js","../src/js/product/ImageLoader.js","../src/js/product/ProductMaterialLoader.js","../src/js/product/DrawingEngine2020.js","../src/js/product/DrawingService2020.js","../src/js/product/DrawingService.js","../src/js/product/Iterator.js","../src/js/product/ProductVisitors.js","../src/js/product/UpdateMaterialImageVisitor.js","../src/js/product/HtmlDrawing/HtmlRenderVisitor.js","../src/js/product/HtmlDrawing/HtmlDrawingService.js","../src/js/product/HtmlDrawing/HtmlAspectRatioSizer.js","../src/js/product/HtmlDrawing/HtmlDrawingEngine.js","../src/js/product/Babylon/FaceUVHelper.js","../src/js/product/Babylon/QuadriLateralFrustumBuilder.js","../src/js/product/Babylon/BabylonBasketBuilders.js","../src/js/product/Babylon/BabylonRenderVisitor.js","../src/js/product/Babylon/BabylonDrawingService.js","../src/js/product/Babylon/BabylonRoom.js","../src/js/product/Babylon/BabylonDrawingEngine.js","../src/js/product/ProductMaterialsMap.js","../src/js/product/ProductDrawing.js","../src/js/product/DesignerService.js","../src/js/product/ProductZoomer.js","../src/js/product/zoomer.js","../src/js/product/ProductPreviewer.js","../src/js/product/ProductPageViewModel.js","../src/js/product.js"],"sourcesContent":["export class ProductViewModelFactory {\r\n constructor(productType) {\r\n this.productType = productType;\r\n }\r\n\r\n createProductEditor(productEditorConfig, designerService, productService) {\r\n }\r\n\r\n /**\r\n * The materialIds object will be populated with properties matching the values in the finishes\r\n * array. Each property's value will be set to true.\r\n * @param {any} materialIds An object which will given additional properties by this method.\r\n * @param {any} prefix Prefix for the property names\r\n * @param {any} finishes An array of finish id's which will become property names on the materialIds parameter.\r\n */\r\n addMaterials(materialIds, prefix, finishes) {\r\n finishes.forEach(finish => materialIds[prefix + '-' + finish] = true);\r\n };\r\n\r\n getAllKnownProductMaterialIds(apiProduct) {\r\n return {};\r\n }\r\n\r\n createProduct(drawingService, apiProduct) {\r\n }\r\n}\r\n\r\nclass ProductViewModelFactoryList {\r\n constructor() {\r\n this.factories = {};\r\n this.knownFactories = [];\r\n }\r\n\r\n get(productType, factoryClass, productClass) {\r\n let factory = this.factories[productType];\r\n if (!factory) {\r\n factory = this.factories[productType] = new factoryClass(productType, productClass);\r\n }\r\n return factory;\r\n }\r\n\r\n getFactory(productType) {\r\n let map = this.knownFactories[productType];\r\n if (!map) {\r\n map = this.knownFactories['FixedDesignerProduct'];\r\n }\r\n return this.get(productType, map.factoryClass, map.productClass);\r\n }\r\n\r\n createProduct(drawingService, apiProduct, config) {\r\n const factory = this.getFactory(apiProduct.type);\r\n return factory.createProduct(drawingService, apiProduct, config);\r\n }\r\n\r\n /**\r\n * Registers a map from product type to product view model factory (and optional product class).\r\n * @param {any} productType Must match the types specified in API products (product.type) returned\r\n * by the Langlo Store API on the LangloWeb (link.langlo.no) server.\r\n * @param {ProductViewModelFactory} factoryClass A subclass of ProductViewModelFactory.\r\n * @param {Component} productClass A subclass of Component.\r\n */\r\n register(productType, factoryClass, productClass) {\r\n this.knownFactories[productType] = { factoryClass, productClass };\r\n }\r\n}\r\n\r\nexport const ProductViewModelFactories = new ProductViewModelFactoryList();\r\n","import { ObjectUtils, Utils, MinDuration, PromiseFlag } from '../main/ObjectUtils.js';\r\nimport { GlobalErrorHandler } from '../main/GlobalErrorHandler.js';\r\n\r\n/**\r\n * The ProductEditor is a ViewModel for the part of the product's user interface which is responsible\r\n * for accepting edits by the user.\r\n */\r\nexport class ProductEditor {\r\n /**\r\n * This constructor is async because--if session storage contains a product--it will communicate\r\n * with the remote designerService to restore the product. This happens on the last line of the\r\n * constructor.\r\n */\r\n constructor(config, designerService, productService) {\r\n this.config = config;\r\n /** An object containing child objects with configuration for each edit field */\r\n this.fieldsCfg = config.fields || {};\r\n this.designerService = designerService;\r\n this.productService = productService;\r\n this.componentType = null; // Must be assigned by subclass\r\n this.designerUpdateFlag = new PromiseFlag(); // Used by addToCart click event handler\r\n\r\n // Debug:\r\n //const abortController = new AbortController();\r\n //this.designerService.getProductDetails(99, abortController.signal);\r\n\r\n const byId = id => document.getElementById(id);\r\n\r\n const elementIds = config.UI.El;\r\n this.elements = {\r\n productData: byId(elementIds.productData),\r\n designerSessionId: byId(elementIds.designerSessionId),\r\n productTitle: byId(elementIds.productTitle),\r\n shortDescription: byId(elementIds.shortDescription),\r\n sku: byId(elementIds.sku),\r\n grossPrice: byId(elementIds.grossPrice),\r\n netPrice: byId(elementIds.netPrice),\r\n discountPercent: byId(elementIds.discountPercent),\r\n resetBtn: byId(elementIds.resetBtn),\r\n productSpinner: byId(elementIds.productSpinner),\r\n errorContainer: byId(elementIds.errorContainer),\r\n errorMessage: byId(elementIds.errorMessage),\r\n addToCartBtn: byId(elementIds.addToCartBtn),\r\n productForm: byId(elementIds.productForm),\r\n };\r\n this.assignEditFieldElements(elementIds);\r\n\r\n if (this.isFieldVisible(this.fieldsCfg.quantity, false)) {\r\n this.elements.quantity = byId(elementIds.quantity);\r\n this.hasQuantity = true;\r\n this.addNumberInputEventHandlers(this.elements.quantity, (changedElement, changedValue) => {\r\n this.quantityChanged(changedValue);\r\n });\r\n }\r\n\r\n\r\n\r\n //this.popupEditorConfig = config.UI.popupEditor; // Must be JIT-assigned\r\n this.validValuesMap = config.validValuesMap;\r\n this.allFinishesMap = config.allFinishesMap;\r\n\r\n // Child element of addToCartBtn - separate element to avoid overwriting icons, etc. on the btn\r\n this.elements.addToCartBtnText = this.elements.addToCartBtn.querySelector('.text');\r\n\r\n this.elements.productForm.addEventListener('focusin', event => {\r\n // Removes the error message whenever the user focuses on an element, except when the currently\r\n // focused element is the one with an error.\r\n if (!event.target.validationMessage) {\r\n this.toggleErrorMessage(false);\r\n }\r\n });\r\n this.addPopupEditorBtnHandlers();\r\n //this.addPopupEditorHandlers(); // This is done JIT\r\n\r\n this.elements.productData.value = null;\r\n if (this.elements.designerSessionId) {\r\n this.elements.designerSessionId.value = null;\r\n }\r\n /* The product we're viewing and editing. Will have all the changes since we started or last\r\n * reset it. */\r\n this.product = {};\r\n /* True if the current product (this.product) is equal to the default product (this.config.defaultProduct). */\r\n this.isDefaultProduct = true;\r\n /* True if the current product (this.product) is equal to the saved product from the cart, if present\r\n * (this.config.savedProduct). If no cart, then saved and default products are the same: */\r\n this.isSavedProduct = true;\r\n /* Assigns the saved product from the cart (if present). If no cart, then saved and default\r\n * products are the same. This call should update the isDefault- and isSavedProduct properties. */\r\n this.assignProduct(this.config.savedProduct, false);\r\n /* Initialize origProduct as a clone of the product. origProduct is the state of the\r\n * product before the property changed timer starts. We compare the changed product property to\r\n * the origProduct's value to determine if we have a need to call the server with the changes. */\r\n this.resetOrigProduct();\r\n /* changedProduct contains the accumulated changes since the property changed timer started. */\r\n this.changedProduct = {};\r\n this.quantity = this.config.quantity;\r\n this.propertyChangedTimerId = null;\r\n // The number of ms. to wait from a property is changed until we call the Store API update method:\r\n this.propertyChangedUpdateDelay = 500;\r\n // Used for detecting committable input events on the number input edit boxes:\r\n this.isKeyDown = false;\r\n\r\n const resetBtn = this.elements.resetBtn;\r\n this.hasResetBtn = resetBtn !== null && resetBtn !== undefined;\r\n if (this.hasResetBtn) {\r\n resetBtn.onclick = () => {\r\n $(resetBtn).tooltip('hide'); // Because it seems to get stuck on screen after the click\r\n const content = this.config.resetBtnPopup;\r\n window.yesNoPopup.display(content.title, content.body,\r\n {\r\n yesBtnLabel: content.yes,\r\n noBtnLabel: content.no,\r\n onYes: (event) => {\r\n if (this.isPropertyChangedTimerActive()) {\r\n this.clearPropertyChangedTimer();\r\n }\r\n /*await*/ this.resetProduct(true, resetBtn);\r\n },\r\n });\r\n };\r\n }\r\n\r\n this.elements.addToCartBtn.onclick = async () => {\r\n this.disableAddToCartBtn(); // Prevent re-entrancy (double clicks, etc.)\r\n try {\r\n try {\r\n /* We must await because we want the updated product back before stringifying it below.\r\n * Also, we want to validate the latest change(s) to avoid passing invalid data to the\r\n * server. 12/27/2019. */\r\n await this.commitPropertyChangesIfPending(true, false);\r\n } catch (ex) {\r\n if (this.isValidationError(ex)) {\r\n /* We rely on the commitPropertyChanges method to having already displayed info about the\r\n * error to the user, and will swallow the error here; no need to report the error to the\r\n * server. But, we do need to skip the submit call below so just return now: */\r\n return;\r\n } else {\r\n throw ex; // Re-throw since this is an unknown exception\r\n }\r\n }\r\n // For passing to server and storing with order, etc.:\r\n this.elements.productData.value = JSON.stringify(this.product);\r\n if (this.elements.designerSessionId) {\r\n this.elements.designerSessionId.value = this.designerService.sessionId;\r\n }\r\n this.elements.productForm.submit();\r\n } finally {\r\n this.enableAddToCartBtn();\r\n }\r\n };\r\n\r\n /* If we are viewing a product from an order line, then use the query string (order id, order\r\n * line id, and transaction no.) as the id for the session storage object. Otherwise, use the\r\n * product id. */\r\n this.sessionStorageId = this.config.isOrderLine\r\n ? 'productEditor.product.' + window.location.search\r\n : 'productEditor.product.' + this.config.productId;\r\n this.NO_SESSION = 'NoSession';\r\n\r\n /* We have 3 event handlers to deal with different event life cycles on desktop and mobile\r\n * browsers. See https://www.igvita.com/2015/11/20/dont-lose-user-and-app-state-use-page-visibility/\r\n * for more. There will be a minor overhead on modern browsers which handle all 3 events, but we'll\r\n * accept that. If we want to optimize we would need to introduce a product \"dirty\" flag. */\r\n window.addEventListener('beforeunload', () => {\r\n this.saveProductToSessionStorage();\r\n });\r\n window.addEventListener('pagehide', () => {\r\n this.saveProductToSessionStorage();\r\n });\r\n document.addEventListener('visibilitychange', () => {\r\n // fires when user switches tabs, apps, goes to homescreen, etc.\r\n if (document.visibilityState == 'hidden') {\r\n this.saveProductToSessionStorage();\r\n }\r\n });\r\n this.isPreviewDrawingLabelActive = false;\r\n\r\n this.isProductRestorePending = false; // Will be set to true when/if we see there's a product in the session storage\r\n /*await*/ this.restoreProductFromSessionStorage();\r\n }\r\n\r\n togglePreviewDrawingFinishLabels(on) {\r\n this.isPreviewDrawingLabelActive = on;\r\n if (this.onPreviewDrawingFinishLabelsToggled) {\r\n this.onPreviewDrawingFinishLabelsToggled(on);\r\n }\r\n }\r\n\r\n assignEditFieldElements(elementIds) {\r\n // To be overridden by subclasses\r\n }\r\n\r\n /**\r\n * Factory method used by the owner ProductPageViewModel.\r\n */\r\n createPopupEditor(editDrawing) {\r\n //return new MultiStepDoorFrontEditor(editDrawing, this.config.UI.popupEditor);\r\n // Must be overridden by subclasses!\r\n }\r\n\r\n /**\r\n * Click handler for the buttons which trigger the modal popup editors. We need the code in these\r\n * handlers to run before the popup event handlers (see the addPopupEditorHandlers() method) so\r\n * that we can prevent the popups from starting their display process in case of validation errors.\r\n */\r\n async handlePopupEditorBtnClick(event) {\r\n if (this.onNeedPopupEditor) {\r\n this.onNeedPopupEditor();\r\n }\r\n if (this.popupEditorDisplayed) {\r\n try {\r\n /* Commit any pending property changes and wait for updated product information to be returned\r\n * from the server. We must wait because 1) we want to display the updated product in the popup,\r\n * and 2) if there's an error (e.g. server-side validation error), then we don't want to display\r\n * the popup at all. 3/10/2021. */\r\n await this.commitPropertyChangesIfPending(true, false);\r\n\r\n // Ok, the update succeeded. Let's continue and display the popup editor:\r\n const btn = event.target;\r\n const popupId = btn.dataset.target;\r\n const popup = jQuery(popupId);\r\n // Pass needed data to the popup (is accessed in the popup's event handlers):\r\n popup.data('compType', btn.dataset.compType);\r\n popup.data('lastPopupElement', btn);\r\n popup.modal(); // Display the popup\r\n } catch (ex) {\r\n if (this.isValidationError(ex)) {\r\n /* The commitPropertyChanges method should already have displayed info about the error to\r\n * the user. We'll just swallow the exception; no need to report the error to the server. */\r\n } else {\r\n throw ex; // Re-throw since this is an unknown exception\r\n }\r\n }\r\n }\r\n }\r\n\r\n /** Adds button html elements to the buttons array parameter. */\r\n registerPopupEditorButtons(buttons) {\r\n // To be overridden by subclasses\r\n }\r\n\r\n /**\r\n * Assigns click handlers for the buttons which trigger modal popup editors. We need the code in\r\n * these handlers to run before the popup event handlers (see the addPopupEditorHandlers() method)\r\n * so that we can prevent the popups from starting their display process in case of validation errors.\r\n */\r\n addPopupEditorBtnHandlers() {\r\n let buttons = [];\r\n this.registerPopupEditorButtons(buttons);\r\n buttons.forEach(btn => {\r\n /* We are going to manually trigger the popup display in code in the handlePopupEditorBtnClick\r\n * event handler. Let's remove any data-toggle attribute on the button (that attribute causes\r\n * BootStrap to attach event handlers which will automatically display the popup). */\r\n delete btn.dataset.toggle;\r\n btn.onclick = async (event) => await this.handlePopupEditorBtnClick(event);\r\n });\r\n }\r\n\r\n /** Assigns BootStrap modal popup event handlers. */\r\n addPopupEditorHandlers() {\r\n // Can be overridden by subclasses!\r\n const popup = jQuery(this.popupEditorConfig.elements.modalPopup);\r\n\r\n popup.on('show.bs.modal', async (event) => {\r\n if (this.popupEditorDisplayed) {\r\n const typeId = popup.data('compType');\r\n this.editedComponentType = this.componentType[typeId];\r\n this.rebuildPopupEditor(this.editedComponentType);\r\n }\r\n });\r\n\r\n popup.on('shown.bs.modal', event => this.doOnPopupEditorDisplayed(event));\r\n\r\n popup.on('click', 'button[data-reason]', event => {\r\n this.lastPopupCloseReason = event.target.dataset.reason;\r\n //alert('click');\r\n //console.log(`Hiding popup, element was clicked: \"${event.target}\", reason is: \"${this.lastPopupCloseReason}\"`);\r\n });\r\n\r\n popup.on('hide.bs.modal', (event) => {\r\n //alert('hiding');\r\n /* Since Safari (at least on ios) doesn't update document.activeElement on button clicks, we can't rely on that.\r\n * Instead we handle the click event and update lastPopupReason there (see above). */\r\n //this.lastPopupCloseReason = null;\r\n //if (document.activeElement) {\r\n // this.lastPopupCloseReason = document.activeElement.dataset.reason;\r\n // console.log(`Hiding popup, document.activeElement is: \"${document.activeElement}\", reason is: \"${this.lastPopupCloseReason}\"`);\r\n //} else {\r\n // console.log('Hiding popup, document.activeElement is undefined!');\r\n //}\r\n });\r\n\r\n popup.on('hidden.bs.modal', (event) => {\r\n //alert('hidden');\r\n //console.log(`Popup closed, reason: \"${this.lastPopupCloseReason}\"`);\r\n if (this.popupEditorClosed) {\r\n const lastElement = popup.data('lastPopupElement');\r\n this.popupEditorClosed(this, this.editedComponentType, this.lastPopupCloseReason === 'Ok', lastElement);\r\n }\r\n });\r\n }\r\n\r\n rebuildValidValuesMaps(validators) {\r\n // Must be overridden by subclasses!\r\n }\r\n\r\n /**\r\n * Helper method for rebuildPopupEditor. Updates the element's caption. If caption is null or empty,\r\n * then element's display will be set to none.\r\n * @param {any} mode If mode = 0, then the 'd-none' css class will be set if caption is unassigned.\r\n * If mode = 1, then the 'invisible' class will be set instead. If mode = 2, then no class will be set.\r\n */\r\n assignText(element, caption, clearWhenCaptionIsNull, mode = 0) {\r\n if (element) {\r\n if (caption) {\r\n element.textContent = caption;\r\n } else if (clearWhenCaptionIsNull) {\r\n element.textContent = null;\r\n }\r\n if (mode !== 2) {\r\n element.classList.remove('d-none', 'invisible');\r\n if (ObjectUtils.isNullOrEmpty(caption)) {\r\n if (mode === 0) {\r\n element.classList.add('d-none');\r\n } else if (mode === 1) {\r\n element.classList.add('invisible');\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n createSortedGalleryItems(componentType) {\r\n // Must be implemented by subclass!\r\n }\r\n\r\n createGalleryItemFragment(item, clickTarget, hideZoomIcon, zoomIconTooltip) {\r\n const elem = document.createElement('div');\r\n elem.classList.add('col', 'col-6', 'col-lg-4');\r\n const zoomIcon = hideZoomIcon ? '' : ``;\r\n elem.innerHTML = `