2023-03-13 09:05:54 +01:00
|
|
|
import type { IconProp } from '@fortawesome/fontawesome-svg-core';
|
2023-03-13 18:01:21 +01:00
|
|
|
import { faAndroid, faApple } from '@fortawesome/free-brands-svg-icons';
|
|
|
|
import { faDesktop } from '@fortawesome/free-solid-svg-icons';
|
2023-03-13 09:05:54 +01:00
|
|
|
import classNames from 'classnames';
|
2023-02-18 11:11:01 +01:00
|
|
|
import { parseISO } from 'date-fns';
|
2023-03-13 09:05:54 +01:00
|
|
|
import { isEmpty, pipe, replace, trim } from 'ramda';
|
|
|
|
import type { ChangeEvent, FC } from 'react';
|
2023-02-18 10:40:37 +01:00
|
|
|
import { useEffect, useState } from 'react';
|
2023-03-13 18:01:21 +01:00
|
|
|
import { Button, FormGroup, Input, Row } from 'reactstrap';
|
2023-02-18 11:11:01 +01:00
|
|
|
import type { InputType } from 'reactstrap/types/lib/Input';
|
2023-07-16 08:47:10 +02:00
|
|
|
import { Checkbox } from '../../utils/Checkbox';
|
|
|
|
import type { DateTimeInputProps } from '../../utils/dates/DateTimeInput';
|
|
|
|
import { DateTimeInput } from '../../utils/dates/DateTimeInput';
|
|
|
|
import { formatIsoDate } from '../../utils/helpers/date';
|
|
|
|
import { IconInput } from '../../utils/IconInput';
|
|
|
|
import { SimpleCard } from '../../utils/SimpleCard';
|
|
|
|
import { handleEventPreventingDefault, hasValue } from '../../utils/utils';
|
2023-02-18 11:11:01 +01:00
|
|
|
import type { DomainSelectorProps } from '../domains/DomainSelector';
|
|
|
|
import type { TagsSelectorProps } from '../tags/helpers/TagsSelector';
|
2023-07-16 22:54:49 +02:00
|
|
|
import { useFeature } from '../utils/features';
|
2023-03-13 09:05:54 +01:00
|
|
|
import type { DeviceLongUrls, ShortUrlData } from './data';
|
2021-06-23 19:52:19 +02:00
|
|
|
import { ShortUrlFormCheckboxGroup } from './helpers/ShortUrlFormCheckboxGroup';
|
2023-02-18 11:11:01 +01:00
|
|
|
import { UseExistingIfFoundInfoIcon } from './UseExistingIfFoundInfoIcon';
|
2021-03-19 19:11:27 +01:00
|
|
|
import './ShortUrlForm.scss';
|
|
|
|
|
2021-03-27 18:39:55 +01:00
|
|
|
export type Mode = 'create' | 'create-basic' | 'edit';
|
|
|
|
|
2021-03-19 19:11:27 +01:00
|
|
|
type DateFields = 'validSince' | 'validUntil';
|
|
|
|
type NonDateFields = 'longUrl' | 'customSlug' | 'shortCodeLength' | 'domain' | 'maxVisits' | 'title';
|
|
|
|
|
|
|
|
export interface ShortUrlFormProps {
|
|
|
|
mode: Mode;
|
|
|
|
saving: boolean;
|
|
|
|
initialState: ShortUrlData;
|
|
|
|
onSave: (shortUrlData: ShortUrlData) => Promise<unknown>;
|
|
|
|
}
|
|
|
|
|
2021-03-20 16:32:12 +01:00
|
|
|
const normalizeTag = pipe(trim, replace(/ /g, '-'));
|
2022-03-26 12:17:42 +01:00
|
|
|
const toDate = (date?: string | Date): Date | undefined => (typeof date === 'string' ? parseISO(date) : date);
|
2021-03-20 16:32:12 +01:00
|
|
|
|
2021-03-19 19:11:27 +01:00
|
|
|
export const ShortUrlForm = (
|
|
|
|
TagsSelector: FC<TagsSelectorProps>,
|
|
|
|
DomainSelector: FC<DomainSelectorProps>,
|
2023-07-16 22:54:49 +02:00
|
|
|
): FC<ShortUrlFormProps> => ({ mode, saving, onSave, initialState }) => {
|
2022-03-26 12:17:42 +01:00
|
|
|
const [shortUrlData, setShortUrlData] = useState(initialState);
|
2023-03-13 09:05:54 +01:00
|
|
|
const reset = () => setShortUrlData(initialState);
|
2023-07-16 22:54:49 +02:00
|
|
|
const supportsDeviceLongUrls = useFeature('deviceLongUrls');
|
2023-03-13 09:05:54 +01:00
|
|
|
|
2021-03-20 16:32:12 +01:00
|
|
|
const isEdit = mode === 'edit';
|
2021-12-19 12:52:49 +01:00
|
|
|
const isBasicMode = mode === 'create-basic';
|
2021-03-19 19:11:27 +01:00
|
|
|
const changeTags = (tags: string[]) => setShortUrlData({ ...shortUrlData, tags: tags.map(normalizeTag) });
|
2023-03-13 09:05:54 +01:00
|
|
|
const setResettableValue = (value: string, initialValue?: any) => {
|
|
|
|
if (hasValue(value)) {
|
|
|
|
return value;
|
|
|
|
}
|
2021-10-17 12:35:11 +02:00
|
|
|
|
2023-03-13 09:05:54 +01:00
|
|
|
// If an initial value was provided for this when the input is "emptied", explicitly set it to null so that the
|
|
|
|
// value gets removed. Otherwise, set undefined so that it gets ignored.
|
|
|
|
return hasValue(initialValue) ? null : undefined;
|
2021-10-17 12:35:11 +02:00
|
|
|
};
|
2021-03-19 19:11:27 +01:00
|
|
|
const submit = handleEventPreventingDefault(async () => onSave({
|
|
|
|
...shortUrlData,
|
2021-03-27 09:49:47 +01:00
|
|
|
validSince: formatIsoDate(shortUrlData.validSince) ?? null,
|
|
|
|
validUntil: formatIsoDate(shortUrlData.validUntil) ?? null,
|
|
|
|
maxVisits: !hasValue(shortUrlData.maxVisits) ? null : Number(shortUrlData.maxVisits),
|
2021-03-20 16:32:12 +01:00
|
|
|
}).then(() => !isEdit && reset()).catch(() => {}));
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
setShortUrlData(initialState);
|
2022-03-26 12:17:42 +01:00
|
|
|
}, [initialState]);
|
2021-03-19 19:11:27 +01:00
|
|
|
|
2023-03-13 09:05:54 +01:00
|
|
|
// TODO Consider extracting these functions to local components
|
2021-12-19 12:52:49 +01:00
|
|
|
const renderOptionalInput = (
|
|
|
|
id: NonDateFields,
|
|
|
|
placeholder: string,
|
|
|
|
type: InputType = 'text',
|
2023-03-13 09:05:54 +01:00
|
|
|
props: any = {},
|
2021-12-19 12:52:49 +01:00
|
|
|
fromGroupProps = {},
|
|
|
|
) => (
|
|
|
|
<FormGroup {...fromGroupProps}>
|
2021-03-19 19:11:27 +01:00
|
|
|
<Input
|
|
|
|
id={id}
|
|
|
|
type={type}
|
|
|
|
placeholder={placeholder}
|
2021-03-20 16:32:12 +01:00
|
|
|
value={shortUrlData[id] ?? ''}
|
2023-03-13 09:05:54 +01:00
|
|
|
onChange={props.onChange ?? ((e) => setShortUrlData({ ...shortUrlData, [id]: e.target.value }))}
|
2021-03-19 19:11:27 +01:00
|
|
|
{...props}
|
|
|
|
/>
|
|
|
|
</FormGroup>
|
|
|
|
);
|
2023-03-13 09:05:54 +01:00
|
|
|
const renderDeviceLongUrlInput = (id: keyof DeviceLongUrls, placeholder: string, icon: IconProp) => (
|
2023-03-13 18:01:21 +01:00
|
|
|
<IconInput
|
|
|
|
icon={icon}
|
|
|
|
id={id}
|
|
|
|
type="url"
|
|
|
|
placeholder={placeholder}
|
|
|
|
value={shortUrlData.deviceLongUrls?.[id] ?? ''}
|
|
|
|
onChange={(e) => setShortUrlData({
|
|
|
|
...shortUrlData,
|
|
|
|
deviceLongUrls: {
|
|
|
|
...(shortUrlData.deviceLongUrls ?? {}),
|
|
|
|
[id]: setResettableValue(e.target.value, initialState.deviceLongUrls?.[id]),
|
|
|
|
},
|
|
|
|
})}
|
|
|
|
/>
|
2023-03-13 09:05:54 +01:00
|
|
|
);
|
2022-10-18 22:02:09 +02:00
|
|
|
const renderDateInput = (id: DateFields, placeholder: string, props: Partial<DateTimeInputProps> = {}) => (
|
|
|
|
<DateTimeInput
|
2022-03-06 11:16:31 +01:00
|
|
|
selected={shortUrlData[id] ? toDate(shortUrlData[id] as string | Date) : null}
|
|
|
|
placeholderText={placeholder}
|
|
|
|
isClearable
|
|
|
|
onChange={(date) => setShortUrlData({ ...shortUrlData, [id]: date })}
|
|
|
|
{...props}
|
|
|
|
/>
|
2021-03-19 19:11:27 +01:00
|
|
|
);
|
|
|
|
const basicComponents = (
|
|
|
|
<>
|
|
|
|
<FormGroup>
|
|
|
|
<Input
|
|
|
|
bsSize="lg"
|
|
|
|
type="url"
|
|
|
|
placeholder="URL to be shortened"
|
|
|
|
required
|
|
|
|
value={shortUrlData.longUrl}
|
|
|
|
onChange={(e) => setShortUrlData({ ...shortUrlData, longUrl: e.target.value })}
|
|
|
|
/>
|
|
|
|
</FormGroup>
|
2021-12-19 12:52:49 +01:00
|
|
|
<Row>
|
|
|
|
{isBasicMode && renderOptionalInput('customSlug', 'Custom slug', 'text', { bsSize: 'lg' }, { className: 'col-lg-6' })}
|
2022-03-24 20:23:46 +01:00
|
|
|
<div className={isBasicMode ? 'col-lg-6 mb-3' : 'col-12'}>
|
2021-12-19 12:52:49 +01:00
|
|
|
<TagsSelector selectedTags={shortUrlData.tags ?? []} onChange={changeTags} />
|
2022-03-06 11:16:31 +01:00
|
|
|
</div>
|
2021-12-19 12:52:49 +01:00
|
|
|
</Row>
|
2021-03-19 19:11:27 +01:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
2022-07-05 20:30:23 +02:00
|
|
|
<form name="shortUrlForm" className="short-url-form" onSubmit={submit}>
|
2021-12-19 12:52:49 +01:00
|
|
|
{isBasicMode && basicComponents}
|
|
|
|
{!isBasicMode && (
|
2021-03-19 19:11:27 +01:00
|
|
|
<>
|
2023-03-13 09:05:54 +01:00
|
|
|
<Row>
|
|
|
|
<div
|
|
|
|
className={classNames('mb-3', { 'col-sm-6': supportsDeviceLongUrls, 'col-12': !supportsDeviceLongUrls })}
|
|
|
|
>
|
|
|
|
<SimpleCard title="Main options" className="mb-3">
|
|
|
|
{basicComponents}
|
|
|
|
</SimpleCard>
|
|
|
|
</div>
|
|
|
|
{supportsDeviceLongUrls && (
|
|
|
|
<div className="col-sm-6 mb-3">
|
|
|
|
<SimpleCard title="Device-specific long URLs">
|
|
|
|
<FormGroup>
|
2023-03-13 18:01:21 +01:00
|
|
|
{renderDeviceLongUrlInput('android', 'Android-specific redirection', faAndroid)}
|
2023-03-13 09:05:54 +01:00
|
|
|
</FormGroup>
|
|
|
|
<FormGroup>
|
2023-03-13 18:01:21 +01:00
|
|
|
{renderDeviceLongUrlInput('ios', 'iOS-specific redirection', faApple)}
|
2023-03-13 09:05:54 +01:00
|
|
|
</FormGroup>
|
|
|
|
{renderDeviceLongUrlInput('desktop', 'Desktop-specific redirection', faDesktop)}
|
|
|
|
</SimpleCard>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</Row>
|
2021-03-19 19:11:27 +01:00
|
|
|
|
2021-03-20 11:18:00 +01:00
|
|
|
<Row>
|
2022-05-01 11:00:46 +02:00
|
|
|
<div className="col-sm-6 mb-3">
|
|
|
|
<SimpleCard title="Customize the short URL">
|
2023-03-13 09:05:54 +01:00
|
|
|
{renderOptionalInput('title', 'Title', 'text', {
|
|
|
|
onChange: ({ target }: ChangeEvent<HTMLInputElement>) => setShortUrlData({
|
|
|
|
...shortUrlData,
|
|
|
|
title: setResettableValue(target.value, initialState.title),
|
|
|
|
}),
|
|
|
|
})}
|
2022-05-01 11:00:46 +02:00
|
|
|
{!isEdit && (
|
|
|
|
<>
|
|
|
|
<Row>
|
|
|
|
<div className="col-lg-6">
|
|
|
|
{renderOptionalInput('customSlug', 'Custom slug', 'text', {
|
|
|
|
disabled: hasValue(shortUrlData.shortCodeLength),
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
<div className="col-lg-6">
|
|
|
|
{renderOptionalInput('shortCodeLength', 'Short code length', 'number', {
|
|
|
|
min: 4,
|
|
|
|
disabled: hasValue(shortUrlData.customSlug),
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
</Row>
|
|
|
|
<DomainSelector
|
|
|
|
value={shortUrlData.domain}
|
|
|
|
onChange={(domain?: string) => setShortUrlData({ ...shortUrlData, domain })}
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</SimpleCard>
|
|
|
|
</div>
|
2021-03-19 19:11:27 +01:00
|
|
|
|
2022-05-01 11:00:46 +02:00
|
|
|
<div className="col-sm-6 mb-3">
|
2021-03-19 19:11:27 +01:00
|
|
|
<SimpleCard title="Limit access to the short URL">
|
|
|
|
{renderOptionalInput('maxVisits', 'Maximum number of visits allowed', 'number', { min: 1 })}
|
2022-03-06 11:16:31 +01:00
|
|
|
<div className="mb-3">
|
|
|
|
{renderDateInput('validSince', 'Enabled since...', { maxDate: shortUrlData.validUntil ? toDate(shortUrlData.validUntil) : undefined })}
|
|
|
|
</div>
|
2021-06-24 20:13:06 +02:00
|
|
|
{renderDateInput('validUntil', 'Enabled until...', { minDate: shortUrlData.validSince ? toDate(shortUrlData.validSince) : undefined })}
|
2021-03-19 19:11:27 +01:00
|
|
|
</SimpleCard>
|
|
|
|
</div>
|
2021-03-20 11:18:00 +01:00
|
|
|
</Row>
|
2021-03-19 19:11:27 +01:00
|
|
|
|
2021-10-13 22:50:48 +02:00
|
|
|
<Row>
|
2022-05-01 11:00:46 +02:00
|
|
|
<div className="col-sm-6 mb-3">
|
2021-10-13 22:50:48 +02:00
|
|
|
<SimpleCard title="Extra checks">
|
|
|
|
<ShortUrlFormCheckboxGroup
|
|
|
|
infoTooltip="If checked, Shlink will try to reach the long URL, failing in case it's not publicly accessible."
|
|
|
|
checked={shortUrlData.validateUrl}
|
|
|
|
onChange={(validateUrl) => setShortUrlData({ ...shortUrlData, validateUrl })}
|
2021-06-23 19:52:19 +02:00
|
|
|
>
|
2021-10-13 22:50:48 +02:00
|
|
|
Validate URL
|
|
|
|
</ShortUrlFormCheckboxGroup>
|
|
|
|
{!isEdit && (
|
|
|
|
<p>
|
|
|
|
<Checkbox
|
|
|
|
inline
|
2022-03-05 13:26:28 +01:00
|
|
|
className="me-2"
|
2021-10-13 22:50:48 +02:00
|
|
|
checked={shortUrlData.findIfExists}
|
|
|
|
onChange={(findIfExists) => setShortUrlData({ ...shortUrlData, findIfExists })}
|
|
|
|
>
|
|
|
|
Use existing URL if found
|
|
|
|
</Checkbox>
|
|
|
|
<UseExistingIfFoundInfoIcon />
|
|
|
|
</p>
|
|
|
|
)}
|
|
|
|
</SimpleCard>
|
|
|
|
</div>
|
2022-12-23 21:06:59 +01:00
|
|
|
<div className="col-sm-6 mb-3">
|
|
|
|
<SimpleCard title="Configure behavior">
|
|
|
|
<ShortUrlFormCheckboxGroup
|
|
|
|
infoTooltip="This short URL will be included in the robots.txt for your Shlink instance, allowing web crawlers (like Google) to index it."
|
|
|
|
checked={shortUrlData.crawlable}
|
|
|
|
onChange={(crawlable) => setShortUrlData({ ...shortUrlData, crawlable })}
|
|
|
|
>
|
|
|
|
Make it crawlable
|
|
|
|
</ShortUrlFormCheckboxGroup>
|
2023-07-24 18:10:22 +02:00
|
|
|
<ShortUrlFormCheckboxGroup
|
|
|
|
infoTooltip="When this short URL is visited, any query params appended to it will be forwarded to the long URL."
|
|
|
|
checked={shortUrlData.forwardQuery}
|
|
|
|
onChange={(forwardQuery) => setShortUrlData({ ...shortUrlData, forwardQuery })}
|
|
|
|
>
|
|
|
|
Forward query params on redirect
|
|
|
|
</ShortUrlFormCheckboxGroup>
|
2022-12-23 21:06:59 +01:00
|
|
|
</SimpleCard>
|
|
|
|
</div>
|
2021-10-13 22:50:48 +02:00
|
|
|
</Row>
|
2021-03-19 19:11:27 +01:00
|
|
|
</>
|
|
|
|
)}
|
|
|
|
|
|
|
|
<div className="text-center">
|
|
|
|
<Button
|
|
|
|
outline
|
|
|
|
color="primary"
|
|
|
|
disabled={saving || isEmpty(shortUrlData.longUrl)}
|
|
|
|
className="btn-xs-block"
|
|
|
|
>
|
2021-03-20 16:32:12 +01:00
|
|
|
{saving ? 'Saving...' : 'Save'}
|
2021-03-19 19:11:27 +01:00
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</form>
|
|
|
|
);
|
|
|
|
};
|