@commercetools-frontend/application-components 24.1.0 → 24.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -84,7 +84,7 @@ var _mapInstanceProperty__default = /*#__PURE__*/_interopDefault(_mapInstancePro
84
84
  var useResizeObserver__default = /*#__PURE__*/_interopDefault(useResizeObserver);
85
85
 
86
86
  // NOTE: This string will be replaced on build time with the package version.
87
- var version = "24.1.0";
87
+ var version = "24.2.0";
88
88
 
89
89
  // We keep these tokens as they are related to page layout components which should have
90
90
  // a slightly different layout (margins/paddings) when used within a Custom View panel.
@@ -637,7 +637,7 @@ FormDialog.TextTitle = TextTitle;
637
637
 
638
638
  const CustomViewSelector = /*#__PURE__*/react$1.lazy(() => {
639
639
  if (typeof window !== 'undefined' && typeof window.app !== 'undefined') {
640
- return Promise.resolve().then(function () { return require('./custom-views-selector-3aa3434d.cjs.dev.js' /* webpackChunkName: "custom-views-selector" */); });
640
+ return Promise.resolve().then(function () { return require('./custom-views-selector-27c160fc.cjs.dev.js' /* webpackChunkName: "custom-views-selector" */); });
641
641
  }
642
642
  return _Promise__default["default"].resolve({
643
643
  default: () => null
@@ -2430,7 +2430,7 @@ const ContentWrapper = /*#__PURE__*/_styled__default["default"]("div", process.e
2430
2430
  styles: "height:100%"
2431
2431
  } : {
2432
2432
  name: "13udsys",
2433
- styles: "height:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["custom-view-loader.tsx"],"names":[],"mappings":"AAyCiC","file":"custom-view-loader.tsx","sourcesContent":["import { useCallback, useEffect, useRef } from 'react';\nimport styled from '@emotion/styled';\nimport { useAllFeatureToggles } from '@flopflip/react-broadcast';\nimport { useIntl } from 'react-intl';\nimport { useShowNotification } from '@commercetools-frontend/actions-global';\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';\nimport {\n  CUSTOM_VIEWS_EVENTS_NAMES,\n  CUSTOM_VIEWS_EVENTS_META,\n  DOMAINS,\n  NOTIFICATION_KINDS_PAGE,\n  CustomViewData,\n} from '@commercetools-frontend/constants';\nimport { reportErrorToSentry } from '@commercetools-frontend/sentry';\nimport ModalPage from '../../modal-pages/internals/modal-page';\nimport messages from './messages';\n\ntype TCustomViewIframeMessage = {\n  source: string;\n  destination: string;\n  eventName: string;\n  eventData: Record<string, unknown>;\n};\n\ntype TCustomViewLoaderProps = {\n  customView: CustomViewData;\n  hostUrl?: string;\n  onClose: () => void;\n};\n\nconst isIframeReady = (iFrameElementRef: HTMLIFrameElement) => {\n  try {\n    return iFrameElementRef?.contentWindow?.document.readyState === 'complete';\n  } catch {\n    // Trying to access the contentWindow of a cross-origin iFrame will throw an error.\n    // We are not supposed to even get here because the iFrame must use\n    // a URL from our very same domain (the custom view is proxied through our http-proxy service).\n    return false;\n  }\n};\n\nconst ContentWrapper = styled.div`\n  height: 100%;\n`;\n\nconst CustomPanelIframe = styled.iframe`\n  height: 100%;\n  width: 100%;\n  border: none;\n`;\n\nfunction CustomViewLoader(props: TCustomViewLoaderProps) {\n  const iFrameElementRef = useRef<HTMLIFrameElement>(null);\n  const dataLocale = useApplicationContext((context) => context.dataLocale);\n  const projectKey = useApplicationContext((context) => context.project?.key);\n  const featureFlags = useAllFeatureToggles();\n  const iFrameCommunicationChannel = useRef(new MessageChannel());\n  const showNotification = useShowNotification();\n  const intl = useIntl();\n\n  const sendInitializationMessages = useCallback(() => {\n    // Transfer port2 to the iFrame so it can send messages back privately\n    iFrameElementRef.current?.contentWindow?.postMessage(\n      CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP,\n      window.location.href,\n      [iFrameCommunicationChannel.current.port2]\n    );\n\n    // Send the initialization message to the iFrame\n    iFrameCommunicationChannel.current.port1.postMessage({\n      source: CUSTOM_VIEWS_EVENTS_META.HOST_APPLICATION_CODE,\n      destination: `${CUSTOM_VIEWS_EVENTS_META.CUSTOM_VIEW_KEY_PREFIX}${props.customView.id}`,\n      eventName: CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_INITIALIZATION,\n      eventData: {\n        context: {\n          dataLocale,\n          projectKey,\n          featureFlags,\n          customViewConfig: props.customView,\n          hostUrl: props.hostUrl || window.location.href,\n        },\n      },\n    } as TCustomViewIframeMessage);\n  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);\n\n  const messageFromIFrameHandler = useCallback(\n    (event: MessageEvent) => {\n      if (event.data.origin === window.location.origin) {\n        switch (event.data.eventName) {\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:\n            props.onClose();\n            break;\n        }\n      }\n    },\n    [props]\n  );\n\n  // onLoad handler is called from the iFrame even where the URL is not valid\n  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready\n  const onLoadSuccessHandler = useCallback(() => {\n    // Show error and block if the iFrame is not ready\n    // (error loading it)\n    if (!isIframeReady(iFrameElementRef.current!)) {\n      showNotification({\n        domain: DOMAINS.PAGE,\n        kind: NOTIFICATION_KINDS_PAGE.error,\n        text: intl.formatMessage(messages.loadError),\n      });\n      return;\n    }\n\n    // Listen for messages from the iFrame\n    iFrameCommunicationChannel.current.port1.onmessage =\n      messageFromIFrameHandler;\n\n    // TODO: Locally (starter templates) we're seeing a little delay while rendering\n    // the iFrame. The CustomViewShell gets rendered, but the effect to start listening\n    // for messages is triggered after the iFrame is ready.\n    // This is a temporary fix to avoid the situation when running locally.\n    setTimeout(() => {\n      sendInitializationMessages();\n    }, 500);\n\n    // We want the effect to run only once so we don't need the dependencies array.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, []);\n\n  useEffect(() => {\n    // Close the channel when the component unmounts\n    const communicationChannel = iFrameCommunicationChannel.current;\n    return () => {\n      communicationChannel?.port1.close();\n    };\n  }, []);\n\n  // Currently we only support custom panels\n  if (props.customView.type !== 'CustomPanel') {\n    reportErrorToSentry(\n      new Error(\n        `CustomViewLoader: Provided Custom View has an unsupported type: ${props.customView.type}. Supported types: ['CustomPanel'].`\n      )\n    );\n    return null;\n  }\n  const panelSize = (props.customView.typeSettings?.size?.toLocaleLowerCase() ||\n    'large') as Lowercase<'SMALL' | 'LARGE'>;\n  const iFrameUrl = [\n    window.location.origin,\n    'custom-views',\n    props.customView.id,\n    'projects',\n    projectKey,\n  ].join('/');\n\n  return (\n    <ModalPage\n      isOpen\n      onClose={props.onClose}\n      size={panelSize === 'small' ? 10 : 30}\n      title={`Custom View: ${props.customView.defaultLabel}`}\n      hideTopBar\n    >\n      <ContentWrapper>\n        <CustomPanelIframe\n          id={`custom-view-${props.customView.id}`}\n          key={`custom-view-${props.customView.id}`}\n          title={`Custom View: ${props.customView.defaultLabel}`}\n          ref={iFrameElementRef}\n          src={iFrameUrl}\n          onLoad={onLoadSuccessHandler}\n        />\n      </ContentWrapper>\n    </ModalPage>\n  );\n}\n\nexport default CustomViewLoader;\n"]} */",
2433
+ styles: "height:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["custom-view-loader.tsx"],"names":[],"mappings":"AAyCiC","file":"custom-view-loader.tsx","sourcesContent":["import { useCallback, useEffect, useRef } from 'react';\nimport styled from '@emotion/styled';\nimport { useAllFeatureToggles } from '@flopflip/react-broadcast';\nimport { useIntl } from 'react-intl';\nimport { useShowNotification } from '@commercetools-frontend/actions-global';\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';\nimport {\n  CUSTOM_VIEWS_EVENTS_NAMES,\n  CUSTOM_VIEWS_EVENTS_META,\n  DOMAINS,\n  NOTIFICATION_KINDS_PAGE,\n  CustomViewData,\n} from '@commercetools-frontend/constants';\nimport { reportErrorToSentry } from '@commercetools-frontend/sentry';\nimport ModalPage from '../../modal-pages/internals/modal-page';\nimport messages from './messages';\n\ntype TCustomViewIframeMessage = {\n  source: string;\n  destination: string;\n  eventName: string;\n  eventData: Record<string, unknown>;\n};\n\ntype TCustomViewLoaderProps = {\n  customView: CustomViewData;\n  hostUrl?: string;\n  onClose: () => void;\n};\n\nconst isIframeReady = (iFrameElementRef: HTMLIFrameElement) => {\n  try {\n    return iFrameElementRef?.contentWindow?.document.readyState === 'complete';\n  } catch {\n    // Trying to access the contentWindow of a cross-origin iFrame will throw an error.\n    // We are not supposed to even get here because the iFrame must use\n    // a URL from our very same domain (the custom view is proxied through our http-proxy service).\n    return false;\n  }\n};\n\nconst ContentWrapper = styled.div`\n  height: 100%;\n`;\n\nconst CustomPanelIframe = styled.iframe`\n  height: 100%;\n  width: 100%;\n  border: none;\n`;\n\nfunction CustomViewLoader(props: TCustomViewLoaderProps) {\n  const iFrameElementRef = useRef<HTMLIFrameElement>(null);\n  const dataLocale = useApplicationContext((context) => context.dataLocale);\n  const projectKey = useApplicationContext((context) => context.project?.key);\n  const featureFlags = useAllFeatureToggles();\n  const iFrameCommunicationChannel = useRef(new MessageChannel());\n  const showNotification = useShowNotification();\n  const intl = useIntl();\n  const hasSentInitializationMessages = useRef(false);\n\n  const sendInitializationMessages = useCallback(() => {\n    // If we have already sent the initialization messages, do not send them again.\n    // The message can be sent either as a response to the CUSTOM_VIEW_READY message\n    // or as a result of a setTimeout after 500ms. In any case, this message should be sent only once.\n    if (hasSentInitializationMessages.current) {\n      return;\n    }\n\n    // Transfer port2 to the iFrame so it can send messages back privately\n    iFrameElementRef.current?.contentWindow?.postMessage(\n      CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP,\n      window.location.href,\n      [iFrameCommunicationChannel.current.port2]\n    );\n\n    // Send the initialization message to the iFrame\n    iFrameCommunicationChannel.current.port1.postMessage({\n      source: CUSTOM_VIEWS_EVENTS_META.HOST_APPLICATION_CODE,\n      destination: `${CUSTOM_VIEWS_EVENTS_META.CUSTOM_VIEW_KEY_PREFIX}${props.customView.id}`,\n      eventName: CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_INITIALIZATION,\n      eventData: {\n        context: {\n          dataLocale,\n          projectKey,\n          featureFlags,\n          customViewConfig: props.customView,\n          hostUrl: props.hostUrl || window.location.href,\n        },\n      },\n    } as TCustomViewIframeMessage);\n\n    hasSentInitializationMessages.current = true;\n  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);\n\n  const messageFromIFrameHandler = useCallback(\n    (event: MessageEvent) => {\n      if (event.data.origin === window.location.origin) {\n        switch (event.data.eventName) {\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:\n            props.onClose();\n            break;\n          // This message will only be sent by custom view shell older than v24.x\n          // For backwards compatibility we will send the initialization messages\n          // after 500ms if this message was not received by then.\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_READY: {\n            sendInitializationMessages();\n          }\n        }\n      }\n    },\n    [props, sendInitializationMessages]\n  );\n\n  // onLoad handler is called from the iFrame even where the URL is not valid\n  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready\n  const onLoadSuccessHandler = useCallback(() => {\n    // Show error and block if the iFrame is not ready\n    // (error loading it)\n    if (!isIframeReady(iFrameElementRef.current!)) {\n      showNotification({\n        domain: DOMAINS.PAGE,\n        kind: NOTIFICATION_KINDS_PAGE.error,\n        text: intl.formatMessage(messages.loadError),\n      });\n      return;\n    }\n\n    // This is for backwards compatibility with custom view shell older than v24.0.0\n    // where the custom-view-shell does not send the CUSTOM_VIEW_READY message yet.\n    setTimeout(() => {\n      sendInitializationMessages();\n    }, 500);\n\n    // We want the effect to run only once so we don't need the dependencies array.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, []);\n\n  useEffect(() => {\n    // Close the channel when the component unmounts\n    const communicationChannel = iFrameCommunicationChannel.current;\n\n    // Listen for messages from the iFrame\n    iFrameCommunicationChannel.current!.port1.onmessage =\n      messageFromIFrameHandler;\n    window.addEventListener('message', messageFromIFrameHandler);\n\n    return () => {\n      window.removeEventListener('message', messageFromIFrameHandler);\n      communicationChannel?.port1.close();\n    };\n  }, []);\n\n  // Currently we only support custom panels\n  if (props.customView.type !== 'CustomPanel') {\n    reportErrorToSentry(\n      new Error(\n        `CustomViewLoader: Provided Custom View has an unsupported type: ${props.customView.type}. Supported types: ['CustomPanel'].`\n      )\n    );\n    return null;\n  }\n  const panelSize = (props.customView.typeSettings?.size?.toLocaleLowerCase() ||\n    'large') as Lowercase<'SMALL' | 'LARGE'>;\n  const iFrameUrl = [\n    window.location.origin,\n    'custom-views',\n    props.customView.id,\n    'projects',\n    projectKey,\n  ].join('/');\n\n  return (\n    <ModalPage\n      isOpen\n      onClose={props.onClose}\n      size={panelSize === 'small' ? 10 : 30}\n      title={`Custom View: ${props.customView.defaultLabel}`}\n      hideTopBar\n    >\n      <ContentWrapper>\n        <CustomPanelIframe\n          id={`custom-view-${props.customView.id}`}\n          key={`custom-view-${props.customView.id}`}\n          title={`Custom View: ${props.customView.defaultLabel}`}\n          ref={iFrameElementRef}\n          src={iFrameUrl}\n          onLoad={onLoadSuccessHandler}\n        />\n      </ContentWrapper>\n    </ModalPage>\n  );\n}\n\nexport default CustomViewLoader;\n"]} */",
2434
2434
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__$2
2435
2435
  });
2436
2436
  const CustomPanelIframe = /*#__PURE__*/_styled__default["default"]("iframe", process.env.NODE_ENV === "production" ? {
@@ -2443,7 +2443,7 @@ const CustomPanelIframe = /*#__PURE__*/_styled__default["default"]("iframe", pro
2443
2443
  styles: "height:100%;width:100%;border:none"
2444
2444
  } : {
2445
2445
  name: "174lt7a",
2446
- styles: "height:100%;width:100%;border:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["custom-view-loader.tsx"],"names":[],"mappings":"AA6CuC","file":"custom-view-loader.tsx","sourcesContent":["import { useCallback, useEffect, useRef } from 'react';\nimport styled from '@emotion/styled';\nimport { useAllFeatureToggles } from '@flopflip/react-broadcast';\nimport { useIntl } from 'react-intl';\nimport { useShowNotification } from '@commercetools-frontend/actions-global';\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';\nimport {\n  CUSTOM_VIEWS_EVENTS_NAMES,\n  CUSTOM_VIEWS_EVENTS_META,\n  DOMAINS,\n  NOTIFICATION_KINDS_PAGE,\n  CustomViewData,\n} from '@commercetools-frontend/constants';\nimport { reportErrorToSentry } from '@commercetools-frontend/sentry';\nimport ModalPage from '../../modal-pages/internals/modal-page';\nimport messages from './messages';\n\ntype TCustomViewIframeMessage = {\n  source: string;\n  destination: string;\n  eventName: string;\n  eventData: Record<string, unknown>;\n};\n\ntype TCustomViewLoaderProps = {\n  customView: CustomViewData;\n  hostUrl?: string;\n  onClose: () => void;\n};\n\nconst isIframeReady = (iFrameElementRef: HTMLIFrameElement) => {\n  try {\n    return iFrameElementRef?.contentWindow?.document.readyState === 'complete';\n  } catch {\n    // Trying to access the contentWindow of a cross-origin iFrame will throw an error.\n    // We are not supposed to even get here because the iFrame must use\n    // a URL from our very same domain (the custom view is proxied through our http-proxy service).\n    return false;\n  }\n};\n\nconst ContentWrapper = styled.div`\n  height: 100%;\n`;\n\nconst CustomPanelIframe = styled.iframe`\n  height: 100%;\n  width: 100%;\n  border: none;\n`;\n\nfunction CustomViewLoader(props: TCustomViewLoaderProps) {\n  const iFrameElementRef = useRef<HTMLIFrameElement>(null);\n  const dataLocale = useApplicationContext((context) => context.dataLocale);\n  const projectKey = useApplicationContext((context) => context.project?.key);\n  const featureFlags = useAllFeatureToggles();\n  const iFrameCommunicationChannel = useRef(new MessageChannel());\n  const showNotification = useShowNotification();\n  const intl = useIntl();\n\n  const sendInitializationMessages = useCallback(() => {\n    // Transfer port2 to the iFrame so it can send messages back privately\n    iFrameElementRef.current?.contentWindow?.postMessage(\n      CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP,\n      window.location.href,\n      [iFrameCommunicationChannel.current.port2]\n    );\n\n    // Send the initialization message to the iFrame\n    iFrameCommunicationChannel.current.port1.postMessage({\n      source: CUSTOM_VIEWS_EVENTS_META.HOST_APPLICATION_CODE,\n      destination: `${CUSTOM_VIEWS_EVENTS_META.CUSTOM_VIEW_KEY_PREFIX}${props.customView.id}`,\n      eventName: CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_INITIALIZATION,\n      eventData: {\n        context: {\n          dataLocale,\n          projectKey,\n          featureFlags,\n          customViewConfig: props.customView,\n          hostUrl: props.hostUrl || window.location.href,\n        },\n      },\n    } as TCustomViewIframeMessage);\n  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);\n\n  const messageFromIFrameHandler = useCallback(\n    (event: MessageEvent) => {\n      if (event.data.origin === window.location.origin) {\n        switch (event.data.eventName) {\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:\n            props.onClose();\n            break;\n        }\n      }\n    },\n    [props]\n  );\n\n  // onLoad handler is called from the iFrame even where the URL is not valid\n  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready\n  const onLoadSuccessHandler = useCallback(() => {\n    // Show error and block if the iFrame is not ready\n    // (error loading it)\n    if (!isIframeReady(iFrameElementRef.current!)) {\n      showNotification({\n        domain: DOMAINS.PAGE,\n        kind: NOTIFICATION_KINDS_PAGE.error,\n        text: intl.formatMessage(messages.loadError),\n      });\n      return;\n    }\n\n    // Listen for messages from the iFrame\n    iFrameCommunicationChannel.current.port1.onmessage =\n      messageFromIFrameHandler;\n\n    // TODO: Locally (starter templates) we're seeing a little delay while rendering\n    // the iFrame. The CustomViewShell gets rendered, but the effect to start listening\n    // for messages is triggered after the iFrame is ready.\n    // This is a temporary fix to avoid the situation when running locally.\n    setTimeout(() => {\n      sendInitializationMessages();\n    }, 500);\n\n    // We want the effect to run only once so we don't need the dependencies array.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, []);\n\n  useEffect(() => {\n    // Close the channel when the component unmounts\n    const communicationChannel = iFrameCommunicationChannel.current;\n    return () => {\n      communicationChannel?.port1.close();\n    };\n  }, []);\n\n  // Currently we only support custom panels\n  if (props.customView.type !== 'CustomPanel') {\n    reportErrorToSentry(\n      new Error(\n        `CustomViewLoader: Provided Custom View has an unsupported type: ${props.customView.type}. Supported types: ['CustomPanel'].`\n      )\n    );\n    return null;\n  }\n  const panelSize = (props.customView.typeSettings?.size?.toLocaleLowerCase() ||\n    'large') as Lowercase<'SMALL' | 'LARGE'>;\n  const iFrameUrl = [\n    window.location.origin,\n    'custom-views',\n    props.customView.id,\n    'projects',\n    projectKey,\n  ].join('/');\n\n  return (\n    <ModalPage\n      isOpen\n      onClose={props.onClose}\n      size={panelSize === 'small' ? 10 : 30}\n      title={`Custom View: ${props.customView.defaultLabel}`}\n      hideTopBar\n    >\n      <ContentWrapper>\n        <CustomPanelIframe\n          id={`custom-view-${props.customView.id}`}\n          key={`custom-view-${props.customView.id}`}\n          title={`Custom View: ${props.customView.defaultLabel}`}\n          ref={iFrameElementRef}\n          src={iFrameUrl}\n          onLoad={onLoadSuccessHandler}\n        />\n      </ContentWrapper>\n    </ModalPage>\n  );\n}\n\nexport default CustomViewLoader;\n"]} */",
2446
+ styles: "height:100%;width:100%;border:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["custom-view-loader.tsx"],"names":[],"mappings":"AA6CuC","file":"custom-view-loader.tsx","sourcesContent":["import { useCallback, useEffect, useRef } from 'react';\nimport styled from '@emotion/styled';\nimport { useAllFeatureToggles } from '@flopflip/react-broadcast';\nimport { useIntl } from 'react-intl';\nimport { useShowNotification } from '@commercetools-frontend/actions-global';\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';\nimport {\n  CUSTOM_VIEWS_EVENTS_NAMES,\n  CUSTOM_VIEWS_EVENTS_META,\n  DOMAINS,\n  NOTIFICATION_KINDS_PAGE,\n  CustomViewData,\n} from '@commercetools-frontend/constants';\nimport { reportErrorToSentry } from '@commercetools-frontend/sentry';\nimport ModalPage from '../../modal-pages/internals/modal-page';\nimport messages from './messages';\n\ntype TCustomViewIframeMessage = {\n  source: string;\n  destination: string;\n  eventName: string;\n  eventData: Record<string, unknown>;\n};\n\ntype TCustomViewLoaderProps = {\n  customView: CustomViewData;\n  hostUrl?: string;\n  onClose: () => void;\n};\n\nconst isIframeReady = (iFrameElementRef: HTMLIFrameElement) => {\n  try {\n    return iFrameElementRef?.contentWindow?.document.readyState === 'complete';\n  } catch {\n    // Trying to access the contentWindow of a cross-origin iFrame will throw an error.\n    // We are not supposed to even get here because the iFrame must use\n    // a URL from our very same domain (the custom view is proxied through our http-proxy service).\n    return false;\n  }\n};\n\nconst ContentWrapper = styled.div`\n  height: 100%;\n`;\n\nconst CustomPanelIframe = styled.iframe`\n  height: 100%;\n  width: 100%;\n  border: none;\n`;\n\nfunction CustomViewLoader(props: TCustomViewLoaderProps) {\n  const iFrameElementRef = useRef<HTMLIFrameElement>(null);\n  const dataLocale = useApplicationContext((context) => context.dataLocale);\n  const projectKey = useApplicationContext((context) => context.project?.key);\n  const featureFlags = useAllFeatureToggles();\n  const iFrameCommunicationChannel = useRef(new MessageChannel());\n  const showNotification = useShowNotification();\n  const intl = useIntl();\n  const hasSentInitializationMessages = useRef(false);\n\n  const sendInitializationMessages = useCallback(() => {\n    // If we have already sent the initialization messages, do not send them again.\n    // The message can be sent either as a response to the CUSTOM_VIEW_READY message\n    // or as a result of a setTimeout after 500ms. In any case, this message should be sent only once.\n    if (hasSentInitializationMessages.current) {\n      return;\n    }\n\n    // Transfer port2 to the iFrame so it can send messages back privately\n    iFrameElementRef.current?.contentWindow?.postMessage(\n      CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP,\n      window.location.href,\n      [iFrameCommunicationChannel.current.port2]\n    );\n\n    // Send the initialization message to the iFrame\n    iFrameCommunicationChannel.current.port1.postMessage({\n      source: CUSTOM_VIEWS_EVENTS_META.HOST_APPLICATION_CODE,\n      destination: `${CUSTOM_VIEWS_EVENTS_META.CUSTOM_VIEW_KEY_PREFIX}${props.customView.id}`,\n      eventName: CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_INITIALIZATION,\n      eventData: {\n        context: {\n          dataLocale,\n          projectKey,\n          featureFlags,\n          customViewConfig: props.customView,\n          hostUrl: props.hostUrl || window.location.href,\n        },\n      },\n    } as TCustomViewIframeMessage);\n\n    hasSentInitializationMessages.current = true;\n  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);\n\n  const messageFromIFrameHandler = useCallback(\n    (event: MessageEvent) => {\n      if (event.data.origin === window.location.origin) {\n        switch (event.data.eventName) {\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:\n            props.onClose();\n            break;\n          // This message will only be sent by custom view shell older than v24.x\n          // For backwards compatibility we will send the initialization messages\n          // after 500ms if this message was not received by then.\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_READY: {\n            sendInitializationMessages();\n          }\n        }\n      }\n    },\n    [props, sendInitializationMessages]\n  );\n\n  // onLoad handler is called from the iFrame even where the URL is not valid\n  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready\n  const onLoadSuccessHandler = useCallback(() => {\n    // Show error and block if the iFrame is not ready\n    // (error loading it)\n    if (!isIframeReady(iFrameElementRef.current!)) {\n      showNotification({\n        domain: DOMAINS.PAGE,\n        kind: NOTIFICATION_KINDS_PAGE.error,\n        text: intl.formatMessage(messages.loadError),\n      });\n      return;\n    }\n\n    // This is for backwards compatibility with custom view shell older than v24.0.0\n    // where the custom-view-shell does not send the CUSTOM_VIEW_READY message yet.\n    setTimeout(() => {\n      sendInitializationMessages();\n    }, 500);\n\n    // We want the effect to run only once so we don't need the dependencies array.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, []);\n\n  useEffect(() => {\n    // Close the channel when the component unmounts\n    const communicationChannel = iFrameCommunicationChannel.current;\n\n    // Listen for messages from the iFrame\n    iFrameCommunicationChannel.current!.port1.onmessage =\n      messageFromIFrameHandler;\n    window.addEventListener('message', messageFromIFrameHandler);\n\n    return () => {\n      window.removeEventListener('message', messageFromIFrameHandler);\n      communicationChannel?.port1.close();\n    };\n  }, []);\n\n  // Currently we only support custom panels\n  if (props.customView.type !== 'CustomPanel') {\n    reportErrorToSentry(\n      new Error(\n        `CustomViewLoader: Provided Custom View has an unsupported type: ${props.customView.type}. Supported types: ['CustomPanel'].`\n      )\n    );\n    return null;\n  }\n  const panelSize = (props.customView.typeSettings?.size?.toLocaleLowerCase() ||\n    'large') as Lowercase<'SMALL' | 'LARGE'>;\n  const iFrameUrl = [\n    window.location.origin,\n    'custom-views',\n    props.customView.id,\n    'projects',\n    projectKey,\n  ].join('/');\n\n  return (\n    <ModalPage\n      isOpen\n      onClose={props.onClose}\n      size={panelSize === 'small' ? 10 : 30}\n      title={`Custom View: ${props.customView.defaultLabel}`}\n      hideTopBar\n    >\n      <ContentWrapper>\n        <CustomPanelIframe\n          id={`custom-view-${props.customView.id}`}\n          key={`custom-view-${props.customView.id}`}\n          title={`Custom View: ${props.customView.defaultLabel}`}\n          ref={iFrameElementRef}\n          src={iFrameUrl}\n          onLoad={onLoadSuccessHandler}\n        />\n      </ContentWrapper>\n    </ModalPage>\n  );\n}\n\nexport default CustomViewLoader;\n"]} */",
2447
2447
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__$2
2448
2448
  });
2449
2449
  function CustomViewLoader(props) {
@@ -2454,7 +2454,15 @@ function CustomViewLoader(props) {
2454
2454
  const iFrameCommunicationChannel = react$1.useRef(new MessageChannel());
2455
2455
  const showNotification = actionsGlobal.useShowNotification();
2456
2456
  const intl = reactIntl.useIntl();
2457
+ const hasSentInitializationMessages = react$1.useRef(false);
2457
2458
  const sendInitializationMessages = react$1.useCallback(() => {
2459
+ // If we have already sent the initialization messages, do not send them again.
2460
+ // The message can be sent either as a response to the CUSTOM_VIEW_READY message
2461
+ // or as a result of a setTimeout after 500ms. In any case, this message should be sent only once.
2462
+ if (hasSentInitializationMessages.current) {
2463
+ return;
2464
+ }
2465
+
2458
2466
  // Transfer port2 to the iFrame so it can send messages back privately
2459
2467
  iFrameElementRef.current?.contentWindow?.postMessage(constants.CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP, window.location.href, [iFrameCommunicationChannel.current.port2]);
2460
2468
 
@@ -2473,6 +2481,7 @@ function CustomViewLoader(props) {
2473
2481
  }
2474
2482
  }
2475
2483
  });
2484
+ hasSentInitializationMessages.current = true;
2476
2485
  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);
2477
2486
  const messageFromIFrameHandler = react$1.useCallback(event => {
2478
2487
  if (event.data.origin === window.location.origin) {
@@ -2480,9 +2489,16 @@ function CustomViewLoader(props) {
2480
2489
  case constants.CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:
2481
2490
  props.onClose();
2482
2491
  break;
2492
+ // This message will only be sent by custom view shell older than v24.x
2493
+ // For backwards compatibility we will send the initialization messages
2494
+ // after 500ms if this message was not received by then.
2495
+ case constants.CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_READY:
2496
+ {
2497
+ sendInitializationMessages();
2498
+ }
2483
2499
  }
2484
2500
  }
2485
- }, [props]);
2501
+ }, [props, sendInitializationMessages]);
2486
2502
 
2487
2503
  // onLoad handler is called from the iFrame even where the URL is not valid
2488
2504
  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready
@@ -2498,13 +2514,8 @@ function CustomViewLoader(props) {
2498
2514
  return;
2499
2515
  }
2500
2516
 
2501
- // Listen for messages from the iFrame
2502
- iFrameCommunicationChannel.current.port1.onmessage = messageFromIFrameHandler;
2503
-
2504
- // TODO: Locally (starter templates) we're seeing a little delay while rendering
2505
- // the iFrame. The CustomViewShell gets rendered, but the effect to start listening
2506
- // for messages is triggered after the iFrame is ready.
2507
- // This is a temporary fix to avoid the situation when running locally.
2517
+ // This is for backwards compatibility with custom view shell older than v24.0.0
2518
+ // where the custom-view-shell does not send the CUSTOM_VIEW_READY message yet.
2508
2519
  _setTimeout__default["default"](() => {
2509
2520
  sendInitializationMessages();
2510
2521
  }, 500);
@@ -2515,7 +2526,12 @@ function CustomViewLoader(props) {
2515
2526
  react$1.useEffect(() => {
2516
2527
  // Close the channel when the component unmounts
2517
2528
  const communicationChannel = iFrameCommunicationChannel.current;
2529
+
2530
+ // Listen for messages from the iFrame
2531
+ iFrameCommunicationChannel.current.port1.onmessage = messageFromIFrameHandler;
2532
+ window.addEventListener('message', messageFromIFrameHandler);
2518
2533
  return () => {
2534
+ window.removeEventListener('message', messageFromIFrameHandler);
2519
2535
  communicationChannel?.port1.close();
2520
2536
  };
2521
2537
  }, []);
@@ -83,7 +83,7 @@ var _mapInstanceProperty__default = /*#__PURE__*/_interopDefault(_mapInstancePro
83
83
  var useResizeObserver__default = /*#__PURE__*/_interopDefault(useResizeObserver);
84
84
 
85
85
  // NOTE: This string will be replaced on build time with the package version.
86
- var version = "24.1.0";
86
+ var version = "24.2.0";
87
87
 
88
88
  // We keep these tokens as they are related to page layout components which should have
89
89
  // a slightly different layout (margins/paddings) when used within a Custom View panel.
@@ -500,7 +500,7 @@ FormDialog.TextTitle = TextTitle;
500
500
 
501
501
  const CustomViewSelector = /*#__PURE__*/react$1.lazy(() => {
502
502
  if (typeof window !== 'undefined' && typeof window.app !== 'undefined') {
503
- return Promise.resolve().then(function () { return require('./custom-views-selector-525f7bfc.cjs.prod.js' /* webpackChunkName: "custom-views-selector" */); });
503
+ return Promise.resolve().then(function () { return require('./custom-views-selector-2199c71c.cjs.prod.js' /* webpackChunkName: "custom-views-selector" */); });
504
504
  }
505
505
  return _Promise__default["default"].resolve({
506
506
  default: () => null
@@ -1814,7 +1814,15 @@ function CustomViewLoader(props) {
1814
1814
  const iFrameCommunicationChannel = react$1.useRef(new MessageChannel());
1815
1815
  const showNotification = actionsGlobal.useShowNotification();
1816
1816
  const intl = reactIntl.useIntl();
1817
+ const hasSentInitializationMessages = react$1.useRef(false);
1817
1818
  const sendInitializationMessages = react$1.useCallback(() => {
1819
+ // If we have already sent the initialization messages, do not send them again.
1820
+ // The message can be sent either as a response to the CUSTOM_VIEW_READY message
1821
+ // or as a result of a setTimeout after 500ms. In any case, this message should be sent only once.
1822
+ if (hasSentInitializationMessages.current) {
1823
+ return;
1824
+ }
1825
+
1818
1826
  // Transfer port2 to the iFrame so it can send messages back privately
1819
1827
  iFrameElementRef.current?.contentWindow?.postMessage(constants.CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP, window.location.href, [iFrameCommunicationChannel.current.port2]);
1820
1828
 
@@ -1833,6 +1841,7 @@ function CustomViewLoader(props) {
1833
1841
  }
1834
1842
  }
1835
1843
  });
1844
+ hasSentInitializationMessages.current = true;
1836
1845
  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);
1837
1846
  const messageFromIFrameHandler = react$1.useCallback(event => {
1838
1847
  if (event.data.origin === window.location.origin) {
@@ -1840,9 +1849,16 @@ function CustomViewLoader(props) {
1840
1849
  case constants.CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:
1841
1850
  props.onClose();
1842
1851
  break;
1852
+ // This message will only be sent by custom view shell older than v24.x
1853
+ // For backwards compatibility we will send the initialization messages
1854
+ // after 500ms if this message was not received by then.
1855
+ case constants.CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_READY:
1856
+ {
1857
+ sendInitializationMessages();
1858
+ }
1843
1859
  }
1844
1860
  }
1845
- }, [props]);
1861
+ }, [props, sendInitializationMessages]);
1846
1862
 
1847
1863
  // onLoad handler is called from the iFrame even where the URL is not valid
1848
1864
  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready
@@ -1858,13 +1874,8 @@ function CustomViewLoader(props) {
1858
1874
  return;
1859
1875
  }
1860
1876
 
1861
- // Listen for messages from the iFrame
1862
- iFrameCommunicationChannel.current.port1.onmessage = messageFromIFrameHandler;
1863
-
1864
- // TODO: Locally (starter templates) we're seeing a little delay while rendering
1865
- // the iFrame. The CustomViewShell gets rendered, but the effect to start listening
1866
- // for messages is triggered after the iFrame is ready.
1867
- // This is a temporary fix to avoid the situation when running locally.
1877
+ // This is for backwards compatibility with custom view shell older than v24.0.0
1878
+ // where the custom-view-shell does not send the CUSTOM_VIEW_READY message yet.
1868
1879
  _setTimeout__default["default"](() => {
1869
1880
  sendInitializationMessages();
1870
1881
  }, 500);
@@ -1875,7 +1886,12 @@ function CustomViewLoader(props) {
1875
1886
  react$1.useEffect(() => {
1876
1887
  // Close the channel when the component unmounts
1877
1888
  const communicationChannel = iFrameCommunicationChannel.current;
1889
+
1890
+ // Listen for messages from the iFrame
1891
+ iFrameCommunicationChannel.current.port1.onmessage = messageFromIFrameHandler;
1892
+ window.addEventListener('message', messageFromIFrameHandler);
1878
1893
  return () => {
1894
+ window.removeEventListener('message', messageFromIFrameHandler);
1879
1895
  communicationChannel?.port1.close();
1880
1896
  };
1881
1897
  }, []);
@@ -48,7 +48,7 @@ import useResizeObserver from '@react-hook/resize-observer';
48
48
  import { useMutationObserver } from '@commercetools-uikit/hooks';
49
49
 
50
50
  // NOTE: This string will be replaced on build time with the package version.
51
- var version = "24.1.0";
51
+ var version = "24.2.0";
52
52
 
53
53
  // We keep these tokens as they are related to page layout components which should have
54
54
  // a slightly different layout (margins/paddings) when used within a Custom View panel.
@@ -601,7 +601,7 @@ FormDialog.TextTitle = TextTitle;
601
601
 
602
602
  const CustomViewSelector = /*#__PURE__*/lazy(() => {
603
603
  if (typeof window !== 'undefined' && typeof window.app !== 'undefined') {
604
- return import('./custom-views-selector-0572529e.esm.js' /* webpackChunkName: "custom-views-selector" */);
604
+ return import('./custom-views-selector-dbb1a703.esm.js' /* webpackChunkName: "custom-views-selector" */);
605
605
  }
606
606
  return _Promise.resolve({
607
607
  default: () => null
@@ -2394,7 +2394,7 @@ const ContentWrapper = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "pro
2394
2394
  styles: "height:100%"
2395
2395
  } : {
2396
2396
  name: "13udsys",
2397
- styles: "height:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["custom-view-loader.tsx"],"names":[],"mappings":"AAyCiC","file":"custom-view-loader.tsx","sourcesContent":["import { useCallback, useEffect, useRef } from 'react';\nimport styled from '@emotion/styled';\nimport { useAllFeatureToggles } from '@flopflip/react-broadcast';\nimport { useIntl } from 'react-intl';\nimport { useShowNotification } from '@commercetools-frontend/actions-global';\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';\nimport {\n  CUSTOM_VIEWS_EVENTS_NAMES,\n  CUSTOM_VIEWS_EVENTS_META,\n  DOMAINS,\n  NOTIFICATION_KINDS_PAGE,\n  CustomViewData,\n} from '@commercetools-frontend/constants';\nimport { reportErrorToSentry } from '@commercetools-frontend/sentry';\nimport ModalPage from '../../modal-pages/internals/modal-page';\nimport messages from './messages';\n\ntype TCustomViewIframeMessage = {\n  source: string;\n  destination: string;\n  eventName: string;\n  eventData: Record<string, unknown>;\n};\n\ntype TCustomViewLoaderProps = {\n  customView: CustomViewData;\n  hostUrl?: string;\n  onClose: () => void;\n};\n\nconst isIframeReady = (iFrameElementRef: HTMLIFrameElement) => {\n  try {\n    return iFrameElementRef?.contentWindow?.document.readyState === 'complete';\n  } catch {\n    // Trying to access the contentWindow of a cross-origin iFrame will throw an error.\n    // We are not supposed to even get here because the iFrame must use\n    // a URL from our very same domain (the custom view is proxied through our http-proxy service).\n    return false;\n  }\n};\n\nconst ContentWrapper = styled.div`\n  height: 100%;\n`;\n\nconst CustomPanelIframe = styled.iframe`\n  height: 100%;\n  width: 100%;\n  border: none;\n`;\n\nfunction CustomViewLoader(props: TCustomViewLoaderProps) {\n  const iFrameElementRef = useRef<HTMLIFrameElement>(null);\n  const dataLocale = useApplicationContext((context) => context.dataLocale);\n  const projectKey = useApplicationContext((context) => context.project?.key);\n  const featureFlags = useAllFeatureToggles();\n  const iFrameCommunicationChannel = useRef(new MessageChannel());\n  const showNotification = useShowNotification();\n  const intl = useIntl();\n\n  const sendInitializationMessages = useCallback(() => {\n    // Transfer port2 to the iFrame so it can send messages back privately\n    iFrameElementRef.current?.contentWindow?.postMessage(\n      CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP,\n      window.location.href,\n      [iFrameCommunicationChannel.current.port2]\n    );\n\n    // Send the initialization message to the iFrame\n    iFrameCommunicationChannel.current.port1.postMessage({\n      source: CUSTOM_VIEWS_EVENTS_META.HOST_APPLICATION_CODE,\n      destination: `${CUSTOM_VIEWS_EVENTS_META.CUSTOM_VIEW_KEY_PREFIX}${props.customView.id}`,\n      eventName: CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_INITIALIZATION,\n      eventData: {\n        context: {\n          dataLocale,\n          projectKey,\n          featureFlags,\n          customViewConfig: props.customView,\n          hostUrl: props.hostUrl || window.location.href,\n        },\n      },\n    } as TCustomViewIframeMessage);\n  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);\n\n  const messageFromIFrameHandler = useCallback(\n    (event: MessageEvent) => {\n      if (event.data.origin === window.location.origin) {\n        switch (event.data.eventName) {\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:\n            props.onClose();\n            break;\n        }\n      }\n    },\n    [props]\n  );\n\n  // onLoad handler is called from the iFrame even where the URL is not valid\n  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready\n  const onLoadSuccessHandler = useCallback(() => {\n    // Show error and block if the iFrame is not ready\n    // (error loading it)\n    if (!isIframeReady(iFrameElementRef.current!)) {\n      showNotification({\n        domain: DOMAINS.PAGE,\n        kind: NOTIFICATION_KINDS_PAGE.error,\n        text: intl.formatMessage(messages.loadError),\n      });\n      return;\n    }\n\n    // Listen for messages from the iFrame\n    iFrameCommunicationChannel.current.port1.onmessage =\n      messageFromIFrameHandler;\n\n    // TODO: Locally (starter templates) we're seeing a little delay while rendering\n    // the iFrame. The CustomViewShell gets rendered, but the effect to start listening\n    // for messages is triggered after the iFrame is ready.\n    // This is a temporary fix to avoid the situation when running locally.\n    setTimeout(() => {\n      sendInitializationMessages();\n    }, 500);\n\n    // We want the effect to run only once so we don't need the dependencies array.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, []);\n\n  useEffect(() => {\n    // Close the channel when the component unmounts\n    const communicationChannel = iFrameCommunicationChannel.current;\n    return () => {\n      communicationChannel?.port1.close();\n    };\n  }, []);\n\n  // Currently we only support custom panels\n  if (props.customView.type !== 'CustomPanel') {\n    reportErrorToSentry(\n      new Error(\n        `CustomViewLoader: Provided Custom View has an unsupported type: ${props.customView.type}. Supported types: ['CustomPanel'].`\n      )\n    );\n    return null;\n  }\n  const panelSize = (props.customView.typeSettings?.size?.toLocaleLowerCase() ||\n    'large') as Lowercase<'SMALL' | 'LARGE'>;\n  const iFrameUrl = [\n    window.location.origin,\n    'custom-views',\n    props.customView.id,\n    'projects',\n    projectKey,\n  ].join('/');\n\n  return (\n    <ModalPage\n      isOpen\n      onClose={props.onClose}\n      size={panelSize === 'small' ? 10 : 30}\n      title={`Custom View: ${props.customView.defaultLabel}`}\n      hideTopBar\n    >\n      <ContentWrapper>\n        <CustomPanelIframe\n          id={`custom-view-${props.customView.id}`}\n          key={`custom-view-${props.customView.id}`}\n          title={`Custom View: ${props.customView.defaultLabel}`}\n          ref={iFrameElementRef}\n          src={iFrameUrl}\n          onLoad={onLoadSuccessHandler}\n        />\n      </ContentWrapper>\n    </ModalPage>\n  );\n}\n\nexport default CustomViewLoader;\n"]} */",
2397
+ styles: "height:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["custom-view-loader.tsx"],"names":[],"mappings":"AAyCiC","file":"custom-view-loader.tsx","sourcesContent":["import { useCallback, useEffect, useRef } from 'react';\nimport styled from '@emotion/styled';\nimport { useAllFeatureToggles } from '@flopflip/react-broadcast';\nimport { useIntl } from 'react-intl';\nimport { useShowNotification } from '@commercetools-frontend/actions-global';\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';\nimport {\n  CUSTOM_VIEWS_EVENTS_NAMES,\n  CUSTOM_VIEWS_EVENTS_META,\n  DOMAINS,\n  NOTIFICATION_KINDS_PAGE,\n  CustomViewData,\n} from '@commercetools-frontend/constants';\nimport { reportErrorToSentry } from '@commercetools-frontend/sentry';\nimport ModalPage from '../../modal-pages/internals/modal-page';\nimport messages from './messages';\n\ntype TCustomViewIframeMessage = {\n  source: string;\n  destination: string;\n  eventName: string;\n  eventData: Record<string, unknown>;\n};\n\ntype TCustomViewLoaderProps = {\n  customView: CustomViewData;\n  hostUrl?: string;\n  onClose: () => void;\n};\n\nconst isIframeReady = (iFrameElementRef: HTMLIFrameElement) => {\n  try {\n    return iFrameElementRef?.contentWindow?.document.readyState === 'complete';\n  } catch {\n    // Trying to access the contentWindow of a cross-origin iFrame will throw an error.\n    // We are not supposed to even get here because the iFrame must use\n    // a URL from our very same domain (the custom view is proxied through our http-proxy service).\n    return false;\n  }\n};\n\nconst ContentWrapper = styled.div`\n  height: 100%;\n`;\n\nconst CustomPanelIframe = styled.iframe`\n  height: 100%;\n  width: 100%;\n  border: none;\n`;\n\nfunction CustomViewLoader(props: TCustomViewLoaderProps) {\n  const iFrameElementRef = useRef<HTMLIFrameElement>(null);\n  const dataLocale = useApplicationContext((context) => context.dataLocale);\n  const projectKey = useApplicationContext((context) => context.project?.key);\n  const featureFlags = useAllFeatureToggles();\n  const iFrameCommunicationChannel = useRef(new MessageChannel());\n  const showNotification = useShowNotification();\n  const intl = useIntl();\n  const hasSentInitializationMessages = useRef(false);\n\n  const sendInitializationMessages = useCallback(() => {\n    // If we have already sent the initialization messages, do not send them again.\n    // The message can be sent either as a response to the CUSTOM_VIEW_READY message\n    // or as a result of a setTimeout after 500ms. In any case, this message should be sent only once.\n    if (hasSentInitializationMessages.current) {\n      return;\n    }\n\n    // Transfer port2 to the iFrame so it can send messages back privately\n    iFrameElementRef.current?.contentWindow?.postMessage(\n      CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP,\n      window.location.href,\n      [iFrameCommunicationChannel.current.port2]\n    );\n\n    // Send the initialization message to the iFrame\n    iFrameCommunicationChannel.current.port1.postMessage({\n      source: CUSTOM_VIEWS_EVENTS_META.HOST_APPLICATION_CODE,\n      destination: `${CUSTOM_VIEWS_EVENTS_META.CUSTOM_VIEW_KEY_PREFIX}${props.customView.id}`,\n      eventName: CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_INITIALIZATION,\n      eventData: {\n        context: {\n          dataLocale,\n          projectKey,\n          featureFlags,\n          customViewConfig: props.customView,\n          hostUrl: props.hostUrl || window.location.href,\n        },\n      },\n    } as TCustomViewIframeMessage);\n\n    hasSentInitializationMessages.current = true;\n  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);\n\n  const messageFromIFrameHandler = useCallback(\n    (event: MessageEvent) => {\n      if (event.data.origin === window.location.origin) {\n        switch (event.data.eventName) {\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:\n            props.onClose();\n            break;\n          // This message will only be sent by custom view shell older than v24.x\n          // For backwards compatibility we will send the initialization messages\n          // after 500ms if this message was not received by then.\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_READY: {\n            sendInitializationMessages();\n          }\n        }\n      }\n    },\n    [props, sendInitializationMessages]\n  );\n\n  // onLoad handler is called from the iFrame even where the URL is not valid\n  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready\n  const onLoadSuccessHandler = useCallback(() => {\n    // Show error and block if the iFrame is not ready\n    // (error loading it)\n    if (!isIframeReady(iFrameElementRef.current!)) {\n      showNotification({\n        domain: DOMAINS.PAGE,\n        kind: NOTIFICATION_KINDS_PAGE.error,\n        text: intl.formatMessage(messages.loadError),\n      });\n      return;\n    }\n\n    // This is for backwards compatibility with custom view shell older than v24.0.0\n    // where the custom-view-shell does not send the CUSTOM_VIEW_READY message yet.\n    setTimeout(() => {\n      sendInitializationMessages();\n    }, 500);\n\n    // We want the effect to run only once so we don't need the dependencies array.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, []);\n\n  useEffect(() => {\n    // Close the channel when the component unmounts\n    const communicationChannel = iFrameCommunicationChannel.current;\n\n    // Listen for messages from the iFrame\n    iFrameCommunicationChannel.current!.port1.onmessage =\n      messageFromIFrameHandler;\n    window.addEventListener('message', messageFromIFrameHandler);\n\n    return () => {\n      window.removeEventListener('message', messageFromIFrameHandler);\n      communicationChannel?.port1.close();\n    };\n  }, []);\n\n  // Currently we only support custom panels\n  if (props.customView.type !== 'CustomPanel') {\n    reportErrorToSentry(\n      new Error(\n        `CustomViewLoader: Provided Custom View has an unsupported type: ${props.customView.type}. Supported types: ['CustomPanel'].`\n      )\n    );\n    return null;\n  }\n  const panelSize = (props.customView.typeSettings?.size?.toLocaleLowerCase() ||\n    'large') as Lowercase<'SMALL' | 'LARGE'>;\n  const iFrameUrl = [\n    window.location.origin,\n    'custom-views',\n    props.customView.id,\n    'projects',\n    projectKey,\n  ].join('/');\n\n  return (\n    <ModalPage\n      isOpen\n      onClose={props.onClose}\n      size={panelSize === 'small' ? 10 : 30}\n      title={`Custom View: ${props.customView.defaultLabel}`}\n      hideTopBar\n    >\n      <ContentWrapper>\n        <CustomPanelIframe\n          id={`custom-view-${props.customView.id}`}\n          key={`custom-view-${props.customView.id}`}\n          title={`Custom View: ${props.customView.defaultLabel}`}\n          ref={iFrameElementRef}\n          src={iFrameUrl}\n          onLoad={onLoadSuccessHandler}\n        />\n      </ContentWrapper>\n    </ModalPage>\n  );\n}\n\nexport default CustomViewLoader;\n"]} */",
2398
2398
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__$2
2399
2399
  });
2400
2400
  const CustomPanelIframe = /*#__PURE__*/_styled("iframe", process.env.NODE_ENV === "production" ? {
@@ -2407,7 +2407,7 @@ const CustomPanelIframe = /*#__PURE__*/_styled("iframe", process.env.NODE_ENV ==
2407
2407
  styles: "height:100%;width:100%;border:none"
2408
2408
  } : {
2409
2409
  name: "174lt7a",
2410
- styles: "height:100%;width:100%;border:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["custom-view-loader.tsx"],"names":[],"mappings":"AA6CuC","file":"custom-view-loader.tsx","sourcesContent":["import { useCallback, useEffect, useRef } from 'react';\nimport styled from '@emotion/styled';\nimport { useAllFeatureToggles } from '@flopflip/react-broadcast';\nimport { useIntl } from 'react-intl';\nimport { useShowNotification } from '@commercetools-frontend/actions-global';\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';\nimport {\n  CUSTOM_VIEWS_EVENTS_NAMES,\n  CUSTOM_VIEWS_EVENTS_META,\n  DOMAINS,\n  NOTIFICATION_KINDS_PAGE,\n  CustomViewData,\n} from '@commercetools-frontend/constants';\nimport { reportErrorToSentry } from '@commercetools-frontend/sentry';\nimport ModalPage from '../../modal-pages/internals/modal-page';\nimport messages from './messages';\n\ntype TCustomViewIframeMessage = {\n  source: string;\n  destination: string;\n  eventName: string;\n  eventData: Record<string, unknown>;\n};\n\ntype TCustomViewLoaderProps = {\n  customView: CustomViewData;\n  hostUrl?: string;\n  onClose: () => void;\n};\n\nconst isIframeReady = (iFrameElementRef: HTMLIFrameElement) => {\n  try {\n    return iFrameElementRef?.contentWindow?.document.readyState === 'complete';\n  } catch {\n    // Trying to access the contentWindow of a cross-origin iFrame will throw an error.\n    // We are not supposed to even get here because the iFrame must use\n    // a URL from our very same domain (the custom view is proxied through our http-proxy service).\n    return false;\n  }\n};\n\nconst ContentWrapper = styled.div`\n  height: 100%;\n`;\n\nconst CustomPanelIframe = styled.iframe`\n  height: 100%;\n  width: 100%;\n  border: none;\n`;\n\nfunction CustomViewLoader(props: TCustomViewLoaderProps) {\n  const iFrameElementRef = useRef<HTMLIFrameElement>(null);\n  const dataLocale = useApplicationContext((context) => context.dataLocale);\n  const projectKey = useApplicationContext((context) => context.project?.key);\n  const featureFlags = useAllFeatureToggles();\n  const iFrameCommunicationChannel = useRef(new MessageChannel());\n  const showNotification = useShowNotification();\n  const intl = useIntl();\n\n  const sendInitializationMessages = useCallback(() => {\n    // Transfer port2 to the iFrame so it can send messages back privately\n    iFrameElementRef.current?.contentWindow?.postMessage(\n      CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP,\n      window.location.href,\n      [iFrameCommunicationChannel.current.port2]\n    );\n\n    // Send the initialization message to the iFrame\n    iFrameCommunicationChannel.current.port1.postMessage({\n      source: CUSTOM_VIEWS_EVENTS_META.HOST_APPLICATION_CODE,\n      destination: `${CUSTOM_VIEWS_EVENTS_META.CUSTOM_VIEW_KEY_PREFIX}${props.customView.id}`,\n      eventName: CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_INITIALIZATION,\n      eventData: {\n        context: {\n          dataLocale,\n          projectKey,\n          featureFlags,\n          customViewConfig: props.customView,\n          hostUrl: props.hostUrl || window.location.href,\n        },\n      },\n    } as TCustomViewIframeMessage);\n  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);\n\n  const messageFromIFrameHandler = useCallback(\n    (event: MessageEvent) => {\n      if (event.data.origin === window.location.origin) {\n        switch (event.data.eventName) {\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:\n            props.onClose();\n            break;\n        }\n      }\n    },\n    [props]\n  );\n\n  // onLoad handler is called from the iFrame even where the URL is not valid\n  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready\n  const onLoadSuccessHandler = useCallback(() => {\n    // Show error and block if the iFrame is not ready\n    // (error loading it)\n    if (!isIframeReady(iFrameElementRef.current!)) {\n      showNotification({\n        domain: DOMAINS.PAGE,\n        kind: NOTIFICATION_KINDS_PAGE.error,\n        text: intl.formatMessage(messages.loadError),\n      });\n      return;\n    }\n\n    // Listen for messages from the iFrame\n    iFrameCommunicationChannel.current.port1.onmessage =\n      messageFromIFrameHandler;\n\n    // TODO: Locally (starter templates) we're seeing a little delay while rendering\n    // the iFrame. The CustomViewShell gets rendered, but the effect to start listening\n    // for messages is triggered after the iFrame is ready.\n    // This is a temporary fix to avoid the situation when running locally.\n    setTimeout(() => {\n      sendInitializationMessages();\n    }, 500);\n\n    // We want the effect to run only once so we don't need the dependencies array.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, []);\n\n  useEffect(() => {\n    // Close the channel when the component unmounts\n    const communicationChannel = iFrameCommunicationChannel.current;\n    return () => {\n      communicationChannel?.port1.close();\n    };\n  }, []);\n\n  // Currently we only support custom panels\n  if (props.customView.type !== 'CustomPanel') {\n    reportErrorToSentry(\n      new Error(\n        `CustomViewLoader: Provided Custom View has an unsupported type: ${props.customView.type}. Supported types: ['CustomPanel'].`\n      )\n    );\n    return null;\n  }\n  const panelSize = (props.customView.typeSettings?.size?.toLocaleLowerCase() ||\n    'large') as Lowercase<'SMALL' | 'LARGE'>;\n  const iFrameUrl = [\n    window.location.origin,\n    'custom-views',\n    props.customView.id,\n    'projects',\n    projectKey,\n  ].join('/');\n\n  return (\n    <ModalPage\n      isOpen\n      onClose={props.onClose}\n      size={panelSize === 'small' ? 10 : 30}\n      title={`Custom View: ${props.customView.defaultLabel}`}\n      hideTopBar\n    >\n      <ContentWrapper>\n        <CustomPanelIframe\n          id={`custom-view-${props.customView.id}`}\n          key={`custom-view-${props.customView.id}`}\n          title={`Custom View: ${props.customView.defaultLabel}`}\n          ref={iFrameElementRef}\n          src={iFrameUrl}\n          onLoad={onLoadSuccessHandler}\n        />\n      </ContentWrapper>\n    </ModalPage>\n  );\n}\n\nexport default CustomViewLoader;\n"]} */",
2410
+ styles: "height:100%;width:100%;border:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["custom-view-loader.tsx"],"names":[],"mappings":"AA6CuC","file":"custom-view-loader.tsx","sourcesContent":["import { useCallback, useEffect, useRef } from 'react';\nimport styled from '@emotion/styled';\nimport { useAllFeatureToggles } from '@flopflip/react-broadcast';\nimport { useIntl } from 'react-intl';\nimport { useShowNotification } from '@commercetools-frontend/actions-global';\nimport { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';\nimport {\n  CUSTOM_VIEWS_EVENTS_NAMES,\n  CUSTOM_VIEWS_EVENTS_META,\n  DOMAINS,\n  NOTIFICATION_KINDS_PAGE,\n  CustomViewData,\n} from '@commercetools-frontend/constants';\nimport { reportErrorToSentry } from '@commercetools-frontend/sentry';\nimport ModalPage from '../../modal-pages/internals/modal-page';\nimport messages from './messages';\n\ntype TCustomViewIframeMessage = {\n  source: string;\n  destination: string;\n  eventName: string;\n  eventData: Record<string, unknown>;\n};\n\ntype TCustomViewLoaderProps = {\n  customView: CustomViewData;\n  hostUrl?: string;\n  onClose: () => void;\n};\n\nconst isIframeReady = (iFrameElementRef: HTMLIFrameElement) => {\n  try {\n    return iFrameElementRef?.contentWindow?.document.readyState === 'complete';\n  } catch {\n    // Trying to access the contentWindow of a cross-origin iFrame will throw an error.\n    // We are not supposed to even get here because the iFrame must use\n    // a URL from our very same domain (the custom view is proxied through our http-proxy service).\n    return false;\n  }\n};\n\nconst ContentWrapper = styled.div`\n  height: 100%;\n`;\n\nconst CustomPanelIframe = styled.iframe`\n  height: 100%;\n  width: 100%;\n  border: none;\n`;\n\nfunction CustomViewLoader(props: TCustomViewLoaderProps) {\n  const iFrameElementRef = useRef<HTMLIFrameElement>(null);\n  const dataLocale = useApplicationContext((context) => context.dataLocale);\n  const projectKey = useApplicationContext((context) => context.project?.key);\n  const featureFlags = useAllFeatureToggles();\n  const iFrameCommunicationChannel = useRef(new MessageChannel());\n  const showNotification = useShowNotification();\n  const intl = useIntl();\n  const hasSentInitializationMessages = useRef(false);\n\n  const sendInitializationMessages = useCallback(() => {\n    // If we have already sent the initialization messages, do not send them again.\n    // The message can be sent either as a response to the CUSTOM_VIEW_READY message\n    // or as a result of a setTimeout after 500ms. In any case, this message should be sent only once.\n    if (hasSentInitializationMessages.current) {\n      return;\n    }\n\n    // Transfer port2 to the iFrame so it can send messages back privately\n    iFrameElementRef.current?.contentWindow?.postMessage(\n      CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP,\n      window.location.href,\n      [iFrameCommunicationChannel.current.port2]\n    );\n\n    // Send the initialization message to the iFrame\n    iFrameCommunicationChannel.current.port1.postMessage({\n      source: CUSTOM_VIEWS_EVENTS_META.HOST_APPLICATION_CODE,\n      destination: `${CUSTOM_VIEWS_EVENTS_META.CUSTOM_VIEW_KEY_PREFIX}${props.customView.id}`,\n      eventName: CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_INITIALIZATION,\n      eventData: {\n        context: {\n          dataLocale,\n          projectKey,\n          featureFlags,\n          customViewConfig: props.customView,\n          hostUrl: props.hostUrl || window.location.href,\n        },\n      },\n    } as TCustomViewIframeMessage);\n\n    hasSentInitializationMessages.current = true;\n  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);\n\n  const messageFromIFrameHandler = useCallback(\n    (event: MessageEvent) => {\n      if (event.data.origin === window.location.origin) {\n        switch (event.data.eventName) {\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:\n            props.onClose();\n            break;\n          // This message will only be sent by custom view shell older than v24.x\n          // For backwards compatibility we will send the initialization messages\n          // after 500ms if this message was not received by then.\n          case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_READY: {\n            sendInitializationMessages();\n          }\n        }\n      }\n    },\n    [props, sendInitializationMessages]\n  );\n\n  // onLoad handler is called from the iFrame even where the URL is not valid\n  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready\n  const onLoadSuccessHandler = useCallback(() => {\n    // Show error and block if the iFrame is not ready\n    // (error loading it)\n    if (!isIframeReady(iFrameElementRef.current!)) {\n      showNotification({\n        domain: DOMAINS.PAGE,\n        kind: NOTIFICATION_KINDS_PAGE.error,\n        text: intl.formatMessage(messages.loadError),\n      });\n      return;\n    }\n\n    // This is for backwards compatibility with custom view shell older than v24.0.0\n    // where the custom-view-shell does not send the CUSTOM_VIEW_READY message yet.\n    setTimeout(() => {\n      sendInitializationMessages();\n    }, 500);\n\n    // We want the effect to run only once so we don't need the dependencies array.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, []);\n\n  useEffect(() => {\n    // Close the channel when the component unmounts\n    const communicationChannel = iFrameCommunicationChannel.current;\n\n    // Listen for messages from the iFrame\n    iFrameCommunicationChannel.current!.port1.onmessage =\n      messageFromIFrameHandler;\n    window.addEventListener('message', messageFromIFrameHandler);\n\n    return () => {\n      window.removeEventListener('message', messageFromIFrameHandler);\n      communicationChannel?.port1.close();\n    };\n  }, []);\n\n  // Currently we only support custom panels\n  if (props.customView.type !== 'CustomPanel') {\n    reportErrorToSentry(\n      new Error(\n        `CustomViewLoader: Provided Custom View has an unsupported type: ${props.customView.type}. Supported types: ['CustomPanel'].`\n      )\n    );\n    return null;\n  }\n  const panelSize = (props.customView.typeSettings?.size?.toLocaleLowerCase() ||\n    'large') as Lowercase<'SMALL' | 'LARGE'>;\n  const iFrameUrl = [\n    window.location.origin,\n    'custom-views',\n    props.customView.id,\n    'projects',\n    projectKey,\n  ].join('/');\n\n  return (\n    <ModalPage\n      isOpen\n      onClose={props.onClose}\n      size={panelSize === 'small' ? 10 : 30}\n      title={`Custom View: ${props.customView.defaultLabel}`}\n      hideTopBar\n    >\n      <ContentWrapper>\n        <CustomPanelIframe\n          id={`custom-view-${props.customView.id}`}\n          key={`custom-view-${props.customView.id}`}\n          title={`Custom View: ${props.customView.defaultLabel}`}\n          ref={iFrameElementRef}\n          src={iFrameUrl}\n          onLoad={onLoadSuccessHandler}\n        />\n      </ContentWrapper>\n    </ModalPage>\n  );\n}\n\nexport default CustomViewLoader;\n"]} */",
2411
2411
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__$2
2412
2412
  });
2413
2413
  function CustomViewLoader(props) {
@@ -2418,7 +2418,15 @@ function CustomViewLoader(props) {
2418
2418
  const iFrameCommunicationChannel = useRef(new MessageChannel());
2419
2419
  const showNotification = useShowNotification();
2420
2420
  const intl = useIntl();
2421
+ const hasSentInitializationMessages = useRef(false);
2421
2422
  const sendInitializationMessages = useCallback(() => {
2423
+ // If we have already sent the initialization messages, do not send them again.
2424
+ // The message can be sent either as a response to the CUSTOM_VIEW_READY message
2425
+ // or as a result of a setTimeout after 500ms. In any case, this message should be sent only once.
2426
+ if (hasSentInitializationMessages.current) {
2427
+ return;
2428
+ }
2429
+
2422
2430
  // Transfer port2 to the iFrame so it can send messages back privately
2423
2431
  iFrameElementRef.current?.contentWindow?.postMessage(CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_BOOTSTRAP, window.location.href, [iFrameCommunicationChannel.current.port2]);
2424
2432
 
@@ -2437,6 +2445,7 @@ function CustomViewLoader(props) {
2437
2445
  }
2438
2446
  }
2439
2447
  });
2448
+ hasSentInitializationMessages.current = true;
2440
2449
  }, [dataLocale, featureFlags, props.customView, props.hostUrl, projectKey]);
2441
2450
  const messageFromIFrameHandler = useCallback(event => {
2442
2451
  if (event.data.origin === window.location.origin) {
@@ -2444,9 +2453,16 @@ function CustomViewLoader(props) {
2444
2453
  case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_CLOSE:
2445
2454
  props.onClose();
2446
2455
  break;
2456
+ // This message will only be sent by custom view shell older than v24.x
2457
+ // For backwards compatibility we will send the initialization messages
2458
+ // after 500ms if this message was not received by then.
2459
+ case CUSTOM_VIEWS_EVENTS_NAMES.CUSTOM_VIEW_READY:
2460
+ {
2461
+ sendInitializationMessages();
2462
+ }
2447
2463
  }
2448
2464
  }
2449
- }, [props]);
2465
+ }, [props, sendInitializationMessages]);
2450
2466
 
2451
2467
  // onLoad handler is called from the iFrame even where the URL is not valid
2452
2468
  // (blocked by CORS, 404, etc.) so we need to make sure the iFrame is ready
@@ -2462,13 +2478,8 @@ function CustomViewLoader(props) {
2462
2478
  return;
2463
2479
  }
2464
2480
 
2465
- // Listen for messages from the iFrame
2466
- iFrameCommunicationChannel.current.port1.onmessage = messageFromIFrameHandler;
2467
-
2468
- // TODO: Locally (starter templates) we're seeing a little delay while rendering
2469
- // the iFrame. The CustomViewShell gets rendered, but the effect to start listening
2470
- // for messages is triggered after the iFrame is ready.
2471
- // This is a temporary fix to avoid the situation when running locally.
2481
+ // This is for backwards compatibility with custom view shell older than v24.0.0
2482
+ // where the custom-view-shell does not send the CUSTOM_VIEW_READY message yet.
2472
2483
  _setTimeout(() => {
2473
2484
  sendInitializationMessages();
2474
2485
  }, 500);
@@ -2479,7 +2490,12 @@ function CustomViewLoader(props) {
2479
2490
  useEffect(() => {
2480
2491
  // Close the channel when the component unmounts
2481
2492
  const communicationChannel = iFrameCommunicationChannel.current;
2493
+
2494
+ // Listen for messages from the iFrame
2495
+ iFrameCommunicationChannel.current.port1.onmessage = messageFromIFrameHandler;
2496
+ window.addEventListener('message', messageFromIFrameHandler);
2482
2497
  return () => {
2498
+ window.removeEventListener('message', messageFromIFrameHandler);
2483
2499
  communicationChannel?.port1.close();
2484
2500
  };
2485
2501
  }, []);
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@commercetools-frontend/application-components",
3
- "version": "24.1.0",
3
+ "version": "24.2.0",
4
4
  "description": "Generic components for building Merchant Center applications",
5
5
  "bugs": "https://github.com/commercetools/merchant-center-application-kit/issues",
6
6
  "repository": {
@@ -36,14 +36,14 @@
36
36
  "dependencies": {
37
37
  "@babel/runtime": "^7.22.15",
38
38
  "@babel/runtime-corejs3": "^7.22.15",
39
- "@commercetools-frontend/actions-global": "24.1.0",
40
- "@commercetools-frontend/application-config": "24.1.0",
41
- "@commercetools-frontend/application-shell-connectors": "24.1.0",
42
- "@commercetools-frontend/assets": "24.1.0",
43
- "@commercetools-frontend/constants": "24.1.0",
44
- "@commercetools-frontend/i18n": "24.1.0",
45
- "@commercetools-frontend/l10n": "24.1.0",
46
- "@commercetools-frontend/sentry": "24.1.0",
39
+ "@commercetools-frontend/actions-global": "24.2.0",
40
+ "@commercetools-frontend/application-config": "24.2.0",
41
+ "@commercetools-frontend/application-shell-connectors": "24.2.0",
42
+ "@commercetools-frontend/assets": "24.2.0",
43
+ "@commercetools-frontend/constants": "24.2.0",
44
+ "@commercetools-frontend/i18n": "24.2.0",
45
+ "@commercetools-frontend/l10n": "24.2.0",
46
+ "@commercetools-frontend/sentry": "24.2.0",
47
47
  "@commercetools-uikit/accessible-button": "20.0.0",
48
48
  "@commercetools-uikit/card": "20.0.0",
49
49
  "@commercetools-uikit/constraints": "20.0.0",
@@ -63,7 +63,7 @@
63
63
  "@commercetools-uikit/utils": "20.0.0",
64
64
  "@emotion/react": "^11.14.0",
65
65
  "@emotion/styled": "^11.14.0",
66
- "@flopflip/react-broadcast": "14.0.2",
66
+ "@flopflip/react-broadcast": "15.1.2",
67
67
  "@radix-ui/react-dialog": "1.1.4",
68
68
  "@react-hook/latest": "1.0.3",
69
69
  "@react-hook/resize-observer": "1.2.6",