From f09c999c4b11d3ed3c6390c53aa3301242a0174d Mon Sep 17 00:00:00 2001 From: Lukasz Ostafin Date: Fri, 13 Dec 2024 14:39:51 +0100 Subject: [PATCH 1/3] IBX-9322: ezobjectrelationlist field allows selecting the same content multiple times --- .../js/scripts/fieldType/ezobjectrelationlist.js | 13 ++++--------- 1 file changed, 4 insertions(+), 9 deletions(-) diff --git a/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js b/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js index 82f94ec72d..f258d10d03 100644 --- a/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js +++ b/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js @@ -113,12 +113,12 @@ sourceInput.dispatchEvent(new CustomEvent(EVENT_CUSTOM)); }; const onConfirm = (items) => { - items = excludeDuplicatedItems(items); + const itemsWithoutDuplicate = excludeDuplicatedItems(items); - renderRows(items); + renderRows(itemsWithoutDuplicate); attachRowsEventHandlers(); - selectedItems = [...selectedItems, ...items.map((item) => item.ContentInfo.Content._id)]; + selectedItems = [...selectedItems, ...itemsWithoutDuplicate.map((item) => item.ContentInfo.Content._id)]; updateInputValue(selectedItems); closeUDW(); @@ -157,11 +157,7 @@ }), ); }; - const excludeDuplicatedItems = (items) => { - selectedItemsMap = items.reduce((total, item) => ({ ...total, [item.ContentInfo.Content._id]: item }), selectedItemsMap); - - return items.filter((item) => selectedItemsMap[item.ContentInfo.Content._id]); - }; + const excludeDuplicatedItems = (items) => items.filter((item) => !selectedItems.includes(item.ContentInfo.Content._id)); const renderRow = (item, index) => { const { escapeHTML } = ibexa.helpers.text; const { formatShortDateTime } = ibexa.helpers.timezone; @@ -304,7 +300,6 @@ updateInputValue(selectedItems); }; let selectedItems = [...fieldContainer.querySelectorAll(SELECTOR_ROW)].map((row) => parseInt(row.dataset.contentId, 10)); - let selectedItemsMap = selectedItems.reduce((total, item) => ({ ...total, [item]: item }), {}); updateAddBtnState(); attachRowsEventHandlers(); From 0287d33c0beac610ed5ada6e468ed25a7d59bbef Mon Sep 17 00:00:00 2001 From: Lukasz Ostafin Date: Tue, 7 Jan 2025 15:56:32 +0100 Subject: [PATCH 2/3] After QA --- .../js/scripts/fieldType/ezobjectrelationlist.js | 10 +++++++++- .../admin/ui/field_type/edit/relation_base.html.twig | 10 ++++++++-- 2 files changed, 17 insertions(+), 3 deletions(-) diff --git a/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js b/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js index f258d10d03..c8bab7a840 100644 --- a/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js +++ b/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js @@ -87,6 +87,7 @@ const itemNodeNameCell = itemNode.querySelector('.ibexa-relations__item-name'); itemNode.dataset.contentId = contentId; + itemNode.dataset.locationId = locationId; itemNode.querySelector('.ibexa-relations__table-action--remove-item').addEventListener('click', removeItem, false); itemNodeNameCell.dataset.ibexaUpdateContentId = contentId; @@ -128,9 +129,15 @@ }; const openUDW = (event) => { event.preventDefault(); - + const selectedItemsRow = fieldContainer.querySelectorAll(SELECTOR_ROW); const config = JSON.parse(event.currentTarget.dataset.udwConfig); const limit = parseInt(event.currentTarget.dataset.limit, 10); + const selectedLocations = [...selectedItemsRow].reduce((locationsIds, selectedItemRow) => { + const { locationId } = selectedItemRow.dataset; + const parsedLocationId = parseInt(locationId, 10); + + return isNaN(parsedLocationId) ? locationsIds : [...locationsIds, parsedLocationId]; + }, []); const title = limit === 1 ? Translator.trans( @@ -151,6 +158,7 @@ onCancel: closeUDW, title, startingLocationId, + selectedLocations, ...config, multiple: isSingle ? false : selectedItemsLimit !== 1, multipleItemsLimit: selectedItemsLimit > 1 ? selectedItemsLimit - selectedItems.length : selectedItemsLimit, diff --git a/src/bundle/Resources/views/themes/admin/ui/field_type/edit/relation_base.html.twig b/src/bundle/Resources/views/themes/admin/ui/field_type/edit/relation_base.html.twig index d25c88d7b2..ba127be78a 100644 --- a/src/bundle/Resources/views/themes/admin/ui/field_type/edit/relation_base.html.twig +++ b/src/bundle/Resources/views/themes/admin/ui/field_type/edit/relation_base.html.twig @@ -170,7 +170,10 @@ {% set body_rows = body_rows|merge([{ cols: body_row_cols, class: 'ibexa-relations__item', - attr: { 'data-content-id': relation.contentInfo.id }, + attr: { + 'data-content-id': relation.contentInfo.id, + 'data-location-id': relation.contentInfo.mainLocationId, + }, }]) %} {% elseif relation.unauthorized %} {% set col_raw_unauthorised %} @@ -194,7 +197,10 @@ { content: col_raw_order_input, raw: true, attr: { class: 'ibexa-relations__order-wrapper' } } ], class: 'ibexa-relations__item', - attr: { 'data-content-id': relation.contentId }, + attr: { + 'data-content-id': relation.contentId, + 'data-location-id': relation.contentInfo.mainLocationId, + }, }]) %} {% endif %} {% endfor %} From d740367b7866242207b8e03321b8a9b052cdddf8 Mon Sep 17 00:00:00 2001 From: Lukasz Ostafin Date: Wed, 8 Jan 2025 13:01:21 +0100 Subject: [PATCH 3/3] After QA --- .../scripts/fieldType/ezobjectrelationlist.js | 1 + .../_selected.locations.item.scss | 6 ++ .../components/finder/finder.leaf.js | 12 ++- .../components/grid-view/grid.view.item.js | 10 +- .../selected.locations.item.js | 4 + .../tree.item.toggle.selection.js | 20 +++- .../hooks/useSelectedLocationsHelpers.js | 9 ++ .../universal.discovery.module.js | 94 +++++++++++-------- 8 files changed, 108 insertions(+), 48 deletions(-) diff --git a/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js b/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js index c8bab7a840..ff4b349a52 100644 --- a/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js +++ b/src/bundle/Resources/public/js/scripts/fieldType/ezobjectrelationlist.js @@ -159,6 +159,7 @@ title, startingLocationId, selectedLocations, + isInitLocationsDeselectionBlocked: true, ...config, multiple: isSingle ? false : selectedItemsLimit !== 1, multipleItemsLimit: selectedItemsLimit > 1 ? selectedItemsLimit - selectedItems.length : selectedItemsLimit, diff --git a/src/bundle/Resources/public/scss/ui/modules/universal-discovery/_selected.locations.item.scss b/src/bundle/Resources/public/scss/ui/modules/universal-discovery/_selected.locations.item.scss index e117b7fd2f..60fbd5a185 100644 --- a/src/bundle/Resources/public/scss/ui/modules/universal-discovery/_selected.locations.item.scss +++ b/src/bundle/Resources/public/scss/ui/modules/universal-discovery/_selected.locations.item.scss @@ -52,5 +52,11 @@ .ibexa-icon { fill: $ibexa-color-dark; } + + &:disabled { + .ibexa-icon { + fill: $ibexa-color-dark-300; + } + } } } diff --git a/src/bundle/ui-dev/src/modules/universal-discovery/components/finder/finder.leaf.js b/src/bundle/ui-dev/src/modules/universal-discovery/components/finder/finder.leaf.js index 08f1530919..ae4dce3916 100644 --- a/src/bundle/ui-dev/src/modules/universal-discovery/components/finder/finder.leaf.js +++ b/src/bundle/ui-dev/src/modules/universal-discovery/components/finder/finder.leaf.js @@ -24,10 +24,11 @@ const FinderLeaf = ({ location }) => { const contentTypesMap = useContext(ContentTypesMapContext); const [, dispatchSelectedLocationsAction] = useContext(SelectedLocationsContext); const [multiple] = useContext(MultipleConfigContext); - const { checkIsSelectable, checkIsSelected, checkIsSelectionBlocked } = useSelectedLocationsHelpers(); + const { checkIsSelectable, checkIsSelected, checkIsSelectionBlocked, checkIsDeselectionBlocked } = useSelectedLocationsHelpers(); const isSelected = checkIsSelected(location); const isNotSelectable = !checkIsSelectable(location); const isSelectionBlocked = checkIsSelectionBlocked(location); + const isDeselectionBlocked = checkIsDeselectionBlocked(location); const markLocation = ({ nativeEvent }) => { const isSelectionButtonClicked = nativeEvent.target.closest('.c-udw-toggle-selection'); const isMarkedLocationClicked = location.id === markedLocationId; @@ -49,7 +50,14 @@ const FinderLeaf = ({ location }) => { } }; const renderToggleSelection = () => { - return ; + return ( + + ); }; const className = createCssClassNames({ 'c-finder-leaf': true, diff --git a/src/bundle/ui-dev/src/modules/universal-discovery/components/grid-view/grid.view.item.js b/src/bundle/ui-dev/src/modules/universal-discovery/components/grid-view/grid.view.item.js index 45abf7930e..9c7611282e 100644 --- a/src/bundle/ui-dev/src/modules/universal-discovery/components/grid-view/grid.view.item.js +++ b/src/bundle/ui-dev/src/modules/universal-discovery/components/grid-view/grid.view.item.js @@ -30,10 +30,11 @@ const GridViewItem = ({ location, version }) => { const containersOnly = useContext(ContainersOnlyContext); const contentTypeInfo = contentTypesMap[location.ContentInfo.Content.ContentType._href]; const { isContainer } = contentTypeInfo; - const { checkIsSelectable, checkIsSelected, checkIsSelectionBlocked } = useSelectedLocationsHelpers(); + const { checkIsSelectable, checkIsSelected, checkIsSelectionBlocked, checkIsDeselectionBlocked } = useSelectedLocationsHelpers(); const isSelected = checkIsSelected(location); const isNotSelectable = !checkIsSelectable(location); const isSelectionBlocked = checkIsSelectionBlocked(location); + const isDeselectionBlocked = checkIsDeselectionBlocked(location); const className = createCssClassNames({ 'ibexa-grid-view-item': true, 'ibexa-grid-view-item--marked': markedLocationId === location.id, @@ -68,7 +69,12 @@ const GridViewItem = ({ location, version }) => { const renderToggleSelection = () => { return (
- +
); }; diff --git a/src/bundle/ui-dev/src/modules/universal-discovery/components/selected-locations/selected.locations.item.js b/src/bundle/ui-dev/src/modules/universal-discovery/components/selected-locations/selected.locations.item.js index 010f4b082d..b458a0ad03 100644 --- a/src/bundle/ui-dev/src/modules/universal-discovery/components/selected-locations/selected.locations.item.js +++ b/src/bundle/ui-dev/src/modules/universal-discovery/components/selected-locations/selected.locations.item.js @@ -11,12 +11,15 @@ import Thumbnail from '../../../common/thumbnail/thumbnail'; import { SelectedLocationsContext, ContentTypesMapContext } from '../../universal.discovery.module'; import { getAdminUiConfig, getTranslator } from '@ibexa-admin-ui/src/bundle/Resources/public/js/scripts/helpers/context.helper'; +import { useSelectedLocationsHelpers } from '../../hooks/useSelectedLocationsHelpers'; const SelectedLocationsItem = ({ location, permissions }) => { const adminUiConfig = getAdminUiConfig(); const Translator = getTranslator(); const refSelectedLocationsItem = useRef(null); const [, dispatchSelectedLocationsAction] = useContext(SelectedLocationsContext); + const { checkIsDeselectionBlocked } = useSelectedLocationsHelpers(); + const isDeselectionBlocked = checkIsDeselectionBlocked(location); const contentTypesMap = useContext(ContentTypesMapContext); const clearLabel = Translator.trans( /*@Desc("Clear selection")*/ 'selected_locations.clear_selection', @@ -65,6 +68,7 @@ const SelectedLocationsItem = ({ location, permissions }) => { onClick={removeFromSelection} title={clearLabel} data-tooltip-container-selector=".c-udw-tab" + disabled={isDeselectionBlocked} > diff --git a/src/bundle/ui-dev/src/modules/universal-discovery/components/tree-item-toggle-selection/tree.item.toggle.selection.js b/src/bundle/ui-dev/src/modules/universal-discovery/components/tree-item-toggle-selection/tree.item.toggle.selection.js index 04b21ffe80..ab28da4e5b 100644 --- a/src/bundle/ui-dev/src/modules/universal-discovery/components/tree-item-toggle-selection/tree.item.toggle.selection.js +++ b/src/bundle/ui-dev/src/modules/universal-discovery/components/tree-item-toggle-selection/tree.item.toggle.selection.js @@ -5,6 +5,7 @@ import { parse as parseTooltip } from '@ibexa-admin-ui/src/bundle/Resources/publ import { UDWContext, + SelectionConfigContext, SelectedLocationsContext, RestInfoContext, MultipleConfigContext, @@ -23,19 +24,23 @@ const TreeItemToggleSelection = ({ locationId, isContainer, contentTypeIdentifie parseTooltip(document.querySelector('.c-list')); }, []); - if (!isUDW) { - return null; - } - + const { isInitLocationsDeselectionBlocked, initSelectedLocations } = useContext(SelectionConfigContext); const [selectedLocations, dispatchSelectedLocationsAction] = useContext(SelectedLocationsContext); const [multiple, multipleItemsLimit] = useContext(MultipleConfigContext); const containersOnly = useContext(ContainersOnlyContext); const allowedContentTypes = useContext(AllowedContentTypesContext); const restInfo = useContext(RestInfoContext); + + if (!isUDW) { + return null; + } + const isSelected = selectedLocations.some((selectedLocation) => selectedLocation.location.id === locationId); const isNotSelectable = (containersOnly && !isContainer) || (allowedContentTypes && !allowedContentTypes.includes(contentTypeIdentifier)); const isSelectionBlocked = multipleItemsLimit !== 0 && !isSelected && selectedLocations.length >= multipleItemsLimit; + const isInitSelectedLocation = initSelectedLocations.includes(locationId); + const isDeselectionBlocked = isSelected && isInitSelectedLocation && isInitLocationsDeselectionBlocked; const location = { id: locationId, }; @@ -51,7 +56,12 @@ const TreeItemToggleSelection = ({ locationId, isContainer, contentTypeIdentifie return ( - + {isNotSelectable &&
} ); diff --git a/src/bundle/ui-dev/src/modules/universal-discovery/hooks/useSelectedLocationsHelpers.js b/src/bundle/ui-dev/src/modules/universal-discovery/hooks/useSelectedLocationsHelpers.js index b8400a849c..bd08d0567d 100644 --- a/src/bundle/ui-dev/src/modules/universal-discovery/hooks/useSelectedLocationsHelpers.js +++ b/src/bundle/ui-dev/src/modules/universal-discovery/hooks/useSelectedLocationsHelpers.js @@ -7,12 +7,14 @@ import { ContentTypesMapContext, MultipleConfigContext, SelectedLocationsContext, + SelectionConfigContext, } from '../universal.discovery.module'; export const useSelectedLocationsHelpers = () => { const [, multipleItemsLimit] = useContext(MultipleConfigContext); const contentTypesMap = useContext(ContentTypesMapContext); const [selectedLocations] = useContext(SelectedLocationsContext); + const { isInitLocationsDeselectionBlocked, initSelectedLocations } = useContext(SelectionConfigContext); const containersOnly = useContext(ContainersOnlyContext); const allowedContentTypes = useContext(AllowedContentTypesContext); const checkIsSelectableWrapped = useCallback( @@ -24,10 +26,17 @@ export const useSelectedLocationsHelpers = () => { (location) => checkIsSelectionBlocked({ location, selectedLocations, multipleItemsLimit }), [selectedLocations, multipleItemsLimit], ); + const checkIsDeselectionBlockedWrapped = (location) => { + const isLocationSelected = checkIsSelected({ location, selectedLocations }); + const isInitSelectedLocation = initSelectedLocations.includes(location.id); + + return isLocationSelected && isInitSelectedLocation && isInitLocationsDeselectionBlocked; + }; return { checkIsSelectable: checkIsSelectableWrapped, checkIsSelected: checkIsSelectedWrapped, checkIsSelectionBlocked: checkIsSelectionBlockedWrapped, + checkIsDeselectionBlocked: checkIsDeselectionBlockedWrapped, }; }; diff --git a/src/bundle/ui-dev/src/modules/universal-discovery/universal.discovery.module.js b/src/bundle/ui-dev/src/modules/universal-discovery/universal.discovery.module.js index 6c05404ac4..50ef686543 100644 --- a/src/bundle/ui-dev/src/modules/universal-discovery/universal.discovery.module.js +++ b/src/bundle/ui-dev/src/modules/universal-discovery/universal.discovery.module.js @@ -186,6 +186,7 @@ export const StartingLocationIdContext = createContext(); export const LoadedLocationsMapContext = createContext(); export const RootLocationIdContext = createContext(); export const SelectedLocationsContext = createContext(); +export const SelectionConfigContext = createContext(); export const CreateContentWidgetContext = createContext(); export const ContentOnTheFlyDataContext = createContext(); export const ContentOnTheFlyConfigContext = createContext(); @@ -237,6 +238,13 @@ const UniversalDiscoveryModule = (props) => { 'm-ud': true, 'm-ud--locations-selected': !!selectedLocations.length && props.allowConfirmation, }); + const selectionConfigValue = useMemo( + () => ({ + isInitLocationsDeselectionBlocked: props.isInitLocationsDeselectionBlocked, + initSelectedLocations: props.selectedLocations, + }), + [], + ); const loadPermissions = () => { const locationIds = selectedLocations .filter((item) => !item.permissions) @@ -536,61 +544,67 @@ const UniversalDiscoveryModule = (props) => { - - - - - - - - - - - - - - - - - + + + + + + + + + + + @@ -651,6 +665,7 @@ UniversalDiscoveryModule.propTypes = { }), ).isRequired, selectedLocations: PropTypes.array, + isInitLocationsDeselectionBlocked: PropTypes.bool, allowRedirects: PropTypes.bool.isRequired, allowConfirmation: PropTypes.bool.isRequired, restInfo: PropTypes.shape({ @@ -674,6 +689,7 @@ UniversalDiscoveryModule.defaultProps = { activeSortOrder: 'ascending', activeView: 'finder', selectedLocations: [], + isInitLocationsDeselectionBlocked: false, restInfo: defaultRestInfo, snackbarEnabledActions: Object.values(SNACKBAR_ACTIONS), };