(ui) update cache to render pinned images alongside unpinned correctly, as well as changes in pinnedness

This commit is contained in:
Mary Hipp 2023-08-11 14:41:48 -04:00 committed by psychedelicious
parent 37be827e17
commit 0a71d6baa1
3 changed files with 108 additions and 23 deletions

View File

@ -134,12 +134,12 @@ const SingleSelectionMenuItems = (props: SingleSelectionMenuItemsProps) => {
}, [copyImageToClipboard, imageDTO.image_url]); }, [copyImageToClipboard, imageDTO.image_url]);
const handlePinImage = useCallback(() => { const handlePinImage = useCallback(() => {
togglePin({ imageName: imageDTO.image_name, pinned: true }); togglePin({ imageDTO, pinned: true });
}, [togglePin, imageDTO.image_name]); }, [togglePin, imageDTO]);
const handleUnpinImage = useCallback(() => { const handleUnpinImage = useCallback(() => {
togglePin({ imageName: imageDTO.image_name, pinned: false }); togglePin({ imageDTO, pinned: false });
}, [togglePin, imageDTO.image_name]); }, [togglePin, imageDTO]);
return ( return (
<> <>

View File

@ -392,23 +392,28 @@ export const imagesApi = api.injectEndpoints({
*/ */
changeImagePinned: build.mutation< changeImagePinned: build.mutation<
ImageDTO, ImageDTO,
{ imageName: string; pinned: boolean } { imageDTO: ImageDTO; pinned: boolean }
>({ >({
query: ({ imageName, pinned }) => ({ query: ({ imageDTO, pinned }) => ({
url: `images/i/${imageName}`, url: `images/i/${imageDTO.image_name}`,
method: 'PATCH', method: 'PATCH',
body: { pinned }, body: { pinned },
}), }),
// invalidatesTags: (result, error, { imageDTO }) => [ invalidatesTags: (result, error, { imageDTO }) => [
// { type: 'BoardImagesTotal', id: imageDTO.board_id ?? 'none' }, {
// { type: 'BoardAssetsTotal', id: imageDTO.board_id ?? 'none' }, type: 'ImageList',
// ], id: getListImagesUrl({
board_id: imageDTO.board_id,
categories: IMAGE_CATEGORIES,
}),
},
],
async onQueryStarted( async onQueryStarted(
{ imageName, pinned }, { imageDTO, pinned },
{ dispatch, queryFulfilled, getState } { dispatch, queryFulfilled, getState }
) { ) {
/** /**
* Cache changes for `changeImageSessionId`: * Cache changes for `changeImagePinned`:
* - *update* getImageDTO * - *update* getImageDTO
*/ */
@ -420,7 +425,7 @@ export const imagesApi = api.injectEndpoints({
dispatch( dispatch(
imagesApi.util.updateQueryData( imagesApi.util.updateQueryData(
'getImageDTO', 'getImageDTO',
imageName, imageDTO.image_name,
(draft) => { (draft) => {
Object.assign(draft, { pinned }); Object.assign(draft, { pinned });
} }
@ -428,6 +433,73 @@ export const imagesApi = api.injectEndpoints({
) )
); );
const categories = getCategories(imageDTO);
const queryArgs = {
board_id: imageDTO.board_id ?? 'none',
categories,
};
const currentCache = imagesApi.endpoints.listImages.select(queryArgs)(
getState()
);
const { data: total } = IMAGE_CATEGORIES.includes(
imageDTO.image_category
)
? boardsApi.endpoints.getBoardImagesTotal.select(
imageDTO.board_id ?? 'none'
)(getState())
: boardsApi.endpoints.getBoardAssetsTotal.select(
imageDTO.board_id ?? 'none'
)(getState());
// IF it eligible for insertion into existing $cache
// "eligible" means either:
// - The cache is fully populated, with all images in the db cached
// OR
// - The image's `created_at` is within the range of the cached images within that pinned state
const updatedImage: ImageDTO = { ...imageDTO, pinned }
const isCacheFullyPopulated =
currentCache.data && currentCache.data.ids.length >= (total ?? 0);
const isInDateRangeForPinnedState = getIsImageInDateRange(
currentCache.data,
updatedImage
);
if (!isInDateRangeForPinnedState) {
// if newly pinned or unpinned image is not in date range for its new state, remove from cache
patches.push(
dispatch(
imagesApi.util.updateQueryData(
'listImages',
queryArgs,
(draft) => {
imagesAdapter.removeOne(draft, updatedImage.image_name);
}
)
)
);
}
if (isCacheFullyPopulated || isInDateRangeForPinnedState) {
// *upsert* to $cache
patches.push(
dispatch(
imagesApi.util.updateQueryData(
'listImages',
queryArgs,
(draft) => {
imagesAdapter.upsertOne(draft, updatedImage);
}
)
)
);
}
try { try {
await queryFulfilled; await queryFulfilled;
} catch { } catch {

View File

@ -14,22 +14,26 @@ export const getIsImageInDateRange = (
if (!data) { if (!data) {
return false; return false;
} }
const cacheImageDTOS = imagesSelectors.selectAll(data);
if (cacheImageDTOS.length > 1) { const totalCachedImageDtos = imagesSelectors.selectAll(data);
// Images are sorted by `created_at` DESC
// check if the image is newer than the oldest image in the cache if (totalCachedImageDtos.length <= 1) {
return true;
}
const cacheImageDTOSForPinnedState = totalCachedImageDtos.filter((image) => image.pinned === imageDTO.pinned);
if (cacheImageDTOSForPinnedState.length > 1) {
// Images are sorted by `pinned` DESC and then `created_at` DESC
// check if the image is newer than the oldest image in the cache for either the pinned group or unpinned group
const createdDate = new Date(imageDTO.created_at); const createdDate = new Date(imageDTO.created_at);
const oldestImage = cacheImageDTOS[cacheImageDTOS.length - 1]; const oldestImage = cacheImageDTOSForPinnedState[cacheImageDTOSForPinnedState.length - 1];
if (!oldestImage) { if (!oldestImage) {
// satisfy TS gods, we already confirmed the array has more than one image // satisfy TS gods, we already confirmed the array has more than one image
return false; return false;
} }
const oldestDate = new Date(oldestImage.created_at); const oldestDate = new Date(oldestImage.created_at);
return createdDate >= oldestDate; return createdDate >= oldestDate;
} else if ([0, 1].includes(cacheImageDTOS.length)) {
// if there are only 1 or 0 images in the cache, we consider the image to be in the date range
return true;
} }
return false; return false;
}; };
@ -45,7 +49,16 @@ export const getCategories = (imageDTO: ImageDTO) => {
// with some other store of data. We will use the RTK Query cache as that store. // with some other store of data. We will use the RTK Query cache as that store.
export const imagesAdapter = createEntityAdapter<ImageDTO>({ export const imagesAdapter = createEntityAdapter<ImageDTO>({
selectId: (image) => image.image_name, selectId: (image) => image.image_name,
sortComparer: (a, b) => dateComparator(b.updated_at, a.updated_at), sortComparer: (a, b) => {
// Compare pinned images first
if (a.pinned && !b.pinned) {
return -1;
}
if (!a.pinned && b.pinned) {
return 1;
}
return dateComparator(b.created_at, a.created_at)
},
}); });
// Create selectors for the adapter. // Create selectors for the adapter.