2023-02-18 11:11:01 +01:00
|
|
|
import { pipe } from 'ramda';
|
2020-11-13 22:44:26 +01:00
|
|
|
import { useEffect, useState } from 'react';
|
2018-09-16 09:35:39 +02:00
|
|
|
import { Modal, ModalBody, ModalFooter, ModalHeader } from 'reactstrap';
|
2023-07-16 08:47:10 +02:00
|
|
|
import { ShlinkApiError } from '../../../api/ShlinkApiError';
|
|
|
|
import { Result } from '../../../utils/Result';
|
|
|
|
import { handleEventPreventingDefault } from '../../../utils/utils';
|
2023-07-24 17:30:58 +02:00
|
|
|
import { isInvalidDeletionError } from '../../api-contract/utils';
|
2023-02-18 11:11:01 +01:00
|
|
|
import type { ShortUrlIdentifier, ShortUrlModalProps } from '../data';
|
|
|
|
import type { ShortUrlDeletion } from '../reducers/shortUrlDeletion';
|
2020-01-11 12:24:45 +01:00
|
|
|
|
2020-08-26 20:37:36 +02:00
|
|
|
interface DeleteShortUrlModalConnectProps extends ShortUrlModalProps {
|
|
|
|
shortUrlDeletion: ShortUrlDeletion;
|
2022-11-22 19:39:07 +01:00
|
|
|
deleteShortUrl: (shortUrl: ShortUrlIdentifier) => Promise<void>;
|
|
|
|
shortUrlDeleted: (shortUrl: ShortUrlIdentifier) => void;
|
2020-08-26 20:37:36 +02:00
|
|
|
resetDeleteShortUrl: () => void;
|
|
|
|
}
|
2020-05-31 10:23:13 +02:00
|
|
|
|
2022-11-25 19:08:40 +01:00
|
|
|
const DELETION_PATTERN = 'delete';
|
|
|
|
|
2022-11-22 19:39:07 +01:00
|
|
|
export const DeleteShortUrlModal = ({
|
|
|
|
shortUrl,
|
|
|
|
toggle,
|
|
|
|
isOpen,
|
|
|
|
shortUrlDeletion,
|
|
|
|
resetDeleteShortUrl,
|
|
|
|
deleteShortUrl,
|
|
|
|
shortUrlDeleted,
|
|
|
|
}: DeleteShortUrlModalConnectProps) => {
|
2022-03-26 12:17:42 +01:00
|
|
|
const [inputValue, setInputValue] = useState('');
|
2018-09-16 09:35:39 +02:00
|
|
|
|
2020-05-31 10:23:13 +02:00
|
|
|
useEffect(() => resetDeleteShortUrl, []);
|
|
|
|
|
2022-11-22 19:39:07 +01:00
|
|
|
const { loading, error, deleted, errorData } = shortUrlDeletion;
|
2020-05-31 10:23:13 +02:00
|
|
|
const close = pipe(resetDeleteShortUrl, toggle);
|
2022-11-22 19:39:07 +01:00
|
|
|
const handleDeleteUrl = handleEventPreventingDefault(() => deleteShortUrl(shortUrl).then(toggle));
|
2018-09-16 10:47:17 +02:00
|
|
|
|
2020-05-31 10:23:13 +02:00
|
|
|
return (
|
2022-11-22 19:39:07 +01:00
|
|
|
<Modal isOpen={isOpen} toggle={close} centered onClosed={() => deleted && shortUrlDeleted(shortUrl)}>
|
2020-05-31 10:23:13 +02:00
|
|
|
<form onSubmit={handleDeleteUrl}>
|
|
|
|
<ModalHeader toggle={close}>
|
|
|
|
<span className="text-danger">Delete short URL</span>
|
|
|
|
</ModalHeader>
|
|
|
|
<ModalBody>
|
|
|
|
<p><b className="text-danger">Caution!</b> You are about to delete a short URL.</p>
|
|
|
|
<p>This action cannot be undone. Once you have deleted it, all the visits stats will be lost.</p>
|
2022-11-25 19:08:40 +01:00
|
|
|
<p>Write <b>{DELETION_PATTERN}</b> to confirm deletion.</p>
|
2018-09-16 09:35:39 +02:00
|
|
|
|
2020-05-31 10:23:13 +02:00
|
|
|
<input
|
|
|
|
type="text"
|
|
|
|
className="form-control"
|
2022-11-25 19:08:40 +01:00
|
|
|
placeholder={`Insert ${DELETION_PATTERN}`}
|
2020-05-31 10:23:13 +02:00
|
|
|
value={inputValue}
|
|
|
|
onChange={(e) => setInputValue(e.target.value)}
|
|
|
|
/>
|
2018-09-16 09:35:39 +02:00
|
|
|
|
2020-12-21 21:19:02 +01:00
|
|
|
{error && (
|
|
|
|
<Result type={isInvalidDeletionError(errorData) ? 'warning' : 'error'} small className="mt-2">
|
|
|
|
<ShlinkApiError errorData={errorData} fallbackMessage="Something went wrong while deleting the URL :(" />
|
2020-12-21 17:54:05 +01:00
|
|
|
</Result>
|
2020-05-31 10:23:13 +02:00
|
|
|
)}
|
|
|
|
</ModalBody>
|
|
|
|
<ModalFooter>
|
|
|
|
<button type="button" className="btn btn-link" onClick={close}>Cancel</button>
|
|
|
|
<button
|
|
|
|
type="submit"
|
|
|
|
className="btn btn-danger"
|
2022-11-25 19:08:40 +01:00
|
|
|
disabled={inputValue !== DELETION_PATTERN || loading}
|
2020-05-31 10:23:13 +02:00
|
|
|
>
|
2022-11-06 12:46:29 +01:00
|
|
|
{loading ? 'Deleting...' : 'Delete'}
|
2020-05-31 10:23:13 +02:00
|
|
|
</button>
|
|
|
|
</ModalFooter>
|
|
|
|
</form>
|
|
|
|
</Modal>
|
|
|
|
);
|
|
|
|
};
|