CHANGE knob/Date to a native datepicker

This commit is contained in:
Norbert de Langen 2018-10-12 11:06:28 +02:00
parent fb774e0c7a
commit 825d4ec53b
No known key found for this signature in database
GPG Key ID: 976651DA156C2825
6 changed files with 143 additions and 255 deletions

View File

@ -35,7 +35,6 @@
"prop-types": "^15.6.2",
"qs": "^6.5.2",
"react-color": "^2.14.1",
"react-datetime": "^2.15.0",
"react-lifecycles-compat": "^3.0.4",
"util-deprecate": "^1.0.2"
},

View File

@ -0,0 +1,143 @@
import React, { Component } from 'react';
import PropTypes from 'prop-types';
const styles = {
display: 'table-cell',
boxSizing: 'border-box',
verticalAlign: 'middle',
height: '25px',
width: '100%',
outline: 'none',
border: '1px solid #f7f4f4',
borderRadius: 2,
fontSize: 11,
padding: '5px',
color: '#444',
flex: 1,
};
const formatDate = date => {
const year = `000${date.getFullYear()}`.slice(-4);
const month = `0${date.getMonth() + 1}`.slice(-2);
const day = `0${date.getDate()}`.slice(-2);
return `${year}-${month}-${day}`;
};
const formatTime = date => {
const hours = `0${date.getHours()}`.slice(-2);
const minutes = `0${date.getMinutes()}`.slice(-2);
return `${hours}:${minutes}`;
};
class DateType extends Component {
static getDerivedStateFromProps() {
console.log('should be true');
return { valid: true };
}
state = {
valid: undefined,
};
componentDidUpdate() {
const { knob } = this.props;
const { valid } = this.state;
const value = new Date(knob.value);
if (valid !== false) {
this.dateInput.value = formatDate(value);
this.timeInput.value = formatTime(value);
}
}
onDateChange = e => {
const { knob, onChange } = this.props;
const { state } = this;
let valid = false;
const [year, month, day] = e.target.value.split('-');
const result = new Date(knob.value);
if (result.getTime()) {
result.setFullYear(parseInt(year, 10));
result.setMonth(parseInt(month, 10) - 1);
result.setDate(parseInt(day, 10));
if (result.getTime()) {
valid = true;
onChange(result.getTime());
}
}
if (valid !== state.valid) {
this.setState({ valid });
}
};
onTimeChange = e => {
const { knob, onChange } = this.props;
const { state } = this;
let valid = false;
const [hours, minutes] = e.target.value.split(':');
const result = new Date(knob.value);
if (result.getTime()) {
result.setHours(parseInt(hours, 10));
result.setMinutes(parseInt(minutes, 10));
if (result.getTime()) {
onChange(result.getTime());
valid = true;
}
}
if (valid !== state.valid) {
this.setState({ valid });
}
};
render() {
const { knob } = this.props;
const { name } = knob;
const { valid } = this.state;
return name ? (
<div style={{ display: 'flex' }}>
<input
style={styles}
type="date"
max="9999-12-31" // I do this because of a rendering bug in chrome
ref={el => {
this.dateInput = el;
}}
id={`${name}date`}
onChange={this.onDateChange}
/>
<input
style={styles}
type="time"
id={`${name}time`}
ref={el => {
this.timeInput = el;
}}
onChange={this.onTimeChange}
/>
{!valid ? <div>invalid</div> : null}
</div>
) : null;
}
}
DateType.defaultProps = {
knob: {},
onChange: value => value,
};
DateType.propTypes = {
knob: PropTypes.shape({
name: PropTypes.string,
value: PropTypes.number,
}),
onChange: PropTypes.func,
};
DateType.serialize = value => new Date(value).getTime() || new Date().getTime();
DateType.deserialize = value => new Date(value).getTime() || new Date().getTime();
export default DateType;

View File

@ -1,50 +0,0 @@
import PropTypes from 'prop-types';
import React from 'react';
import styled from '@emotion/styled';
import Datetime from 'react-datetime';
import style from './styles';
const DateInput = styled(Datetime)(style);
class DateType extends React.Component {
shouldComponentUpdate(nextProps) {
const { knob } = this.props;
return nextProps.knob.value !== knob.value;
}
handleChange = date => {
const { onChange } = this.props;
const value = date.valueOf();
onChange(value);
};
render() {
const { knob } = this.props;
return (
<DateInput
value={knob.value ? new Date(knob.value) : null}
type="date"
onChange={this.handleChange}
size="flex"
/>
);
}
}
DateType.propTypes = {
knob: PropTypes.shape({
name: PropTypes.string,
value: PropTypes.number,
}).isRequired,
onChange: PropTypes.func.isRequired,
};
DateType.serialize = value => String(value);
DateType.deserialize = value => parseFloat(value);
export default DateType;

View File

