123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766 |
- import {Fragment, useCallback, useEffect, useMemo, useRef, useState} from 'react';
- import styled from '@emotion/styled';
- import {mat3, vec2} from 'gl-matrix';
- import space from 'sentry/styles/space';
- import {CallTreeNode} from 'sentry/utils/profiling/callTreeNode';
- import {CanvasPoolManager, CanvasScheduler} from 'sentry/utils/profiling/canvasScheduler';
- import {DifferentialFlamegraph} from 'sentry/utils/profiling/differentialFlamegraph';
- import {Flamegraph} from 'sentry/utils/profiling/flamegraph';
- import {useFlamegraphSearch} from 'sentry/utils/profiling/flamegraph/useFlamegraphSearch';
- import {
- useDispatchFlamegraphState,
- useFlamegraphState,
- } from 'sentry/utils/profiling/flamegraph/useFlamegraphState';
- import {useFlamegraphTheme} from 'sentry/utils/profiling/flamegraph/useFlamegraphTheme';
- import {FlamegraphCanvas} from 'sentry/utils/profiling/flamegraphCanvas';
- import {FlamegraphView} from 'sentry/utils/profiling/flamegraphView';
- import {formatColorForFrame, Rect} from 'sentry/utils/profiling/gl/utils';
- import {useContextMenu} from 'sentry/utils/profiling/hooks/useContextMenu';
- import {useInternalFlamegraphDebugMode} from 'sentry/utils/profiling/hooks/useInternalFlamegraphDebugMode';
- import {FlamegraphRenderer} from 'sentry/utils/profiling/renderers/flamegraphRenderer';
- import {GridRenderer} from 'sentry/utils/profiling/renderers/gridRenderer';
- import {SampleTickRenderer} from 'sentry/utils/profiling/renderers/sampleTickRenderer';
- import {SelectedFrameRenderer} from 'sentry/utils/profiling/renderers/selectedFrameRenderer';
- import {TextRenderer} from 'sentry/utils/profiling/renderers/textRenderer';
- import usePrevious from 'sentry/utils/usePrevious';
- import {FlamegraphFrame} from '../../utils/profiling/flamegraphFrame';
- import {BoundTooltip} from './boundTooltip';
- import {FlamegraphOptionsContextMenu} from './flamegraphOptionsContextMenu';
- function formatWeightToProfileDuration(frame: CallTreeNode, flamegraph: Flamegraph) {
- return `(${Math.round((frame.totalWeight / flamegraph.profile.duration) * 100)}%)`;
- }
- interface FlamegraphZoomViewProps {
- canvasBounds: Rect;
- canvasPoolManager: CanvasPoolManager;
- flamegraph: Flamegraph | DifferentialFlamegraph;
- flamegraphCanvas: FlamegraphCanvas | null;
- flamegraphCanvasRef: HTMLCanvasElement | null;
- flamegraphOverlayCanvasRef: HTMLCanvasElement | null;
- flamegraphRenderer: FlamegraphRenderer | null;
- flamegraphView: FlamegraphView | null;
- setFlamegraphCanvasRef: React.Dispatch<React.SetStateAction<HTMLCanvasElement | null>>;
- setFlamegraphOverlayCanvasRef: React.Dispatch<
- React.SetStateAction<HTMLCanvasElement | null>
- >;
- }
- function FlamegraphZoomView({
- canvasPoolManager,
- canvasBounds,
- flamegraphRenderer,
- flamegraph,
- flamegraphCanvas,
- flamegraphCanvasRef,
- flamegraphOverlayCanvasRef,
- flamegraphView,
- setFlamegraphCanvasRef,
- setFlamegraphOverlayCanvasRef,
- }: FlamegraphZoomViewProps): React.ReactElement {
- const flamegraphTheme = useFlamegraphTheme();
- const [flamegraphSearch] = useFlamegraphSearch();
- const isInternalFlamegraphDebugModeEnabled = useInternalFlamegraphDebugMode();
- const [lastInteraction, setLastInteraction] = useState<
- 'pan' | 'click' | 'zoom' | 'scroll' | null
- >(null);
- const [dispatch, {previousState, nextState}] = useDispatchFlamegraphState();
- const scheduler = useMemo(() => new CanvasScheduler(), []);
- const [flamegraphState, dispatchFlamegraphState] = useFlamegraphState();
- const [startPanVector, setStartPanVector] = useState<vec2 | null>(null);
- const [configSpaceCursor, setConfigSpaceCursor] = useState<vec2 | null>(null);
- const textRenderer: TextRenderer | null = useMemo(() => {
- if (!flamegraphOverlayCanvasRef) {
- return null;
- }
- return new TextRenderer(flamegraphOverlayCanvasRef, flamegraph, flamegraphTheme);
- }, [flamegraph, flamegraphOverlayCanvasRef, flamegraphTheme]);
- const gridRenderer: GridRenderer | null = useMemo(() => {
- if (!flamegraphOverlayCanvasRef) {
- return null;
- }
- return new GridRenderer(
- flamegraphOverlayCanvasRef,
- flamegraphTheme,
- flamegraph.formatter
- );
- }, [flamegraphOverlayCanvasRef, flamegraph, flamegraphTheme]);
- const sampleTickRenderer: SampleTickRenderer | null = useMemo(() => {
- if (!isInternalFlamegraphDebugModeEnabled) {
- return null;
- }
- if (!flamegraphOverlayCanvasRef || !flamegraphView?.configSpace) {
- return null;
- }
- return new SampleTickRenderer(
- flamegraphOverlayCanvasRef,
- flamegraph,
- flamegraphView.configSpace,
- flamegraphTheme
- );
- }, [
- isInternalFlamegraphDebugModeEnabled,
- flamegraphOverlayCanvasRef,
- flamegraph,
- flamegraphView?.configSpace,
- flamegraphTheme,
- ]);
- const selectedFrameRenderer = useMemo(() => {
- if (!flamegraphOverlayCanvasRef) {
- return null;
- }
- return new SelectedFrameRenderer(flamegraphOverlayCanvasRef);
- }, [flamegraphOverlayCanvasRef]);
- const hoveredNode = useMemo(() => {
- if (!configSpaceCursor || !flamegraphRenderer) {
- return null;
- }
- return flamegraphRenderer.getHoveredNode(configSpaceCursor);
- }, [configSpaceCursor, flamegraphRenderer]);
- useEffect(() => {
- const onKeyDown = (evt: KeyboardEvent) => {
- if (!flamegraphView) {
- return;
- }
- if (evt.key === 'z' && evt.metaKey) {
- const action = evt.shiftKey ? 'redo' : 'undo';
- if (action === 'undo') {
- const previousPosition = previousState?.position?.view;
- // If previous position is empty, reset the view to it's max
- if (previousPosition?.isEmpty()) {
- canvasPoolManager.dispatch('reset zoom', []);
- } else if (
- previousPosition &&
- !previousPosition?.equals(flamegraphView.configView)
- ) {
- // We need to always dispatch with the height of the current view,
- // because the height may have changed due to window resizing and
- // calling it with the old height may result in the flamegraph
- // being drawn into a very small or very large area.
- canvasPoolManager.dispatch('set config view', [
- previousPosition.withHeight(flamegraphView.configView.height),
- ]);
- }
- }
- if (action === 'redo') {
- const nextPosition = nextState?.position?.view;
- if (nextPosition && !nextPosition.equals(flamegraphView.configView)) {
- // We need to always dispatch with the height of the current view,
- // because the height may have changed due to window resizing and
- // calling it with the old height may result in the flamegraph
- // being drawn into a very small or very large area.
- canvasPoolManager.dispatch('set config view', [
- nextPosition.withHeight(flamegraphView.configView.height),
- ]);
- }
- }
- dispatchFlamegraphState({type: action});
- }
- };
- document.addEventListener('keydown', onKeyDown);
- return () => {
- document.removeEventListener('keydown', onKeyDown);
- };
- }, [
- canvasPoolManager,
- dispatchFlamegraphState,
- nextState,
- previousState,
- flamegraphView,
- ]);
- const previousInteraction = usePrevious(lastInteraction);
- const beforeInteractionConfigView = useRef<Rect | null>(null);
- useEffect(() => {
- if (!flamegraphView) {
- return;
- }
- // Check if we are starting a new interaction
- if (previousInteraction === null && lastInteraction) {
- beforeInteractionConfigView.current = flamegraphView.configView.clone();
- return;
- }
- if (
- beforeInteractionConfigView.current &&
- !beforeInteractionConfigView.current.equals(flamegraphView.configView)
- ) {
- dispatch({type: 'checkpoint', payload: flamegraphView.configView.clone()});
- }
- }, [dispatch, lastInteraction, previousInteraction, flamegraphView]);
- useEffect(() => {
- if (!flamegraphCanvas || !flamegraphView || !flamegraphRenderer) {
- return undefined;
- }
- const drawRectangles = () => {
- flamegraphRenderer.draw(
- flamegraphView.fromConfigView(flamegraphCanvas.physicalSpace),
- flamegraphState.search.results
- );
- };
- scheduler.registerBeforeFrameCallback(drawRectangles);
- scheduler.draw();
- return () => {
- scheduler.unregisterBeforeFrameCallback(drawRectangles);
- };
- }, [
- flamegraphCanvas,
- flamegraphRenderer,
- flamegraphState.search.results,
- scheduler,
- flamegraphView,
- ]);
- useEffect(() => {
- if (!flamegraphCanvas || !flamegraphView || !textRenderer || !gridRenderer) {
- return undefined;
- }
- const clearOverlayCanvas = () => {
- textRenderer.context.clearRect(
- 0,
- 0,
- textRenderer.canvas.width,
- textRenderer.canvas.height
- );
- };
- const drawText = () => {
- textRenderer.draw(
- flamegraphView.configView,
- flamegraphView.fromConfigView(flamegraphCanvas.physicalSpace),
- flamegraphSearch.results
- );
- };
- const drawGrid = () => {
- gridRenderer.draw(
- flamegraphView.configView,
- flamegraphCanvas.physicalSpace,
- flamegraphView.fromConfigView(flamegraphCanvas.physicalSpace),
- flamegraphView.toConfigView(flamegraphCanvas.logicalSpace)
- );
- };
- const drawInternalSampleTicks = () => {
- if (!sampleTickRenderer) {
- return;
- }
- sampleTickRenderer.draw(
- flamegraphView.fromConfigView(flamegraphCanvas.physicalSpace),
- flamegraphView.configView
- );
- };
- scheduler.registerBeforeFrameCallback(clearOverlayCanvas);
- scheduler.registerAfterFrameCallback(drawText);
- scheduler.registerAfterFrameCallback(drawGrid);
- scheduler.registerAfterFrameCallback(drawInternalSampleTicks);
- scheduler.draw();
- return () => {
- scheduler.unregisterBeforeFrameCallback(clearOverlayCanvas);
- scheduler.unregisterAfterFrameCallback(drawText);
- scheduler.unregisterAfterFrameCallback(drawGrid);
- scheduler.unregisterAfterFrameCallback(drawInternalSampleTicks);
- };
- }, [
- flamegraphCanvas,
- flamegraphView,
- scheduler,
- flamegraph,
- flamegraphTheme,
- textRenderer,
- gridRenderer,
- selectedFrameRenderer,
- sampleTickRenderer,
- canvasPoolManager,
- flamegraphSearch,
- ]);
- useEffect(() => {
- if (!flamegraphCanvas || !flamegraphView || !selectedFrameRenderer) {
- return undefined;
- }
- const state: {selectedNode: FlamegraphFrame | null} = {
- selectedNode: null,
- };
- const onNodeHighlight = (
- node: FlamegraphFrame | null,
- mode: 'hover' | 'selected'
- ) => {
- if (mode === 'selected') {
- state.selectedNode = node;
- }
- scheduler.draw();
- };
- const drawSelectedFrameBorder = () => {
- if (state.selectedNode) {
- selectedFrameRenderer.draw(
- new Rect(
- state.selectedNode.start,
- state.selectedNode.depth,
- state.selectedNode.end - state.selectedNode.start,
- 1
- ),
- {
- BORDER_COLOR: flamegraphTheme.COLORS.SELECTED_FRAME_BORDER_COLOR,
- BORDER_WIDTH: flamegraphTheme.SIZES.FRAME_BORDER_WIDTH,
- },
- flamegraphView.fromConfigView(flamegraphCanvas.physicalSpace)
- );
- }
- };
- scheduler.on('highlight frame', onNodeHighlight);
- scheduler.registerAfterFrameCallback(drawSelectedFrameBorder);
- return () => {
- scheduler.off('highlight frame', onNodeHighlight);
- scheduler.unregisterAfterFrameCallback(drawSelectedFrameBorder);
- };
- }, [
- flamegraphView,
- flamegraphCanvas,
- scheduler,
- selectedFrameRenderer,
- flamegraphTheme,
- ]);
- useEffect(() => {
- if (!flamegraphCanvas || !flamegraphView || !selectedFrameRenderer) {
- return undefined;
- }
- const drawHoveredFrameBorder = () => {
- if (hoveredNode) {
- selectedFrameRenderer.draw(
- new Rect(
- hoveredNode.start,
- hoveredNode.depth,
- hoveredNode.end - hoveredNode.start,
- 1
- ),
- {
- BORDER_COLOR: flamegraphTheme.COLORS.HOVERED_FRAME_BORDER_COLOR,
- BORDER_WIDTH: flamegraphTheme.SIZES.HOVERED_FRAME_BORDER_WIDTH,
- },
- flamegraphView.fromConfigView(flamegraphCanvas.physicalSpace)
- );
- }
- };
- scheduler.registerAfterFrameCallback(drawHoveredFrameBorder);
- scheduler.draw();
- return () => {
- scheduler.unregisterAfterFrameCallback(drawHoveredFrameBorder);
- };
- }, [
- flamegraphView,
- flamegraphCanvas,
- scheduler,
- hoveredNode,
- selectedFrameRenderer,
- flamegraphTheme,
- ]);
- useEffect(() => {
- if (!flamegraphCanvas || !flamegraphView) {
- return undefined;
- }
- const onResetZoom = () => {
- setConfigSpaceCursor(null);
- };
- const onZoomIntoFrame = () => {
- setConfigSpaceCursor(null);
- };
- scheduler.on('reset zoom', onResetZoom);
- scheduler.on('zoom at frame', onZoomIntoFrame);
- return () => {
- scheduler.off('reset zoom', onResetZoom);
- scheduler.off('zoom at frame', onZoomIntoFrame);
- };
- }, [
- flamegraphCanvas,
- canvasPoolManager,
- dispatchFlamegraphState,
- scheduler,
- flamegraphView,
- ]);
- useEffect(() => {
- canvasPoolManager.registerScheduler(scheduler);
- return () => canvasPoolManager.unregisterScheduler(scheduler);
- }, [canvasPoolManager, scheduler]);
- const onCanvasMouseDown = useCallback((evt: React.MouseEvent<HTMLCanvasElement>) => {
- const logicalMousePos = vec2.fromValues(
- evt.nativeEvent.offsetX,
- evt.nativeEvent.offsetY
- );
- const physicalMousePos = vec2.scale(
- vec2.create(),
- logicalMousePos,
- window.devicePixelRatio
- );
- setLastInteraction('click');
- setStartPanVector(physicalMousePos);
- }, []);
- const onCanvasMouseUp = useCallback(
- (evt: React.MouseEvent<HTMLCanvasElement>) => {
- evt.preventDefault();
- evt.stopPropagation();
- if (!configSpaceCursor) {
- setLastInteraction(null);
- setStartPanVector(null);
- return;
- }
- // Only dispatch the zoom action if the new clicked node is not the same as the old selected node.
- // This essentially tracks double click action on a rectangle
- if (lastInteraction === 'click') {
- if (
- hoveredNode &&
- flamegraphState.profiles.selectedRoot &&
- hoveredNode === flamegraphState.profiles.selectedRoot
- ) {
- // If double click is fired on a node, then zoom into it
- canvasPoolManager.dispatch('zoom at frame', [hoveredNode, 'exact']);
- }
- canvasPoolManager.dispatch('highlight frame', [hoveredNode, 'selected']);
- dispatchFlamegraphState({type: 'set selected root', payload: hoveredNode});
- }
- setLastInteraction(null);
- setStartPanVector(null);
- },
- [
- configSpaceCursor,
- flamegraphState.profiles.selectedRoot,
- dispatchFlamegraphState,
- hoveredNode,
- canvasPoolManager,
- lastInteraction,
- ]
- );
- const onMouseDrag = useCallback(
- (evt: React.MouseEvent<HTMLCanvasElement>) => {
- if (!flamegraphCanvas || !flamegraphView || !startPanVector) {
- return;
- }
- const logicalMousePos = vec2.fromValues(
- evt.nativeEvent.offsetX,
- evt.nativeEvent.offsetY
- );
- const physicalMousePos = vec2.scale(
- vec2.create(),
- logicalMousePos,
- window.devicePixelRatio
- );
- const physicalDelta = vec2.subtract(
- vec2.create(),
- startPanVector,
- physicalMousePos
- );
- if (physicalDelta[0] === 0 && physicalDelta[1] === 0) {
- return;
- }
- const physicalToConfig = mat3.invert(
- mat3.create(),
- flamegraphView.fromConfigView(flamegraphCanvas.physicalSpace)
- );
- const [m00, m01, m02, m10, m11, m12] = physicalToConfig;
- const configDelta = vec2.transformMat3(vec2.create(), physicalDelta, [
- m00,
- m01,
- m02,
- m10,
- m11,
- m12,
- 0,
- 0,
- 0,
- ]);
- canvasPoolManager.dispatch('transform config view', [
- mat3.fromTranslation(mat3.create(), configDelta),
- ]);
- setStartPanVector(physicalMousePos);
- },
- [flamegraphCanvas, flamegraphView, startPanVector, canvasPoolManager]
- );
- const onCanvasMouseMove = useCallback(
- (evt: React.MouseEvent<HTMLCanvasElement>) => {
- if (!flamegraphCanvas || !flamegraphView) {
- return;
- }
- const configSpaceMouse = flamegraphView.getConfigViewCursor(
- vec2.fromValues(evt.nativeEvent.offsetX, evt.nativeEvent.offsetY),
- flamegraphCanvas
- );
- setConfigSpaceCursor(configSpaceMouse);
- if (startPanVector) {
- onMouseDrag(evt);
- setLastInteraction('pan');
- } else {
- setLastInteraction(null);
- }
- },
- [flamegraphCanvas, flamegraphView, setConfigSpaceCursor, onMouseDrag, startPanVector]
- );
- const onCanvasMouseLeave = useCallback(() => {
- setConfigSpaceCursor(null);
- setStartPanVector(null);
- setLastInteraction(null);
- }, []);
- const zoom = useCallback(
- (evt: WheelEvent) => {
- if (!flamegraphCanvas || !flamegraphView) {
- return;
- }
- const identity = mat3.identity(mat3.create());
- const scale = 1 - evt.deltaY * 0.01 * -1; // -1 to invert scale
- const mouseInConfigView = flamegraphView.getConfigViewCursor(
- vec2.fromValues(evt.offsetX, evt.offsetY),
- flamegraphCanvas
- );
- const configCenter = vec2.fromValues(
- mouseInConfigView[0],
- flamegraphView.configView.y
- );
- const invertedConfigCenter = vec2.multiply(
- vec2.create(),
- vec2.fromValues(-1, -1),
- configCenter
- );
- const translated = mat3.translate(mat3.create(), identity, configCenter);
- const scaled = mat3.scale(mat3.create(), translated, vec2.fromValues(scale, 1));
- const translatedBack = mat3.translate(mat3.create(), scaled, invertedConfigCenter);
- canvasPoolManager.dispatch('transform config view', [translatedBack]);
- },
- [flamegraphCanvas, flamegraphView, canvasPoolManager]
- );
- const scroll = useCallback(
- (evt: WheelEvent) => {
- if (!flamegraphCanvas || !flamegraphView) {
- return;
- }
- const physicalDelta = vec2.fromValues(evt.deltaX, evt.deltaY);
- const physicalToConfig = mat3.invert(
- mat3.create(),
- flamegraphView.fromConfigView(flamegraphCanvas.physicalSpace)
- );
- const [m00, m01, m02, m10, m11, m12] = physicalToConfig;
- const configDelta = vec2.transformMat3(vec2.create(), physicalDelta, [
- m00,
- m01,
- m02,
- m10,
- m11,
- m12,
- 0,
- 0,
- 0,
- ]);
- const translate = mat3.fromTranslation(mat3.create(), configDelta);
- canvasPoolManager.dispatch('transform config view', [translate]);
- },
- [flamegraphCanvas, flamegraphView, canvasPoolManager]
- );
- useEffect(() => {
- if (!flamegraphCanvasRef) {
- return undefined;
- }
- let wheelStopTimeoutId: number | undefined;
- function onCanvasWheel(evt: WheelEvent) {
- window.clearTimeout(wheelStopTimeoutId);
- wheelStopTimeoutId = window.setTimeout(() => {
- setLastInteraction(null);
- }, 300);
- evt.preventDefault();
- // When we zoom, we want to clear cursor so that any tooltips
- // rendered on the flamegraph are removed from the flamegraphView
- setConfigSpaceCursor(null);
- // pinch to zoom is recognized as `ctrlKey + wheelEvent`
- if (evt.metaKey || evt.ctrlKey) {
- zoom(evt);
- setLastInteraction('zoom');
- } else {
- scroll(evt);
- setLastInteraction('scroll');
- }
- }
- flamegraphCanvasRef.addEventListener('wheel', onCanvasWheel);
- return () => {
- window.clearTimeout(wheelStopTimeoutId);
- flamegraphCanvasRef.removeEventListener('wheel', onCanvasWheel);
- };
- }, [flamegraphCanvasRef, zoom, scroll]);
- const contextMenu = useContextMenu({container: flamegraphCanvasRef});
- return (
- <Fragment>
- <CanvasContainer>
- <Canvas
- ref={canvas => setFlamegraphCanvasRef(canvas)}
- onMouseDown={onCanvasMouseDown}
- onMouseUp={onCanvasMouseUp}
- onMouseMove={onCanvasMouseMove}
- onMouseLeave={onCanvasMouseLeave}
- onContextMenu={contextMenu.handleContextMenu}
- style={{cursor: lastInteraction === 'pan' ? 'grab' : 'default'}}
- />
- <Canvas
- ref={canvas => setFlamegraphOverlayCanvasRef(canvas)}
- style={{
- pointerEvents: 'none',
- }}
- />
- <FlamegraphOptionsContextMenu contextMenu={contextMenu} />
- {flamegraphCanvas &&
- flamegraphRenderer &&
- flamegraphView &&
- configSpaceCursor &&
- hoveredNode?.frame?.name ? (
- <BoundTooltip
- bounds={canvasBounds}
- cursor={configSpaceCursor}
- flamegraphCanvas={flamegraphCanvas}
- flamegraphView={flamegraphView}
- >
- <HoveredFrameMainInfo>
- <FrameColorIndicator
- backgroundColor={formatColorForFrame(hoveredNode, flamegraphRenderer)}
- />
- {flamegraphRenderer.flamegraph.formatter(hoveredNode.node.totalWeight)}{' '}
- {formatWeightToProfileDuration(
- hoveredNode.node,
- flamegraphRenderer.flamegraph
- )}{' '}
- {hoveredNode.frame.name}
- </HoveredFrameMainInfo>
- <HoveredFrameTimelineInfo>
- {flamegraphRenderer.flamegraph.timelineFormatter(hoveredNode.start)}{' '}
- {' \u2014 '}
- {flamegraphRenderer.flamegraph.timelineFormatter(hoveredNode.end)}
- </HoveredFrameTimelineInfo>
- </BoundTooltip>
- ) : null}
- </CanvasContainer>
- </Fragment>
- );
- }
- const HoveredFrameTimelineInfo = styled('div')`
- color: ${p => p.theme.subText};
- `;
- const HoveredFrameMainInfo = styled('div')`
- display: flex;
- align-items: center;
- `;
- const FrameColorIndicator = styled('div')<{
- backgroundColor: React.CSSProperties['backgroundColor'];
- }>`
- width: 12px;
- height: 12px;
- min-width: 12px;
- min-height: 12px;
- border-radius: 2px;
- display: inline-block;
- background-color: ${p => p.backgroundColor};
- margin-right: ${space(1)};
- `;
- const CanvasContainer = styled('div')`
- display: flex;
- flex-direction: column;
- height: 100%;
- position: relative;
- `;
- const Canvas = styled('canvas')`
- left: 0;
- top: 0;
- width: 100%;
- height: 100%;
- user-select: none;
- position: absolute;
- `;
- export {FlamegraphZoomView};
|