diff --git a/src/constants.js b/src/constants.js index bf4696d734..411d3f2486 100644 --- a/src/constants.js +++ b/src/constants.js @@ -78,5 +78,5 @@ export const REGEX_RULES = { }; export const IFRAME_FEATURE_POLICY = ( - 'microphone *; camera *; midi *; geolocation *; encrypted-media *, clipboard-write *' + 'microphone *; camera *; midi *; geolocation *; encrypted-media *; clipboard-write *' ); diff --git a/src/course-unit/CourseUnit.test.jsx b/src/course-unit/CourseUnit.test.jsx index 8b13e30a60..6d64615de0 100644 --- a/src/course-unit/CourseUnit.test.jsx +++ b/src/course-unit/CourseUnit.test.jsx @@ -1687,15 +1687,11 @@ describe('', () => { waitFor(() => { const iframe = getByTitle(xblockContainerIframeMessages.xblockIframeTitle.defaultMessage); - const currentXBlockId = courseVerticalChildrenMock.children[0].block_id; + const usageId = courseVerticalChildrenMock.children[0].block_id; expect(iframe).toBeInTheDocument(); - simulatePostMessageEvent(messageTypes.currentXBlockId, { - id: currentXBlockId, - }); - simulatePostMessageEvent(messageTypes.manageXBlockAccess, { - id: currentXBlockId, + usageId, }); }); @@ -1717,7 +1713,7 @@ describe('', () => { const iframe = getByTitle(xblockContainerIframeMessages.xblockIframeTitle.defaultMessage); expect(iframe).toBeInTheDocument(); simulatePostMessageEvent(messageTypes.manageXBlockAccess, { - id: courseVerticalChildrenMock.children[0].block_id, + usageId: courseVerticalChildrenMock.children[0].block_id, }); }); @@ -1752,7 +1748,7 @@ describe('', () => { const iframe = getByTitle(xblockContainerIframeMessages.xblockIframeTitle.defaultMessage); expect(iframe).toBeInTheDocument(); simulatePostMessageEvent(messageTypes.manageXBlockAccess, { - id: courseVerticalChildrenMock.children[0].block_id, + usageId: courseVerticalChildrenMock.children[0].block_id, }); }); @@ -1816,6 +1812,7 @@ describe('', () => { waitFor(() => { simulatePostMessageEvent(messageTypes.duplicateXBlock, {}); + simulatePostMessageEvent(messageTypes.newXBlockEditor, {}); expect(mockedUsedNavigate) .toHaveBeenCalledWith(`/course/${courseId}/editor/html/${targetBlockId}`, { replace: true }); }); diff --git a/src/course-unit/constants.js b/src/course-unit/constants.js index 5c147be8f4..b19c7ba378 100644 --- a/src/course-unit/constants.js +++ b/src/course-unit/constants.js @@ -61,7 +61,6 @@ export const messageTypes = { duplicateXBlock: 'duplicateXBlock', refreshXBlockPositions: 'refreshPositions', newXBlockEditor: 'newXBlockEditor', - currentXBlockId: 'currentXBlockId', toggleCourseXBlockDropdown: 'toggleCourseXBlockDropdown', }; @@ -75,3 +74,9 @@ export const COMPONENT_TYPES = { video: 'video', dragAndDrop: 'drag-and-drop-v2', }; + +export const COMPONENT_TYPES_WITH_NEW_EDITOR = { + html: 'html', + problem: 'problem', + video: 'video', +}; diff --git a/src/course-unit/xblock-container-iframe/hooks/index.ts b/src/course-unit/xblock-container-iframe/hooks/index.ts new file mode 100644 index 0000000000..c49993dc1e --- /dev/null +++ b/src/course-unit/xblock-container-iframe/hooks/index.ts @@ -0,0 +1,5 @@ +export { useIframeMessages } from './useIframeMessages'; +export { useIframeContent } from './useIframeContent'; +export { useMessageHandlers } from './useMessageHandlers'; +export { useIFrameBehavior } from './useIFrameBehavior'; +export { useLoadBearingHook } from './useLoadBearingHook'; diff --git a/src/course-unit/xblock-container-iframe/tests/hooks.test.tsx b/src/course-unit/xblock-container-iframe/hooks/tests/hooks.test.tsx similarity index 97% rename from src/course-unit/xblock-container-iframe/tests/hooks.test.tsx rename to src/course-unit/xblock-container-iframe/hooks/tests/hooks.test.tsx index 13b5467622..8883efb04d 100644 --- a/src/course-unit/xblock-container-iframe/tests/hooks.test.tsx +++ b/src/course-unit/xblock-container-iframe/hooks/tests/hooks.test.tsx @@ -3,8 +3,8 @@ import { renderHook, act } from '@testing-library/react-hooks'; import { useKeyedState } from '@edx/react-unit-test-utils'; import { logError } from '@edx/frontend-platform/logging'; -import { stateKeys, messageTypes } from '../../constants'; -import { useIFrameBehavior, useLoadBearingHook } from '../hooks'; +import { stateKeys, messageTypes } from '../../../constants'; +import { useLoadBearingHook, useIFrameBehavior } from '..'; jest.mock('@edx/react-unit-test-utils', () => ({ useKeyedState: jest.fn(), diff --git a/src/course-unit/xblock-container-iframe/hooks/types.ts b/src/course-unit/xblock-container-iframe/hooks/types.ts new file mode 100644 index 0000000000..3974656c49 --- /dev/null +++ b/src/course-unit/xblock-container-iframe/hooks/types.ts @@ -0,0 +1,25 @@ +export type UseMessageHandlersTypes = { + courseId: string; + navigate: (path: string) => void; + dispatch: (action: any) => void; + setIframeOffset: (height: number) => void; + handleDeleteXBlock: (usageId: string) => void; + handleRefetchXBlocks: () => void; + handleDuplicateXBlock: (blockType: string, usageId: string) => void; + handleManageXBlockAccess: (usageId: string) => void; +}; + +export type MessageHandlersTypes = Record void>; + +export interface UseIFrameBehaviorTypes { + id: string; + iframeUrl: string; + onLoaded?: boolean; +} + +export interface UseIFrameBehaviorReturnTypes { + iframeHeight: number; + handleIFrameLoad: () => void; + showError: boolean; + hasLoaded: boolean; +} diff --git a/src/course-unit/xblock-container-iframe/hooks.tsx b/src/course-unit/xblock-container-iframe/hooks/useIFrameBehavior.tsx similarity index 58% rename from src/course-unit/xblock-container-iframe/hooks.tsx rename to src/course-unit/xblock-container-iframe/hooks/useIFrameBehavior.tsx index 1a81e7852a..832ac94cd3 100644 --- a/src/course-unit/xblock-container-iframe/hooks.tsx +++ b/src/course-unit/xblock-container-iframe/hooks/useIFrameBehavior.tsx @@ -1,57 +1,12 @@ -import { - useState, useLayoutEffect, useCallback, useEffect, -} from 'react'; +import { useCallback, useEffect } from 'react'; import { logError } from '@edx/frontend-platform/logging'; // eslint-disable-next-line import/no-extraneous-dependencies import { useKeyedState } from '@edx/react-unit-test-utils'; -import { useEventListener } from '../../generic/hooks'; -import { stateKeys, messageTypes } from '../constants'; - -interface UseIFrameBehaviorParams { - id: string; - iframeUrl: string; - onLoaded?: boolean; -} - -interface UseIFrameBehaviorReturn { - iframeHeight: number; - handleIFrameLoad: () => void; - showError: boolean; - hasLoaded: boolean; -} - -/** - * We discovered an error in Firefox where - upon iframe load - React would cease to call any - * useEffect hooks until the user interacts with the page again. This is particularly confusing - * when navigating between sequences, as the UI partially updates leaving the user in a nebulous - * state. - * - * We were able to solve this error by using a layout effect to update some component state, which - * executes synchronously on render. Somehow this forces React to continue it's lifecycle - * immediately, rather than waiting for user interaction. This layout effect could be anywhere in - * the parent tree, as far as we can tell - we chose to add a conspicuously 'load bearing' (that's - * a joke) one here so it wouldn't be accidentally removed elsewhere. - * - * If we remove this hook when one of these happens: - * 1. React figures out that there's an issue here and fixes a bug. - * 2. We cease to use an iframe for unit rendering. - * 3. Firefox figures out that there's an issue in their iframe loading and fixes a bug. - * 4. We stop supporting Firefox. - * 5. An enterprising engineer decides to create a repo that reproduces the problem, submits it to - * Firefox/React for review, and they kindly help us figure out what in the world is happening - * so we can fix it. - * - * This hook depends on the unit id just to make sure it re-evaluates whenever the ID changes. If - * we change whether or not the Unit component is re-mounted when the unit ID changes, this may - * become important, as this hook will otherwise only evaluate the useLayoutEffect once. - */ -export const useLoadBearingHook = (id: string): void => { - const setValue = useState(0)[1]; - useLayoutEffect(() => { - setValue(currentValue => currentValue + 1); - }, [id]); -}; +import { useEventListener } from '../../../generic/hooks'; +import { stateKeys, messageTypes } from '../../constants'; +import { useLoadBearingHook } from './useLoadBearingHook'; +import { UseIFrameBehaviorTypes, UseIFrameBehaviorReturnTypes } from './types'; /** * Custom hook to manage iframe behavior. @@ -70,7 +25,7 @@ export const useIFrameBehavior = ({ id, iframeUrl, onLoaded = true, -}: UseIFrameBehaviorParams): UseIFrameBehaviorReturn => { +}: UseIFrameBehaviorTypes): UseIFrameBehaviorReturnTypes => { // Do not remove this hook. See function description. useLoadBearingHook(id); diff --git a/src/course-unit/xblock-container-iframe/hooks/useIframeContent.tsx b/src/course-unit/xblock-container-iframe/hooks/useIframeContent.tsx new file mode 100644 index 0000000000..abbc98b212 --- /dev/null +++ b/src/course-unit/xblock-container-iframe/hooks/useIframeContent.tsx @@ -0,0 +1,33 @@ +import { useEffect, useCallback, RefObject } from 'react'; + +import { messageTypes } from '../../constants'; + +/** + * Hook for managing iframe content and providing utilities to interact with the iframe. + * + * @param {React.RefObject} iframeRef - A React ref for the iframe element. + * @param {(ref: React.RefObject) => void} setIframeRef - + * A function to associate the iframeRef with the parent context. + * @param {(type: string, payload: any) => void} sendMessageToIframe - A function to send messages to the iframe. + * + * @returns {Object} - An object containing utility functions. + * @returns {() => void} return.refreshIframeContent - + * A function to refresh the iframe content by sending a specific message. + */ +export const useIframeContent = ( + iframeRef: RefObject, + setIframeRef: (ref: RefObject) => void, + sendMessageToIframe: (type: string, payload: any) => void, +): { refreshIframeContent: () => void } => { + useEffect(() => { + setIframeRef(iframeRef); + }, [setIframeRef, iframeRef]); + + // TODO: this artificial delay is a temporary solution + // to ensure the iframe content is properly refreshed. + const refreshIframeContent = useCallback(() => { + setTimeout(() => sendMessageToIframe(messageTypes.refreshXBlock, null), 1000); + }, [sendMessageToIframe]); + + return { refreshIframeContent }; +}; diff --git a/src/course-unit/xblock-container-iframe/hooks/useIframeMessages.tsx b/src/course-unit/xblock-container-iframe/hooks/useIframeMessages.tsx new file mode 100644 index 0000000000..6f192615da --- /dev/null +++ b/src/course-unit/xblock-container-iframe/hooks/useIframeMessages.tsx @@ -0,0 +1,20 @@ +import { useEffect } from 'react'; + +/** + * Hook for managing and handling messages received by the iframe. + * + * @param {Record void>} messageHandlers - + * A mapping of message types to their corresponding handler functions. + */ +export const useIframeMessages = (messageHandlers: Record void>) => { + useEffect(() => { + const handleMessage = (event: MessageEvent) => { + const { type, payload } = event.data || {}; + if (type in messageHandlers) { + messageHandlers[type](payload); + } + }; + window.addEventListener('message', handleMessage); + return () => window.removeEventListener('message', handleMessage); + }, [messageHandlers]); +}; diff --git a/src/course-unit/xblock-container-iframe/hooks/useLoadBearingHook.tsx b/src/course-unit/xblock-container-iframe/hooks/useLoadBearingHook.tsx new file mode 100644 index 0000000000..73a38b0223 --- /dev/null +++ b/src/course-unit/xblock-container-iframe/hooks/useLoadBearingHook.tsx @@ -0,0 +1,33 @@ +import { useLayoutEffect, useState } from 'react'; + +/** + * We discovered an error in Firefox where - upon iframe load - React would cease to call any + * useEffect hooks until the user interacts with the page again. This is particularly confusing + * when navigating between sequences, as the UI partially updates leaving the user in a nebulous + * state. + * + * We were able to solve this error by using a layout effect to update some component state, which + * executes synchronously on render. Somehow this forces React to continue it's lifecycle + * immediately, rather than waiting for user interaction. This layout effect could be anywhere in + * the parent tree, as far as we can tell - we chose to add a conspicuously 'load bearing' (that's + * a joke) one here so it wouldn't be accidentally removed elsewhere. + * + * If we remove this hook when one of these happens: + * 1. React figures out that there's an issue here and fixes a bug. + * 2. We cease to use an iframe for unit rendering. + * 3. Firefox figures out that there's an issue in their iframe loading and fixes a bug. + * 4. We stop supporting Firefox. + * 5. An enterprising engineer decides to create a repo that reproduces the problem, submits it to + * Firefox/React for review, and they kindly help us figure out what in the world is happening + * so we can fix it. + * + * This hook depends on the unit id just to make sure it re-evaluates whenever the ID changes. If + * we change whether or not the Unit component is re-mounted when the unit ID changes, this may + * become important, as this hook will otherwise only evaluate the useLayoutEffect once. + */ +export const useLoadBearingHook = (id: string): void => { + const setValue = useState(0)[1]; + useLayoutEffect(() => { + setValue(currentValue => currentValue + 1); + }, [id]); +}; diff --git a/src/course-unit/xblock-container-iframe/hooks/useMessageHandlers.tsx b/src/course-unit/xblock-container-iframe/hooks/useMessageHandlers.tsx new file mode 100644 index 0000000000..974f7bf0c6 --- /dev/null +++ b/src/course-unit/xblock-container-iframe/hooks/useMessageHandlers.tsx @@ -0,0 +1,38 @@ +import { useMemo } from 'react'; + +import { copyToClipboard } from '../../../generic/data/thunks'; +import { messageTypes } from '../../constants'; +import { MessageHandlersTypes, UseMessageHandlersTypes } from './types'; + +/** + * Hook for creating message handlers used to handle iframe messages. + * + * @param params - The parameters required to create message handlers. + * @returns {MessageHandlersTypes} - An object mapping message types to their handler functions. + */ +export const useMessageHandlers = ({ + courseId, + navigate, + dispatch, + setIframeOffset, + handleDeleteXBlock, + handleRefetchXBlocks, + handleDuplicateXBlock, + handleManageXBlockAccess, +}: UseMessageHandlersTypes): MessageHandlersTypes => useMemo(() => ({ + [messageTypes.copyXBlock]: ({ usageId }) => dispatch(copyToClipboard(usageId)), + [messageTypes.deleteXBlock]: ({ usageId }) => handleDeleteXBlock(usageId), + [messageTypes.newXBlockEditor]: ({ blockType, usageId }) => navigate(`/course/${courseId}/editor/${blockType}/${usageId}`), + [messageTypes.duplicateXBlock]: ({ blockType, usageId }) => handleDuplicateXBlock(blockType, usageId), + [messageTypes.manageXBlockAccess]: ({ usageId }) => handleManageXBlockAccess(usageId), + [messageTypes.refreshXBlockPositions]: handleRefetchXBlocks, + [messageTypes.toggleCourseXBlockDropdown]: ({ + courseXBlockDropdownHeight, + }: { courseXBlockDropdownHeight: number }) => setIframeOffset(courseXBlockDropdownHeight), +}), [ + courseId, + handleDeleteXBlock, + handleRefetchXBlocks, + handleDuplicateXBlock, + handleManageXBlockAccess, +]); diff --git a/src/course-unit/xblock-container-iframe/index.tsx b/src/course-unit/xblock-container-iframe/index.tsx index b26ae8e0ed..3b2dca75bb 100644 --- a/src/course-unit/xblock-container-iframe/index.tsx +++ b/src/course-unit/xblock-container-iframe/index.tsx @@ -2,25 +2,28 @@ import { useRef, FC, useEffect, useState, useMemo, useCallback, } from 'react'; import { useIntl } from '@edx/frontend-platform/i18n'; -import { getConfig } from '@edx/frontend-platform'; import { useToggle } from '@openedx/paragon'; import { useDispatch } from 'react-redux'; import { useNavigate } from 'react-router-dom'; import DeleteModal from '../../generic/delete-modal/DeleteModal'; import ConfigureModal from '../../generic/configure-modal/ConfigureModal'; -import { copyToClipboard } from '../../generic/data/thunks'; -import { COURSE_BLOCK_NAMES, IFRAME_FEATURE_POLICY } from '../../constants'; -import { messageTypes, COMPONENT_TYPES } from '../constants'; +import { IFRAME_FEATURE_POLICY } from '../../constants'; +import { COMPONENT_TYPES_WITH_NEW_EDITOR } from '../constants'; import { fetchCourseUnitQuery } from '../data/thunk'; import { useIframe } from '../context/hooks'; -import { useIFrameBehavior } from './hooks'; +import { + useMessageHandlers, + useIframeContent, + useIframeMessages, + useIFrameBehavior, +} from './hooks'; +import { formatAccessManagedXBlockData, getIframeUrl } from './utils'; import messages from './messages'; import { XBlockContainerIframeProps, - XBlockDataTypes, - MessagePayloadTypes, + AccessManagedXBlockDataTypes, } from './types'; const XBlockContainerIframe: FC = ({ @@ -33,133 +36,100 @@ const XBlockContainerIframe: FC = ({ const [isDeleteModalOpen, openDeleteModal, closeDeleteModal] = useToggle(false); const [isConfigureModalOpen, openConfigureModal, closeConfigureModal] = useToggle(false); - const { setIframeRef, sendMessageToIframe } = useIframe(); - const [currentXBlockId, setCurrentXBlockId] = useState(null); - const [currentXBlockData, setCurrentXBlockData] = useState({}); - const [courseXBlockIframeOffset, setCourseXBlockIframeOffset] = useState(0); + const [accessManagedXBlockData, setAccessManagedXBlockData] = useState({}); + const [iframeOffset, setIframeOffset] = useState(0); + const [deleteXBlockId, setDeleteXBlockId] = useState(null); + const [configureXBlockId, setConfigureXBlockId] = useState(null); - const iframeUrl = useMemo(() => `${getConfig().STUDIO_BASE_URL}/container_embed/${blockId}`, [blockId]); + const iframeUrl = useMemo(() => getIframeUrl(blockId), [blockId]); + + const { setIframeRef, sendMessageToIframe } = useIframe(); + const { iframeHeight } = useIFrameBehavior({ id: blockId, iframeUrl }); + const { refreshIframeContent } = useIframeContent(iframeRef, setIframeRef, sendMessageToIframe); useEffect(() => { setIframeRef(iframeRef); }, [setIframeRef]); - useEffect(() => { - if (currentXBlockId) { - const foundXBlockInfo = courseVerticalChildren?.find(xblock => xblock.blockId === currentXBlockId); - if (foundXBlockInfo) { - const { name, userPartitionInfo, blockType } = foundXBlockInfo; - - setCurrentXBlockData({ - category: COURSE_BLOCK_NAMES.component.id, - displayName: name, - userPartitionInfo, - showCorrectness: 'always', - blockType, - id: currentXBlockId, - }); - } - } - }, [isConfigureModalOpen, currentXBlockId, courseVerticalChildren]); - - const handleRefreshIframe = useCallback(() => { - // TODO: this artificial delay is a temporary solution - // to ensure the iframe content is properly refreshed. - setTimeout(() => { - sendMessageToIframe(messageTypes.refreshXBlock, null); - }, 1000); - }, [sendMessageToIframe]); - const handleDuplicateXBlock = useCallback( - (xblockData: XBlockDataTypes) => { - const duplicateAndNavigate = (blockType: string) => { - unitXBlockActions.handleDuplicate(xblockData.id); - if ([COMPONENT_TYPES.html, COMPONENT_TYPES.problem, COMPONENT_TYPES.video].includes(blockType)) { - navigate(`/course/${courseId}/editor/${blockType}/${xblockData.id}`); - } - handleRefreshIframe(); - }; - - duplicateAndNavigate(xblockData.blockType); + (blockType: string, usageId: string) => { + unitXBlockActions.handleDuplicate(usageId); + if (COMPONENT_TYPES_WITH_NEW_EDITOR[blockType]) { + navigate(`/course/${courseId}/editor/${blockType}/${usageId}`); + } + refreshIframeContent(); }, - [unitXBlockActions, courseId, navigate, handleRefreshIframe], + [unitXBlockActions, courseId, navigate, refreshIframeContent], ); - const handleRefetchXBlocks = useCallback(() => { - // TODO: this artificial delay is a temporary solution - // to ensure the iframe content is properly refreshed. - setTimeout(() => { - dispatch(fetchCourseUnitQuery(blockId)); - }, 1000); - }, [dispatch, blockId]); - - useEffect(() => { - const messageHandlers: Record void> = { - [messageTypes.deleteXBlock]: openDeleteModal, - [messageTypes.manageXBlockAccess]: () => openConfigureModal(), - [messageTypes.copyXBlock]: () => dispatch(copyToClipboard(currentXBlockId)), - [messageTypes.duplicateXBlock]: () => handleDuplicateXBlock(currentXBlockData), - [messageTypes.refreshXBlockPositions]: handleRefetchXBlocks, - [messageTypes.newXBlockEditor]: ({ url }) => navigate(`/course/${courseId}/editor${url}`), - [messageTypes.currentXBlockId]: ({ id }) => setCurrentXBlockId(id), - [messageTypes.toggleCourseXBlockDropdown]: ({ - courseXBlockDropdownHeight, - }) => setCourseXBlockIframeOffset(courseXBlockDropdownHeight), - }; - - const handleMessage = (event: MessageEvent) => { - const { type, payload } = event.data || {}; - - if (type && messageHandlers[type]) { - messageHandlers[type](payload); - } - }; - - window.addEventListener('message', handleMessage); + const handleDeleteXBlock = (usageId: string) => { + setDeleteXBlockId(usageId); + openDeleteModal(); + }; - return () => { - window.removeEventListener('message', handleMessage); - }; - }, [dispatch, blockId, courseVerticalChildren, currentXBlockId, currentXBlockData]); + const handleManageXBlockAccess = (usageId: string) => { + openConfigureModal(); + setConfigureXBlockId(usageId); + const foundXBlock = courseVerticalChildren?.find(xblock => xblock.blockId === usageId); + if (foundXBlock) { + setAccessManagedXBlockData(formatAccessManagedXBlockData(foundXBlock, usageId)); + } + }; - const { iframeHeight } = useIFrameBehavior({ - id: blockId, - iframeUrl, - }); + const handleRefetchXBlocks = useCallback(() => { + setTimeout(() => dispatch(fetchCourseUnitQuery(blockId)), 1000); + }, [dispatch, blockId]); - const handleDeleteItemSubmit = () => { - if (currentXBlockId) { - unitXBlockActions.handleDelete(currentXBlockId); + const onDeleteSubmit = () => { + if (deleteXBlockId) { + unitXBlockActions.handleDelete(deleteXBlockId); closeDeleteModal(); - handleRefreshIframe(); + refreshIframeContent(); } }; - const onConfigureSubmit = (...args: any[]) => { - if (currentXBlockId) { - handleConfigureSubmit(currentXBlockId, ...args, closeConfigureModal); - handleRefreshIframe(); + const onManageXBlockAccessSubmit = (...args: any[]) => { + if (configureXBlockId) { + handleConfigureSubmit(configureXBlockId, ...args, closeConfigureModal); + setAccessManagedXBlockData({}); + refreshIframeContent(); } }; + const messageHandlers = useMessageHandlers({ + courseId, + navigate, + dispatch, + setIframeOffset, + handleDeleteXBlock, + handleRefetchXBlocks, + handleDuplicateXBlock, + handleManageXBlockAccess, + }); + + useIframeMessages(messageHandlers); + return ( <> - {currentXBlockId && ( + {Object.keys(accessManagedXBlockData).length ? ( { + closeConfigureModal(); + setAccessManagedXBlockData({}); + }} + onConfigureSubmit={onManageXBlockAccessSubmit} + currentItemData={accessManagedXBlockData as AccessManagedXBlockDataTypes} isSelfPaced={false} /> - )} + ) : null}