kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
177 lines (159 loc) • 6.05 kB
JavaScript
// Copyright (c) 2020 Uber Technologies, Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
import React, {PureComponent} from 'react';
import styled from 'styled-components';
import moment from 'moment';
import {createSelector} from 'reselect';
import {Minus} from 'components/common/icons';
import {DEFAULT_TIME_FORMAT} from 'constants/default-settings';
import {CenterFlexbox} from 'components/common/styled-components';
const StyledTimeDisplay = styled.div`
background-color: ${props => props.theme.panelBackground};
border-radius: ${props => props.theme.timeDisplayBorderRadius}px;
bottom: ${props => `calc(100% + ${props.theme.bottomPanelGap}px)`};
color: ${props => props.theme.titleTextColor};
display: flex;
height: ${props => props.theme.timeDisplayHeight}px;
justify-content: center;
left: calc(50% - 88px);
min-width: ${props => props.theme.timeDisplayMinWidth}px;
opacity: ${props => props.theme.timeDisplayOpacity};
padding: ${props => props.theme.timeDisplayPadding};
position: absolute;
`;
const StyledTimeDisplayGroups = styled.div`
align-items: center;
display: flex;
flex-direction: row;
`;
const StyledTimeDisplayRows = styled.div`
display: flex;
flex-direction: column;
justify-content: center;
`;
const StyledTimeDisplayTop = styled.div.attrs({
className: 'animation-control__time-display__top'
})`
color: ${props => props.theme.textColor};
display: flex;
font-size: 12px;
font-weight: 500;
justify-content: center;
`;
const StyledTimeDisplayBottom = styled.div.attrs({
className: 'animation-control__time-display__bottom'
})`
color: ${props => props.theme.titleTextColor};
display: flex;
font-size: 14px;
font-weight: 500;
justify-content: center;
`;
const StyledTimeValueGroup = styled.div.attrs({
className: 'animation-control__time-value-group'
})`
display: flex;
flex-direction: column;
`;
const StyledHorizontalBar = styled.div.attrs({
className: 'animation-control__horizontal-bar'
})`
margin: 0 12px;
`;
const TimeDivider = () => (
<StyledHorizontalBar>
<Minus height="12px" />
</StyledHorizontalBar>
);
const TimeDisplayRow = ({timeValues = []}) => (
<CenterFlexbox>
<div>{timeValues[0]}</div>
{timeValues[1] ? <TimeDivider /> : null}
{timeValues[1] ? <div>{timeValues[1]}</div> : null}
</CenterFlexbox>
);
export default function FloatingTimeDisplayFactory() {
class FloatingTimeDisplay extends PureComponent {
timeSelector = props => props.currentTime;
formatSelector = props => props.format;
displayTimeSelector = createSelector(
this.timeSelector,
this.formatSelector,
(currentTime, format) => {
const groupTime = Array.isArray(currentTime) ? currentTime : [currentTime];
return groupTime.reduce(
(accu, curr) => {
const displayDateTime = moment.utc(curr).format(format);
const [displayDate, displayTime] = displayDateTime.split(' ');
if (!accu.displayDate.includes(displayDate)) {
accu.displayDate.push(displayDate);
}
accu.displayTime.push(displayTime);
return accu;
},
{displayDate: [], displayTime: []}
);
}
);
render() {
const {displayDate, displayTime} = this.displayTimeSelector(this.props);
const twoGroups = displayDate.length === 2 && displayTime.length === 2;
const bottomRow = displayTime.length ? displayTime : displayDate.length ? displayDate : null;
const topRow = displayDate.length && displayTime.length ? displayDate : null;
return (
<StyledTimeDisplay className="animation-control__time-display">
{twoGroups ? (
<StyledTimeDisplayGroups>
<StyledTimeValueGroup>
{/* Time Start */}
<StyledTimeDisplayTop>{displayDate[0]}</StyledTimeDisplayTop>
<StyledTimeDisplayBottom>{displayTime[0]}</StyledTimeDisplayBottom>
</StyledTimeValueGroup>
<TimeDivider />
<StyledTimeValueGroup>
{/* Time End */}
<StyledTimeDisplayTop>{displayDate[1]}</StyledTimeDisplayTop>
<StyledTimeDisplayBottom>{displayTime[1]}</StyledTimeDisplayBottom>
</StyledTimeValueGroup>
</StyledTimeDisplayGroups>
) : (
<StyledTimeDisplayRows>
{topRow ? (
<StyledTimeDisplayTop>
<TimeDisplayRow timeValues={topRow} />
</StyledTimeDisplayTop>
) : null}
{bottomRow ? (
<StyledTimeDisplayBottom>
<TimeDisplayRow timeValues={bottomRow} />
</StyledTimeDisplayBottom>
) : null}
</StyledTimeDisplayRows>
)}
</StyledTimeDisplay>
);
}
}
FloatingTimeDisplay.defaultProps = {
format: DEFAULT_TIME_FORMAT,
currentTime: null
};
return FloatingTimeDisplay;
}