shlink-web-client/src/short-urls/CreateShortUrl.js

139 lines
5 KiB
JavaScript
Raw Normal View History

2018-07-21 11:38:54 +03:00
import downIcon from '@fortawesome/fontawesome-free-solid/faAngleDoubleDown';
import upIcon from '@fortawesome/fontawesome-free-solid/faAngleDoubleUp';
import FontAwesomeIcon from '@fortawesome/react-fontawesome';
2018-07-28 19:59:32 +03:00
import { assoc, dissoc, isNil, pick, pipe, pluck, replace } from 'ramda';
import React from 'react';
2018-07-28 19:59:32 +03:00
import { connect } from 'react-redux';
2018-07-21 11:38:54 +03:00
import ReactTags from 'react-tag-autocomplete';
import { Collapse } from 'reactstrap';
import '../../node_modules/react-datepicker/dist/react-datepicker.css';
2018-07-31 22:24:34 +03:00
import DateInput from '../common/DateInput';
import './CreateShortUrl.scss';
import CreateShortUrlResult from './helpers/CreateShortUrlResult';
import { createShortUrl, resetCreateShortUrl } from './reducers/shortUrlCreationResult';
2018-07-20 23:32:50 +03:00
export class CreateShortUrl extends React.Component {
2018-07-21 11:38:54 +03:00
state = {
longUrl: '',
tags: [],
customSlug: undefined,
validSince: undefined,
validUntil: undefined,
maxVisits: undefined,
2018-07-21 11:38:54 +03:00
moreOptionsVisible: false
};
render() {
const { createShortUrl, shortUrlCreationResult, resetCreateShortUrl } = this.props;
2018-07-21 11:38:54 +03:00
const addTag = tag => this.setState({
tags: [].concat(this.state.tags, assoc('name', replace(/ /g, '-', tag.name), tag))
});
const removeTag = i => {
const tags = this.state.tags.slice(0);
tags.splice(i, 1);
this.setState({ tags });
};
const renderOptionalInput = (id, placeholder, type = 'text', props = {}) =>
<input
className="form-control"
type={type}
placeholder={placeholder}
value={this.state[id]}
onChange={e => this.setState({ [id]: e.target.value })}
{...props}
/>;
2018-07-28 10:34:31 +03:00
const createDateInput = (id, placeholder, props = {}) =>
2018-07-31 22:24:34 +03:00
<DateInput
2018-07-28 10:34:31 +03:00
selected={this.state[id]}
placeholderText={placeholder}
onChange={date => this.setState({ [id]: date })}
2018-08-09 21:13:46 +03:00
isClearable
2018-07-28 10:34:31 +03:00
{...props}
/>;
2018-07-28 19:59:32 +03:00
const formatDate = date => isNil(date) ? date : date.format();
const save = e => {
e.preventDefault();
createShortUrl(pipe(
2018-07-28 19:59:32 +03:00
dissoc('moreOptionsVisible'), // Remove moreOptionsVisible property
assoc('tags', pluck('name', this.state.tags)), // Map tags array to use only their names
assoc('validSince', formatDate(this.state.validSince)),
assoc('validUntil', formatDate(this.state.validUntil))
)(this.state));
};
2018-07-21 11:38:54 +03:00
return (
<div className="short-urls-container">
<form onSubmit={save}>
2018-07-21 11:38:54 +03:00
<div className="form-group">
<input
className="form-control form-control-lg"
type="url"
placeholder="Insert the URL to be shortened"
required
value={this.state.longUrl}
onChange={e => this.setState({ longUrl: e.target.value })}
/>
</div>
<Collapse isOpen={this.state.moreOptionsVisible}>
<div className="form-group">
<ReactTags
tags={this.state.tags}
handleAddition={addTag}
handleDelete={removeTag}
allowNew={true}
placeholder="Add tags you want to apply to the URL"
/>
</div>
<div className="row">
<div className="col-sm-6">
<div className="form-group">
{renderOptionalInput('customSlug', 'Custom slug')}
</div>
<div className="form-group">
{renderOptionalInput('maxVisits', 'Maximum number of visits allowed', 'number', { min: 1 })}
</div>
</div>
<div className="col-sm-6">
2018-07-31 22:24:34 +03:00
<div className="form-group">
{createDateInput('validSince', 'Enabled since...', { maxDate: this.state.validUntil })}
2018-07-21 11:38:54 +03:00
</div>
2018-07-31 22:24:34 +03:00
<div className="form-group">
{createDateInput('validUntil', 'Enabled until...', { minDate: this.state.validSince })}
2018-07-21 11:38:54 +03:00
</div>
</div>
</div>
</Collapse>
<div>
<button
type="button"
className="btn btn-outline-secondary create-short-url__btn"
onClick={() => this.setState({ moreOptionsVisible: !this.state.moreOptionsVisible })}
>
<FontAwesomeIcon icon={this.state.moreOptionsVisible ? upIcon : downIcon} />
&nbsp;
{this.state.moreOptionsVisible ? 'Less' : 'More'} options
</button>
<button
className="btn btn-outline-primary create-short-url__btn float-right"
disabled={shortUrlCreationResult.loading}
>
{shortUrlCreationResult.loading ? 'Creating...' : 'Create'}
</button>
2018-07-21 11:38:54 +03:00
</div>
<CreateShortUrlResult {...shortUrlCreationResult} resetCreateShortUrl={resetCreateShortUrl} />
2018-07-21 11:38:54 +03:00
</form>
</div>
);
}
2018-07-20 23:32:50 +03:00
}
export default connect(pick(['shortUrlCreationResult']), {
createShortUrl,
resetCreateShortUrl
})(CreateShortUrl);