@ -1,202 +0,0 @@
import { Input } from '@storybook/components';
export default ({ theme, size }) => ({
...Input.sizes({ size, theme }),
'&.rdt': {
position: 'relative',
},
'& > input': {
width: '100%',
height: 32,
...Input.styles({ theme }),
...Input.alignment({ theme }),
},
'& .rdtPicker': {
display: 'none',
position: 'absolute',
width: 200,
padding: 4,
marginTop: 1,
zIndex: 99999,
background: theme.barFill,
},
'&.rdtOpen .rdtPicker': {
display: 'block',
},
'&.rdt .rdtPicker': {
boxShadow: 'none',
position: 'static',
},
'& .rdtPicker .rdtTimeToggle': {
textAlign: 'center',
fontSize: 11,
},
'& .rdtPicker table': {
width: '100%',
margin: 0,
},
'& .rdtPicker td, & .rdtPicker th': {
textAlign: 'center',
height: 32,
boxSizing: 'border-box',
},
'& .rdtPicker td': {
cursor: 'pointer',
},
'& .rdtPicker td.rdtDay:hover, & .rdtPicker td.rdtHour:hover, & .rdtPicker td.rdtMinute:hover, & .rdtPicker td.rdtSecond:hover, & .rdtPicker .rdtTimeToggle:hover': {
color: theme.highlightColor,
textDecoration: 'underline',
cursor: 'pointer',
},
'& .rdtPicker td.rdtOld, & .rdtPicker td.rdtNew': {
color: '#999999',
},
'& .rdtPicker td.rdtToday': {
position: 'relative',
},
'& .rdtPicker td.rdtToday:before': {
content: '""',
display: 'inline-block',
borderLeft: '7px solid transparent',
borderBottom: `7px solid ${theme.highlightColor}`,
borderTopColor: 'rgba(0, 0, 0, 0.2)',
position: 'absolute',
bottom: 4,
right: 4,
},
'& .rdtPicker td.rdtActive, & .rdtPicker td.rdtActive:hover': {
backgroundColor: theme.highlightColor,
color: '#fff',
textShadow:
'0 -1px 0 rgba(0,0,0,0.25), 0 1px 0 rgba(0,0,0,0.25), -1px 0 0 rgba(0,0,0,0.25), 1px 0 0 rgba(0,0,0,0.25)',
},
'& .rdtPicker td.rdtActive.rdtToday:before': {
borderBottomColor: '#fff',
},
'& .rdtPicker td.rdtDisabled, & .rdtPicker td.rdtDisabled:hover': {
background: 'none',
color: '#999999',
cursor: 'not-allowed',
},
'& .rdtPicker td span.rdtOld': {
color: '#999999',
},
'& .rdtPicker td span.rdtDisabled, & .rdtPicker td span.rdtDisabled:hover': {
background: 'none',
color: '#999999',
cursor: 'not-allowed',
},
'& .rdtPicker th': {
borderBottom: `1px solid ${theme.highlightColor}`,
},
'& .rdtPicker .dow': {
width: '14.2857%',
fontSize: 11,
borderBottom: 'none',
},
'& .rdtPicker th.rdtSwitch': {
width: 100,
fontSize: 11,
},
'& .rdtPicker th.rdtNext, & .rdtPicker th.rdtPrev': {
fontSize: 11,
verticalAlign: 'middle',
},
'& .rdtPrev span, & .rdtNext span': {
display: 'block',
userSelect: 'none',
},
'& .rdtPicker th.rdtDisabled, & .rdtPicker th.rdtDisabled:hover': {
background: 'none',
color: '#999999',
cursor: 'not-allowed',
},
'& .rdtPicker thead tr:first-child th': {
cursor: 'pointer',
},
'& .rdtPicker thead tr:first-child th:hover': {
color: theme.highlightColor,
},
'& .rdtPicker tfoot': {
borderTop: '1px solid #f9f9f9',
},
'& .rdtPicker button': {
border: 'none',
background: 'none',
cursor: 'pointer',
},
'& .rdtPicker button:hover': {
color: theme.highlightColor,
},
'& .rdtPicker thead button': {
width: '100%',
height: '100%',
},
'& td.rdtMonth, & td.rdtYear': {
height: 50,
width: '25%',
cursor: 'pointer',
},
'& td.rdtMonth:hover, & td.rdtYear:hover': {
color: theme.highlightColor,
},
'& td.rdtDay': {
fontSize: 11,
},
'& .rdtCounters': {
display: 'inline-block',
},
'& .rdtCounters > div': {
float: 'left',
},
'& .rdtCounter': {
height: 100,
width: 40,
},
'& .rdtCounterSeparator': {
lineHeight: '100px',
},
'& .rdtCounter .rdtBtn': {
height: '40%',
lineHeight: '40px',
cursor: 'pointer',
display: 'block',
fontSize: 11,
userSelect: 'none',
},
'& .rdtCounter .rdtBtn:hover': {
color: theme.highlightColor,
},
'& .rdtCounter .rdtCount': {
height: '20%',
fontSize: 11,
},
'& .rdtMilli': {
verticalSlign: 'middle',
paddingLeft: 8,
width: 48,
},
'& .rdtMilli input': {
width: '100%',
fontSize: 11,
marginTop: 37,
},
});

View File

@ -31,7 +31,6 @@
"@storybook/core": "4.0.0-alpha.24",
"common-tags": "^1.8.0",
"global": "^4.3.2",
"moment": "^2.22.2",
"raw-loader": "^0.5.1",
"react": "^16.5.2",
"react-dom": "^16.5.2"

View File

@ -18,7 +18,6 @@
"style-loader": "^0.22.1",
"css-loader": "^1.0.0",
"raw-loader": "^0.5.1",
"moment": "^2.22.2",
"riot-hot-reload": "^1.0.0",
"riot-compiler": "^3.5.1",
"storybook-addon-jsx": "latest",