Files
react/packages/react-devtools-shared/src/devtools/views/Profiler/SnapshotCommitListItem.js
Brian Vaughn cb88572227 Improve DevTools Profiler commit-selector UX (#20943)
* Improve DevTools Profiler commit-selector UX

1. Use natural log of durations (rather than linear) when calculating bar height. This reduces the impact of one (or few) outlier times on more common smaller durations. (Continue to use linear for bar color though.)
2. Decrease the minimum bar height to make the differences in height more noticeable.
3. Add a background hover highlight to increase contrast.
4. Add hover tooltip with commit duration and timestamp.
2021-03-08 11:09:55 -05:00

102 lines
2.6 KiB
JavaScript

/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
* @flow
*/
import * as React from 'react';
import {memo} from 'react';
import {areEqual} from 'react-window';
import {getGradientColor, formatDuration, formatTime} from './utils';
import styles from './SnapshotCommitListItem.css';
import type {ItemData} from './SnapshotCommitList';
type Props = {
data: ItemData,
index: number,
style: Object,
...
};
function SnapshotCommitListItem({data: itemData, index, style}: Props) {
const {
commitDurations,
commitTimes,
filteredCommitIndices,
maxDuration,
selectedCommitIndex,
selectCommitIndex,
setHoveredCommitIndex,
startCommitDrag,
} = itemData;
index = filteredCommitIndices[index];
const commitDuration = commitDurations[index];
const commitTime = commitTimes[index];
// Use natural log for bar height.
// This prevents one (or a few) outliers from squishing the majority of other commits.
// So rather than e.g. _█_ we get something more like e.g. ▄█_
const heightScale =
Math.min(
1,
Math.max(0, Math.log(commitDuration) / Math.log(maxDuration)),
) || 0;
// Use a linear scale for color.
// This gives some visual contrast between cheaper and more expensive commits
// and somewhat compensates for the log scale height.
const colorScale =
Math.min(1, Math.max(0, commitDuration / maxDuration)) || 0;
const isSelected = selectedCommitIndex === index;
// Leave a 1px gap between snapshots
const width = parseFloat(style.width) - 1;
const handleMouseDown = ({buttons, target}: any) => {
if (buttons === 1) {
selectCommitIndex(index);
startCommitDrag({
commitIndex: index,
left: target.getBoundingClientRect().left,
sizeIncrement: parseFloat(style.width),
});
}
};
return (
<div
className={styles.Outer}
onMouseDown={handleMouseDown}
onMouseEnter={() => setHoveredCommitIndex(index)}
style={{
...style,
width,
borderBottom: isSelected
? '3px solid var(--color-tab-selected-border)'
: undefined,
}}
title={`Duration ${formatDuration(commitDuration)}ms at ${formatTime(
commitTime,
)}s`}>
<div
className={styles.Inner}
style={{
height: `${Math.round(heightScale * 100)}%`,
backgroundColor:
commitDuration > 0 ? getGradientColor(colorScale) : undefined,
}}
/>
</div>
);
}
export default memo<Props>(SnapshotCommitListItem, areEqual);