Converted CreateShortUrl to functional component

This commit is contained in:
Alejandro Celaya 2020-03-29 19:36:45 +02:00
parent bd29670108
commit 74ebd4e572
4 changed files with 85 additions and 102 deletions

20
package-lock.json generated
View file

@ -13894,9 +13894,9 @@
} }
}, },
"react": { "react": {
"version": "16.10.2", "version": "16.13.1",
"resolved": "https://registry.npmjs.org/react/-/react-16.10.2.tgz", "resolved": "https://registry.npmjs.org/react/-/react-16.13.1.tgz",
"integrity": "sha512-MFVIq0DpIhrHFyqLU0S3+4dIcBhhOvBE8bJ/5kHPVOVaGdo0KuiQzpcjCPsf585WvhypqtrMILyoE2th6dT+Lw==", "integrity": "sha512-YMZQQq32xHLX0bz5Mnibv1/LHb3Sqzngu7xstSM+vrkE5Kzr9xE0yMByK5kMoTK30YVJE61WfbxIFFvfeDKT1w==",
"requires": { "requires": {
"loose-envify": "^1.1.0", "loose-envify": "^1.1.0",
"object-assign": "^4.1.1", "object-assign": "^4.1.1",
@ -14106,14 +14106,14 @@
} }
}, },
"react-dom": { "react-dom": {
"version": "16.10.2", "version": "16.13.1",
"resolved": "https://registry.npmjs.org/react-dom/-/react-dom-16.10.2.tgz", "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-16.13.1.tgz",
"integrity": "sha512-kWGDcH3ItJK4+6Pl9DZB16BXYAZyrYQItU4OMy0jAkv5aNqc+mAKb4TpFtAteI6TJZu+9ZlNhaeNQSVQDHJzkw==", "integrity": "sha512-81PIMmVLnCNLO/fFOQxdQkvEq/+Hfpv24XNJfpyZhTRfO0QcmQIF/PgCa1zCOj2w1hrn12MFLyaJ/G0+Mxtfag==",
"requires": { "requires": {
"loose-envify": "^1.1.0", "loose-envify": "^1.1.0",
"object-assign": "^4.1.1", "object-assign": "^4.1.1",
"prop-types": "^15.6.2", "prop-types": "^15.6.2",
"scheduler": "^0.16.2" "scheduler": "^0.19.1"
} }
}, },
"react-error-overlay": { "react-error-overlay": {
@ -15279,9 +15279,9 @@
"dev": true "dev": true
}, },
"scheduler": { "scheduler": {
"version": "0.16.2", "version": "0.19.1",
"resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.16.2.tgz", "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.19.1.tgz",
"integrity": "sha512-BqYVWqwz6s1wZMhjFvLfVR5WXP7ZY32M/wYPo04CcuPM7XZEbV2TBNW7Z0UkguPTl0dWMA59VbNXxK6q+pHItg==", "integrity": "sha512-n/zwRWRYSUj0/3g/otKDRPMh6qv2SYMWNq85IEa8iZyAv8od9zDYpGSnpBEjNgcMNq6Scbu5KfIPxNF72R/2EA==",
"requires": { "requires": {
"loose-envify": "^1.1.0", "loose-envify": "^1.1.0",
"object-assign": "^4.1.1" "object-assign": "^4.1.1"

View file

@ -43,13 +43,13 @@
"prop-types": "^15.7.2", "prop-types": "^15.7.2",
"qs": "^6.9.0", "qs": "^6.9.0",
"ramda": "^0.26.1", "ramda": "^0.26.1",
"react": "^16.10.2", "react": "^16.13.1",
"react-autosuggest": "^9.4.3", "react-autosuggest": "^9.4.3",
"react-chartjs-2": "^2.8.0", "react-chartjs-2": "^2.8.0",
"react-color": "^2.17.3", "react-color": "^2.17.3",
"react-copy-to-clipboard": "^5.0.1", "react-copy-to-clipboard": "^5.0.1",
"react-datepicker": "~1.5.0", "react-datepicker": "~1.5.0",
"react-dom": "^16.10.2", "react-dom": "^16.13.1",
"react-external-link": "^1.0.0", "react-external-link": "^1.0.0",
"react-leaflet": "^2.4.0", "react-leaflet": "^2.4.0",
"react-moment": "^0.9.5", "react-moment": "^0.9.5",

View file

@ -1,7 +1,7 @@
import { faAngleDoubleDown as downIcon, faAngleDoubleUp as upIcon } from '@fortawesome/free-solid-svg-icons'; import { faAngleDoubleDown as downIcon, faAngleDoubleUp as upIcon } from '@fortawesome/free-solid-svg-icons';
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
import { assoc, dissoc, isEmpty, isNil, pipe, replace, trim } from 'ramda'; import { isEmpty, isNil, pipe, replace, trim } from 'ramda';
import React from 'react'; import React, { useState } from 'react';
import { Collapse, FormGroup, Input } from 'reactstrap'; import { Collapse, FormGroup, Input } from 'reactstrap';
import * as PropTypes from 'prop-types'; import * as PropTypes from 'prop-types';
import DateInput from '../utils/DateInput'; import DateInput from '../utils/DateInput';
@ -9,25 +9,23 @@ import Checkbox from '../utils/Checkbox';
import { serverType } from '../servers/prop-types'; import { serverType } from '../servers/prop-types';
import { versionMatch } from '../utils/helpers/version'; import { versionMatch } from '../utils/helpers/version';
import { hasValue } from '../utils/utils'; import { hasValue } from '../utils/utils';
import { useToggle } from '../utils/helpers/hooks';
import { createShortUrlResultType } from './reducers/shortUrlCreation'; import { createShortUrlResultType } from './reducers/shortUrlCreation';
import UseExistingIfFoundInfoIcon from './UseExistingIfFoundInfoIcon'; import UseExistingIfFoundInfoIcon from './UseExistingIfFoundInfoIcon';
const normalizeTag = pipe(trim, replace(/ /g, '-')); const normalizeTag = pipe(trim, replace(/ /g, '-'));
const formatDate = (date) => isNil(date) ? date : date.format(); const formatDate = (date) => isNil(date) ? date : date.format();
const CreateShortUrl = ( const propTypes = {
TagsSelector,
CreateShortUrlResult,
ForServerVersion
) => class CreateShortUrl extends React.Component {
static propTypes = {
createShortUrl: PropTypes.func, createShortUrl: PropTypes.func,
shortUrlCreationResult: createShortUrlResultType, shortUrlCreationResult: createShortUrlResultType,
resetCreateShortUrl: PropTypes.func, resetCreateShortUrl: PropTypes.func,
selectedServer: serverType, selectedServer: serverType,
}; };
state = { const CreateShortUrl = (TagsSelector, CreateShortUrlResult, ForServerVersion) => {
const CreateShortUrlComp = ({ createShortUrl, shortUrlCreationResult, resetCreateShortUrl, selectedServer }) => {
const [ shortUrlCreation, setShortUrlCreation ] = useState({
longUrl: '', longUrl: '',
tags: [], tags: [],
customSlug: undefined, customSlug: undefined,
@ -37,21 +35,18 @@ const CreateShortUrl = (
validUntil: undefined, validUntil: undefined,
maxVisits: undefined, maxVisits: undefined,
findIfExists: false, findIfExists: false,
moreOptionsVisible: false, });
}; const [ moreOptionsVisible, toggleMoreOptionsVisible ] = useToggle(false);
render() { const changeTags = (tags) => setShortUrlCreation({ ...shortUrlCreation, tags: tags.map(normalizeTag) });
const { createShortUrl, shortUrlCreationResult, resetCreateShortUrl } = this.props;
const changeTags = (tags) => this.setState({ tags: tags.map(normalizeTag) });
const renderOptionalInput = (id, placeholder, type = 'text', props = {}) => ( const renderOptionalInput = (id, placeholder, type = 'text', props = {}) => (
<FormGroup> <FormGroup>
<Input <Input
id={id} id={id}
type={type} type={type}
placeholder={placeholder} placeholder={placeholder}
value={this.state[id]} value={shortUrlCreation[id]}
onChange={(e) => this.setState({ [id]: e.target.value })} onChange={(e) => setShortUrlCreation({ ...shortUrlCreation, [id]: e.target.value })}
{...props} {...props}
/> />
</FormGroup> </FormGroup>
@ -59,23 +54,23 @@ const CreateShortUrl = (
const renderDateInput = (id, placeholder, props = {}) => ( const renderDateInput = (id, placeholder, props = {}) => (
<div className="form-group"> <div className="form-group">
<DateInput <DateInput
selected={this.state[id]} selected={shortUrlCreation[id]}
placeholderText={placeholder} placeholderText={placeholder}
isClearable isClearable
onChange={(date) => this.setState({ [id]: date })} onChange={(date) => setShortUrlCreation({ ...shortUrlCreation, [id]: date })}
{...props} {...props}
/> />
</div> </div>
); );
const save = (e) => { const save = (e) => {
e.preventDefault(); e.preventDefault();
createShortUrl(pipe( createShortUrl({
dissoc('moreOptionsVisible'), ...shortUrlCreation,
assoc('validSince', formatDate(this.state.validSince)), validSince: formatDate(shortUrlCreation.validSince),
assoc('validUntil', formatDate(this.state.validUntil)) validUntil: formatDate(shortUrlCreation.validUntil),
)(this.state)); });
}; };
const currentServerVersion = this.props.selectedServer && this.props.selectedServer.version; const currentServerVersion = selectedServer && selectedServer.version;
const disableDomain = !versionMatch(currentServerVersion, { minVersion: '1.19.0-beta.1' }); const disableDomain = !versionMatch(currentServerVersion, { minVersion: '1.19.0-beta.1' });
const disableShortCodeLength = !versionMatch(currentServerVersion, { minVersion: '2.1.0' }); const disableShortCodeLength = !versionMatch(currentServerVersion, { minVersion: '2.1.0' });
@ -87,14 +82,14 @@ const CreateShortUrl = (
type="url" type="url"
placeholder="Insert the URL to be shortened" placeholder="Insert the URL to be shortened"
required required
value={this.state.longUrl} value={shortUrlCreation.longUrl}
onChange={(e) => this.setState({ longUrl: e.target.value })} onChange={(e) => setShortUrlCreation({ ...shortUrlCreation, longUrl: e.target.value })}
/> />
</div> </div>
<Collapse isOpen={this.state.moreOptionsVisible}> <Collapse isOpen={moreOptionsVisible}>
<div className="form-group"> <div className="form-group">
<TagsSelector tags={this.state.tags} onChange={changeTags} /> <TagsSelector tags={shortUrlCreation.tags} onChange={changeTags} />
</div> </div>
<div className="row"> <div className="row">
@ -104,7 +99,7 @@ const CreateShortUrl = (
<div className="col-sm-4"> <div className="col-sm-4">
{renderOptionalInput('shortCodeLength', 'Short code length', 'number', { {renderOptionalInput('shortCodeLength', 'Short code length', 'number', {
min: 4, min: 4,
disabled: disableShortCodeLength || hasValue(this.state.customSlug), disabled: disableShortCodeLength || hasValue(shortUrlCreation.customSlug),
...disableShortCodeLength && { ...disableShortCodeLength && {
title: 'Shlink 2.1.0 or higher is required to be able to provide the short code length', title: 'Shlink 2.1.0 or higher is required to be able to provide the short code length',
}, },
@ -123,10 +118,10 @@ const CreateShortUrl = (
{renderOptionalInput('maxVisits', 'Maximum number of visits allowed', 'number', { min: 1 })} {renderOptionalInput('maxVisits', 'Maximum number of visits allowed', 'number', { min: 1 })}
</div> </div>
<div className="col-sm-4"> <div className="col-sm-4">
{renderDateInput('validSince', 'Enabled since...', { maxDate: this.state.validUntil })} {renderDateInput('validSince', 'Enabled since...', { maxDate: shortUrlCreation.validUntil })}
</div> </div>
<div className="col-sm-4"> <div className="col-sm-4">
{renderDateInput('validUntil', 'Enabled until...', { minDate: this.state.validSince })} {renderDateInput('validUntil', 'Enabled until...', { minDate: shortUrlCreation.validSince })}
</div> </div>
</div> </div>
@ -134,8 +129,8 @@ const CreateShortUrl = (
<div className="mb-4 text-right"> <div className="mb-4 text-right">
<Checkbox <Checkbox
className="mr-2" className="mr-2"
checked={this.state.findIfExists} checked={shortUrlCreation.findIfExists}
onChange={(findIfExists) => this.setState({ findIfExists })} onChange={(findIfExists) => setShortUrlCreation({ ...shortUrlCreation, findIfExists })}
> >
Use existing URL if found Use existing URL if found
</Checkbox> </Checkbox>
@ -145,18 +140,14 @@ const CreateShortUrl = (
</Collapse> </Collapse>
<div> <div>
<button <button type="button" className="btn btn-outline-secondary" onClick={toggleMoreOptionsVisible}>
type="button" <FontAwesomeIcon icon={moreOptionsVisible ? upIcon : downIcon} />
className="btn btn-outline-secondary"
onClick={() => this.setState(({ moreOptionsVisible }) => ({ moreOptionsVisible: !moreOptionsVisible }))}
>
<FontAwesomeIcon icon={this.state.moreOptionsVisible ? upIcon : downIcon} />
&nbsp; &nbsp;
{this.state.moreOptionsVisible ? 'Less' : 'More'} options {moreOptionsVisible ? 'Less' : 'More'} options
</button> </button>
<button <button
className="btn btn-outline-primary float-right" className="btn btn-outline-primary float-right"
disabled={shortUrlCreationResult.loading || isEmpty(this.state.longUrl)} disabled={shortUrlCreationResult.loading || isEmpty(shortUrlCreation.longUrl)}
> >
{shortUrlCreationResult.loading ? 'Creating...' : 'Create'} {shortUrlCreationResult.loading ? 'Creating...' : 'Create'}
</button> </button>
@ -165,7 +156,11 @@ const CreateShortUrl = (
<CreateShortUrlResult {...shortUrlCreationResult} resetCreateShortUrl={resetCreateShortUrl} /> <CreateShortUrlResult {...shortUrlCreationResult} resetCreateShortUrl={resetCreateShortUrl} />
</form> </form>
); );
} };
CreateShortUrlComp.propTypes = propTypes;
return CreateShortUrlComp;
}; };
export default CreateShortUrl; export default CreateShortUrl;

View file

@ -25,31 +25,20 @@ describe('<CreateShortUrl />', () => {
createShortUrl.mockReset(); createShortUrl.mockReset();
}); });
it('saves short URL with data set in form controls', (done) => { it('saves short URL with data set in form controls', () => {
const validSince = moment('2017-01-01'); const validSince = moment('2017-01-01');
const validUntil = moment('2017-01-06'); const validUntil = moment('2017-01-06');
const urlInput = wrapper.find('.form-control-lg'); wrapper.find('.form-control-lg').simulate('change', { target: { value: 'https://long-domain.com/foo/bar' } });
const tagsInput = wrapper.find(TagsSelector); wrapper.find('TagsSelector').simulate('change', [ 'tag_foo', 'tag_bar' ]);
const customSlugInput = wrapper.find('#customSlug'); wrapper.find('#customSlug').simulate('change', { target: { value: 'my-slug' } });
const domain = wrapper.find('#domain'); wrapper.find('#domain').simulate('change', { target: { value: 'example.com' } });
const maxVisitsInput = wrapper.find('#maxVisits'); wrapper.find('#maxVisits').simulate('change', { target: { value: '20' } });
const dateInputs = wrapper.find(DateInput); wrapper.find('#shortCodeLength').simulate('change', { target: { value: 15 } });
const validSinceInput = dateInputs.at(0); wrapper.find(DateInput).at(0).simulate('change', validSince);
const validUntilInput = dateInputs.at(1); wrapper.find(DateInput).at(1).simulate('change', validUntil);
wrapper.find('form').simulate('submit', { preventDefault: identity });
urlInput.simulate('change', { target: { value: 'https://long-domain.com/foo/bar' } });
tagsInput.simulate('change', [ 'tag_foo', 'tag_bar' ]);
customSlugInput.simulate('change', { target: { value: 'my-slug' } });
domain.simulate('change', { target: { value: 'example.com' } });
maxVisitsInput.simulate('change', { target: { value: '20' } });
validSinceInput.simulate('change', validSince);
validUntilInput.simulate('change', validUntil);
setImmediate(() => {
const form = wrapper.find('form');
form.simulate('submit', { preventDefault: identity });
expect(createShortUrl).toHaveBeenCalledTimes(1); expect(createShortUrl).toHaveBeenCalledTimes(1);
expect(createShortUrl).toHaveBeenCalledWith({ expect(createShortUrl).toHaveBeenCalledWith({
longUrl: 'https://long-domain.com/foo/bar', longUrl: 'https://long-domain.com/foo/bar',
@ -60,8 +49,7 @@ describe('<CreateShortUrl />', () => {
validUntil: validUntil.format(), validUntil: validUntil.format(),
maxVisits: '20', maxVisits: '20',
findIfExists: false, findIfExists: false,
}); shortCodeLength: 15,
done();
}); });
}); });
}); });