
META-INF.resources.page_editor.app.components.Layout.js Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of com.liferay.layout.content.page.editor.web
Show all versions of com.liferay.layout.content.page.editor.web
Liferay Layout Content Page Editor Web
The newest version!
/**
* SPDX-FileCopyrightText: (c) 2000 Liferay, Inc. https://liferay.com
* SPDX-License-Identifier: LGPL-2.1-or-later OR LicenseRef-Liferay-DXP-EULA-2.0.0-2023-06
*/
import ClayAlert from '@clayui/alert';
import PropTypes from 'prop-types';
import React, {useEffect, useRef} from 'react';
import {
LayoutDataPropTypes,
getLayoutDataItemPropTypes,
} from '../../prop_types/index';
import {LAYOUT_DATA_ITEM_TYPES} from '../config/constants/layoutDataItemTypes';
import {config} from '../config/index';
import {useSelectItem} from '../contexts/ControlsContext';
import {useSelector} from '../contexts/StoreContext';
import {deepEqual} from '../utils/checkDeepEqual';
import useDropContainerId from '../utils/useDropContainerId';
import {FormStepWithControls} from './layout_data_items/FormStep';
import {FormStepContainerWithControls} from './layout_data_items/FormStepContainer';
import FragmentWithControls from './layout_data_items/FragmentWithControls';
import {
CollectionItemWithControls,
CollectionWithControls,
ColumnWithControls,
ContainerWithControls,
DropZoneWithControls,
FormWithControls,
Root,
RowWithControls,
} from './layout_data_items/index';
const LAYOUT_DATA_ITEMS = {
[LAYOUT_DATA_ITEM_TYPES.collection]: CollectionWithControls,
[LAYOUT_DATA_ITEM_TYPES.collectionItem]: CollectionItemWithControls,
[LAYOUT_DATA_ITEM_TYPES.column]: ColumnWithControls,
[LAYOUT_DATA_ITEM_TYPES.container]: ContainerWithControls,
[LAYOUT_DATA_ITEM_TYPES.dropZone]: DropZoneWithControls,
[LAYOUT_DATA_ITEM_TYPES.form]: FormWithControls,
[LAYOUT_DATA_ITEM_TYPES.formStep]: FormStepWithControls,
[LAYOUT_DATA_ITEM_TYPES.formStepContainer]: FormStepContainerWithControls,
[LAYOUT_DATA_ITEM_TYPES.fragment]: FragmentWithControls,
[LAYOUT_DATA_ITEM_TYPES.fragmentDropZone]: Root,
[LAYOUT_DATA_ITEM_TYPES.root]: Root,
[LAYOUT_DATA_ITEM_TYPES.row]: RowWithControls,
};
const Layout = React.memo(({mainItemId}) => {
const layoutData = useSelector((state) => state.layoutData);
const layoutRef = useRef(null);
const selectItem = useSelectItem();
const mainItem = layoutData.items[mainItemId];
const onClick = (event) => {
if (event.target === event.currentTarget) {
selectItem(null);
}
};
useEffect(() => {
const layout = layoutRef.current;
const preventLinkClick = (event) => {
const closestElement = event.target.closest('[href]');
if (
closestElement &&
!closestElement.dataset.lfrPageEditorHrefEnabled
) {
event.preventDefault();
}
};
if (layout) {
layout.addEventListener('click', preventLinkClick);
}
return () => {
if (layout) {
layout.removeEventListener('click', preventLinkClick);
}
};
}, [layoutRef]);
const hasWarningMessages =
config.isConversionDraft &&
config.layoutConversionWarningMessages &&
!!config.layoutConversionWarningMessages.length;
return (
<>
{config.isConversionDraft && (
{hasWarningMessages && (
{config.layoutConversionWarningMessages.map(
(message) => (
<>
{message}
>
)
)}
)}
)}
{mainItem && (
<>
>
)}
>
);
});
Layout.displayName = 'Layout';
Layout.propTypes = {
mainItemId: PropTypes.string.isRequired,
};
export default Layout;
class LayoutDataItem extends React.Component {
static getDerivedStateFromError(error) {
return {error};
}
static destructureItem(item, layoutData) {
return {
...item,
children: item.children.map((child) =>
LayoutDataItem.destructureItem(
layoutData.items[child],
layoutData
)
),
};
}
constructor(props) {
super(props);
this.state = {
error: null,
};
}
shouldComponentUpdate(nextProps, nextState) {
return (
nextState.error ||
!deepEqual(this.props.item, nextProps.item) ||
!deepEqual(
LayoutDataItem.destructureItem(
this.props.item,
this.props.layoutData
),
LayoutDataItem.destructureItem(
nextProps.item,
nextProps.layoutData
)
)
);
}
render() {
return this.state.error ? (
{Liferay.Language.get(
'an-unexpected-error-occurred-while-rendering-this-item'
)}
) : (
);
}
}
LayoutDataItem.propTypes = {
item: getLayoutDataItemPropTypes().isRequired,
layoutData: LayoutDataPropTypes.isRequired,
};
function LayoutDataItemContent({item, layoutData}) {
const Component = LAYOUT_DATA_ITEMS[item.type];
const componentRef = useRef(null);
return (
<>
{item.children.map((childId) => {
return (
);
})}
>
);
}
LayoutDataItemContent.propTypes = {
item: getLayoutDataItemPropTypes().isRequired,
layoutData: LayoutDataPropTypes.isRequired,
};
function LayoutClassManager({layoutRef}) {
const dropContainerId = useDropContainerId();
useEffect(() => {
if (dropContainerId) {
layoutRef.current?.classList.add('is-dragging');
}
else {
layoutRef.current?.classList.remove('is-dragging');
}
}, [dropContainerId, layoutRef]);
return null;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy