2022-11-09 19:13:44 +01:00
|
|
|
import { createSlice } from '@reduxjs/toolkit';
|
2022-12-22 18:30:02 +01:00
|
|
|
import { assocPath, last, pipe, reject } from 'ramda';
|
2023-07-24 17:30:58 +02:00
|
|
|
import type { ShlinkApiClient, ShlinkShortUrlsListParams, ShlinkShortUrlsResponse } from '../../api-contract';
|
2023-07-26 20:04:50 +02:00
|
|
|
import { createAsyncThunk } from '../../utils/redux';
|
2023-02-18 11:11:01 +01:00
|
|
|
import { createNewVisits } from '../../visits/reducers/visitCreation';
|
|
|
|
import type { ShortUrl } from '../data';
|
|
|
|
import { shortUrlMatches } from '../helpers';
|
2023-02-18 10:40:37 +01:00
|
|
|
import type { createShortUrl } from './shortUrlCreation';
|
2023-02-18 11:11:01 +01:00
|
|
|
import { shortUrlDeleted } from './shortUrlDeletion';
|
2023-02-18 10:40:37 +01:00
|
|
|
import type { editShortUrl } from './shortUrlEdition';
|
2020-08-27 18:31:56 +02:00
|
|
|
|
2022-11-09 19:13:44 +01:00
|
|
|
const REDUCER_PREFIX = 'shlink/shortUrlsList';
|
2022-01-08 10:51:34 +01:00
|
|
|
export const ITEMS_IN_OVERVIEW_PAGE = 5;
|
|
|
|
|
2020-08-27 18:31:56 +02:00
|
|
|
export interface ShortUrlsList {
|
2020-08-30 19:45:17 +02:00
|
|
|
shortUrls?: ShlinkShortUrlsResponse;
|
2020-08-27 18:31:56 +02:00
|
|
|
loading: boolean;
|
|
|
|
error: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const initialState: ShortUrlsList = {
|
|
|
|
loading: true,
|
|
|
|
error: false,
|
|
|
|
};
|
|
|
|
|
2023-07-26 20:04:50 +02:00
|
|
|
export const listShortUrls = (apiClientFactory: () => ShlinkApiClient) => createAsyncThunk(
|
2022-11-09 19:13:44 +01:00
|
|
|
`${REDUCER_PREFIX}/listShortUrls`,
|
2023-07-26 20:04:50 +02:00
|
|
|
(params: ShlinkShortUrlsListParams | void): Promise<ShlinkShortUrlsResponse> => apiClientFactory().listShortUrls(
|
|
|
|
params ?? {},
|
|
|
|
),
|
2022-11-09 19:13:44 +01:00
|
|
|
);
|
2020-09-12 11:31:44 +02:00
|
|
|
|
2022-11-09 19:13:44 +01:00
|
|
|
export const shortUrlsListReducerCreator = (
|
|
|
|
listShortUrlsThunk: ReturnType<typeof listShortUrls>,
|
|
|
|
editShortUrlThunk: ReturnType<typeof editShortUrl>,
|
|
|
|
createShortUrlThunk: ReturnType<typeof createShortUrl>,
|
|
|
|
) => createSlice({
|
|
|
|
name: REDUCER_PREFIX,
|
|
|
|
initialState,
|
|
|
|
reducers: {},
|
|
|
|
extraReducers: (builder) => {
|
|
|
|
builder.addCase(listShortUrlsThunk.pending, (state) => ({ ...state, loading: true, error: false }));
|
|
|
|
builder.addCase(listShortUrlsThunk.rejected, () => ({ loading: false, error: true }));
|
|
|
|
builder.addCase(
|
|
|
|
listShortUrlsThunk.fulfilled,
|
|
|
|
(_, { payload: shortUrls }) => ({ loading: false, error: false, shortUrls }),
|
|
|
|
);
|
2021-04-24 17:58:37 +02:00
|
|
|
|
2022-11-09 19:13:44 +01:00
|
|
|
builder.addCase(
|
|
|
|
createShortUrlThunk.fulfilled,
|
|
|
|
pipe(
|
|
|
|
// The only place where the list and the creation form coexist is the overview page.
|
|
|
|
// There we can assume we are displaying page 1, and therefore, we can safely prepend the new short URL.
|
|
|
|
// We can also remove the items above the amount that is displayed there.
|
|
|
|
(state, { payload }) => (!state.shortUrls ? state : assocPath(
|
|
|
|
['shortUrls', 'data'],
|
|
|
|
[payload, ...state.shortUrls.data.slice(0, ITEMS_IN_OVERVIEW_PAGE - 1)],
|
|
|
|
state,
|
|
|
|
)),
|
|
|
|
(state: ShortUrlsList) => (!state.shortUrls ? state : assocPath(
|
|
|
|
['shortUrls', 'pagination', 'totalItems'],
|
|
|
|
state.shortUrls.pagination.totalItems + 1,
|
|
|
|
state,
|
|
|
|
)),
|
|
|
|
),
|
|
|
|
);
|
2020-08-27 18:31:56 +02:00
|
|
|
|
2022-11-09 19:13:44 +01:00
|
|
|
builder.addCase(
|
|
|
|
editShortUrlThunk.fulfilled,
|
|
|
|
(state, { payload: editedShortUrl }) => (!state.shortUrls ? state : assocPath(
|
|
|
|
['shortUrls', 'data'],
|
|
|
|
state.shortUrls.data.map((shortUrl) => {
|
|
|
|
const { shortCode, domain } = editedShortUrl;
|
|
|
|
return shortUrlMatches(shortUrl, shortCode, domain) ? editedShortUrl : shortUrl;
|
|
|
|
}),
|
|
|
|
state,
|
|
|
|
)),
|
|
|
|
);
|
2020-08-27 18:31:56 +02:00
|
|
|
|
2022-11-09 19:13:44 +01:00
|
|
|
builder.addCase(
|
2022-11-22 19:39:07 +01:00
|
|
|
shortUrlDeleted,
|
2022-11-09 19:13:44 +01:00
|
|
|
pipe(
|
|
|
|
(state, { payload }) => (!state.shortUrls ? state : assocPath(
|
|
|
|
['shortUrls', 'data'],
|
|
|
|
reject<ShortUrl, ShortUrl[]>((shortUrl) =>
|
|
|
|
shortUrlMatches(shortUrl, payload.shortCode, payload.domain), state.shortUrls.data),
|
|
|
|
state,
|
|
|
|
)),
|
|
|
|
(state) => (!state.shortUrls ? state : assocPath(
|
|
|
|
['shortUrls', 'pagination', 'totalItems'],
|
|
|
|
state.shortUrls.pagination.totalItems - 1,
|
|
|
|
state,
|
|
|
|
)),
|
|
|
|
),
|
|
|
|
);
|
2020-08-27 18:31:56 +02:00
|
|
|
|
2022-11-09 19:13:44 +01:00
|
|
|
builder.addCase(
|
|
|
|
createNewVisits,
|
|
|
|
(state, { payload }) => assocPath(
|
|
|
|
['shortUrls', 'data'],
|
|
|
|
state.shortUrls?.data?.map(
|
2022-12-22 18:30:02 +01:00
|
|
|
// Find the last of the new visit for this short URL, and pick its short URL. It will have an up-to-date amount of visits.
|
|
|
|
(currentShortUrl) => last(
|
|
|
|
payload.createdVisits.filter(
|
|
|
|
({ shortUrl }) => shortUrl && shortUrlMatches(currentShortUrl, shortUrl.shortCode, shortUrl.domain),
|
|
|
|
),
|
|
|
|
)?.shortUrl ?? currentShortUrl,
|
2022-11-09 19:13:44 +01:00
|
|
|
),
|
|
|
|
state,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
});
|