| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301 |
- import React, { PureComponent } from 'react';
- import * as rangeUtil from '@grafana/ui/src/utils/rangeutil';
- import { Input, RawTimeRange, TimeRange, TIME_FORMAT } from '@grafana/ui';
- import { toUtc, isDateTime, dateTime } from '@grafana/ui/src/utils/moment_wrapper';
- interface TimePickerProps {
- isOpen?: boolean;
- isUtc?: boolean;
- range: TimeRange;
- onChangeTime?: (range: RawTimeRange, scanning?: boolean) => void;
- }
- interface TimePickerState {
- isOpen: boolean;
- isUtc: boolean;
- rangeString: string;
- refreshInterval?: string;
- initialRange: RawTimeRange;
- // Input-controlled text, keep these in a shape that is human-editable
- fromRaw: string;
- toRaw: string;
- }
- const getRaw = (isUtc: boolean, range: any) => {
- const rawRange = {
- from: range.raw.from,
- to: range.raw.to,
- };
- if (isDateTime(rawRange.from)) {
- if (!isUtc) {
- rawRange.from = rawRange.from.local();
- }
- rawRange.from = rawRange.from.format(TIME_FORMAT);
- }
- if (isDateTime(rawRange.to)) {
- if (!isUtc) {
- rawRange.to = rawRange.to.local();
- }
- rawRange.to = rawRange.to.format(TIME_FORMAT);
- }
- return rawRange;
- };
- /**
- * TimePicker with dropdown menu for relative dates.
- *
- * Initialize with a range that is either based on relative rawRange.strings,
- * or on Moment objects.
- * Internally the component needs to keep a string representation in `fromRaw`
- * and `toRaw` for the controlled inputs.
- * When a time is picked, `onChangeTime` is called with the new range that
- * is again based on relative time strings or Moment objects.
- */
- export default class TimePicker extends PureComponent<TimePickerProps, TimePickerState> {
- dropdownEl: any;
- constructor(props) {
- super(props);
- const { range, isUtc, isOpen } = props;
- const rawRange = getRaw(props.isUtc, range);
- this.state = {
- isOpen: isOpen,
- isUtc: isUtc,
- rangeString: rangeUtil.describeTimeRange(range.raw),
- fromRaw: rawRange.from,
- toRaw: rawRange.to,
- initialRange: range.raw,
- refreshInterval: '',
- };
- } //Temp solution... How do detect if ds supports table format?
- static getDerivedStateFromProps(props: TimePickerProps, state: TimePickerState) {
- if (
- state.initialRange &&
- state.initialRange.from === props.range.raw.from &&
- state.initialRange.to === props.range.raw.to
- ) {
- return state;
- }
- const { range } = props;
- const rawRange = getRaw(props.isUtc, range);
- return {
- ...state,
- fromRaw: rawRange.from,
- toRaw: rawRange.to,
- initialRange: range.raw,
- rangeString: rangeUtil.describeTimeRange(range.raw),
- };
- }
- move(direction: number, scanning?: boolean): RawTimeRange {
- const { onChangeTime, range: origRange } = this.props;
- const range = {
- from: toUtc(origRange.from),
- to: toUtc(origRange.to),
- };
- const timespan = (range.to.valueOf() - range.from.valueOf()) / 2;
- let to, from;
- if (direction === -1) {
- to = range.to.valueOf() - timespan;
- from = range.from.valueOf() - timespan;
- } else if (direction === 1) {
- to = range.to.valueOf() + timespan;
- from = range.from.valueOf() + timespan;
- } else {
- to = range.to.valueOf();
- from = range.from.valueOf();
- }
- const nextTimeRange = {
- from: this.props.isUtc ? toUtc(from) : dateTime(from),
- to: this.props.isUtc ? toUtc(to) : dateTime(to),
- };
- if (onChangeTime) {
- onChangeTime(nextTimeRange);
- }
- return nextTimeRange;
- }
- handleChangeFrom = e => {
- this.setState({
- fromRaw: e.target.value,
- });
- };
- handleChangeTo = e => {
- this.setState({
- toRaw: e.target.value,
- });
- };
- handleClickApply = () => {
- const { onChangeTime, isUtc } = this.props;
- let rawRange;
- this.setState(
- state => {
- const { toRaw, fromRaw } = this.state;
- rawRange = {
- from: fromRaw,
- to: toRaw,
- };
- if (rawRange.from.indexOf('now') === -1) {
- rawRange.from = isUtc ? toUtc(rawRange.from, TIME_FORMAT) : dateTime(rawRange.from, TIME_FORMAT);
- }
- if (rawRange.to.indexOf('now') === -1) {
- rawRange.to = isUtc ? toUtc(rawRange.to, TIME_FORMAT) : dateTime(rawRange.to, TIME_FORMAT);
- }
- const rangeString = rangeUtil.describeTimeRange(rawRange);
- return {
- isOpen: false,
- rangeString,
- };
- },
- () => {
- if (onChangeTime) {
- onChangeTime(rawRange);
- }
- }
- );
- };
- handleClickLeft = () => this.move(-1);
- handleClickPicker = () => {
- this.setState(state => ({
- isOpen: !state.isOpen,
- }));
- };
- handleClickRight = () => this.move(1);
- handleClickRefresh = () => {};
- handleClickRelativeOption = range => {
- const { onChangeTime } = this.props;
- const rangeString = rangeUtil.describeTimeRange(range);
- const rawRange = {
- from: range.from,
- to: range.to,
- };
- this.setState(
- {
- toRaw: rawRange.to,
- fromRaw: rawRange.from,
- isOpen: false,
- rangeString,
- },
- () => {
- if (onChangeTime) {
- onChangeTime(rawRange);
- }
- }
- );
- };
- getTimeOptions() {
- return rangeUtil.getRelativeTimesList({}, this.state.rangeString);
- }
- dropdownRef = el => {
- this.dropdownEl = el;
- };
- renderDropdown() {
- const { fromRaw, isOpen, toRaw } = this.state;
- if (!isOpen) {
- return null;
- }
- const timeOptions = this.getTimeOptions();
- return (
- <div ref={this.dropdownRef} className="gf-timepicker-dropdown">
- <div className="popover-box">
- <div className="popover-box__header">
- <span className="popover-box__title">Quick ranges</span>
- </div>
- <div className="popover-box__body gf-timepicker-relative-section">
- {Object.keys(timeOptions).map(section => {
- const group = timeOptions[section];
- return (
- <ul key={section}>
- {group.map((option: any) => (
- <li className={option.active ? 'active' : ''} key={option.display}>
- <a onClick={() => this.handleClickRelativeOption(option)}>{option.display}</a>
- </li>
- ))}
- </ul>
- );
- })}
- </div>
- </div>
- <div className="popover-box">
- <div className="popover-box__header">
- <span className="popover-box__title">Custom range</span>
- </div>
- <div className="popover-box__body gf-timepicker-absolute-section">
- <label className="small">From:</label>
- <div className="gf-form-inline">
- <div className="gf-form max-width-28">
- <Input
- type="text"
- className="gf-form-input input-large timepicker-from"
- value={fromRaw}
- onChange={this.handleChangeFrom}
- />
- </div>
- </div>
- <label className="small">To:</label>
- <div className="gf-form-inline">
- <div className="gf-form max-width-28">
- <Input
- type="text"
- className="gf-form-input input-large timepicker-to"
- value={toRaw}
- onChange={this.handleChangeTo}
- />
- </div>
- </div>
- <div className="gf-form">
- <button className="btn gf-form-btn btn-secondary" onClick={this.handleClickApply}>
- Apply
- </button>
- </div>
- </div>
- </div>
- </div>
- );
- }
- render() {
- const { isUtc, rangeString, refreshInterval } = this.state;
- return (
- <div className="timepicker">
- <div className="navbar-buttons">
- <button className="btn navbar-button navbar-button--tight timepicker-left" onClick={this.handleClickLeft}>
- <i className="fa fa-chevron-left" />
- </button>
- <button className="btn navbar-button gf-timepicker-nav-btn" onClick={this.handleClickPicker}>
- <i className="fa fa-clock-o" />
- <span className="timepicker-rangestring">{rangeString}</span>
- {isUtc ? <span className="gf-timepicker-utc">UTC</span> : null}
- {refreshInterval ? <span className="text-warning"> Refresh every {refreshInterval}</span> : null}
- </button>
- <button className="btn navbar-button navbar-button--tight timepicker-right" onClick={this.handleClickRight}>
- <i className="fa fa-chevron-right" />
- </button>
- </div>
- {this.renderDropdown()}
- </div>
- );
- }
- }
|