openreplay/frontend/app/components/Session_/Player/Controls/Controls.tsx

474 lines
14 KiB
TypeScript

import { STORAGE_TYPES, StorageType, selectStorageType } from "Player";
import { Switch } from "antd";
import cn from "classnames";
import { observer } from "mobx-react-lite";
import React from "react";
import { connect } from "react-redux";
import { PlayerContext } from "App/components/Session/playerContext";
import { useStore } from "App/mstore";
import { FullScreenButton, PlayButton, PlayingState } from "App/player-ui";
import { session as sessionRoute, withSiteId } from "App/routes";
import useShortcuts from "Components/Session/Player/ReplayPlayer/useShortcuts";
import {
LaunchConsoleShortcut,
LaunchEventsShortcut,
LaunchNetworkShortcut,
LaunchPerformanceShortcut,
LaunchStateShortcut,
LaunchXRaShortcut,
} from 'Components/Session_/Player/Controls/components/KeyboardHelp';
import {
CONSOLE,
GRAPHQL,
INSPECTOR,
NETWORK,
OVERVIEW,
PERFORMANCE,
PROFILER,
STACKEVENTS,
STORAGE,
changeSkipInterval,
fullscreenOff,
fullscreenOn,
toggleBottomBlock,
} from 'Duck/components/player';
import { fetchSessions } from "Duck/liveSearch";
import { Icon } from "UI";
import ControlButton from "./ControlButton";
import Timeline from "./Timeline";
import PlayerControls from "./components/PlayerControls";
import styles from "./controls.module.css";
export const SKIP_INTERVALS = {
2: 2e3,
5: 5e3,
10: 1e4,
15: 15e3,
20: 2e4,
30: 3e4,
60: 6e4
};
function getStorageName(type: any) {
switch (type) {
case STORAGE_TYPES.REDUX:
return "Redux";
case STORAGE_TYPES.MOBX:
return "Mobx";
case STORAGE_TYPES.VUEX:
return "Vuex";
case STORAGE_TYPES.NGRX:
return "NgRx";
case STORAGE_TYPES.ZUSTAND:
return "Zustand";
case STORAGE_TYPES.NONE:
return "State";
default:
return "State";
}
}
function Controls(props: any) {
const { player, store } = React.useContext(PlayerContext);
const { uxtestingStore } = useStore();
const {
playing,
completed,
skip,
speed,
messagesLoading,
markedTargets,
inspectorMode
} = store.get();
const {
bottomBlock,
toggleBottomBlock,
fullscreen,
changeSkipInterval,
skipInterval,
disabledRedux,
showStorageRedux,
session,
previousSessionId,
nextSessionId,
siteId,
setActiveTab
} = props;
const disabled = disabledRedux || messagesLoading || inspectorMode || markedTargets;
const sessionTz = session?.timezone;
const nextHandler = () => {
props.history.push(withSiteId(sessionRoute(nextSessionId), siteId));
};
const prevHandler = () => {
props.history.push(withSiteId(sessionRoute(previousSessionId), siteId));
};
useShortcuts({
skipInterval,
fullScreenOn: props.fullscreenOn,
fullScreenOff: props.fullscreenOff,
toggleBottomBlock,
openNextSession: nextHandler,
openPrevSession: prevHandler,
setActiveTab
});
const forthTenSeconds = () => {
// @ts-ignore
player.jumpInterval(SKIP_INTERVALS[skipInterval]);
};
const backTenSeconds = () => {
// @ts-ignore
player.jumpInterval(-SKIP_INTERVALS[skipInterval]);
};
const toggleBottomTools = (blockName: number) => {
player.toggleInspectorMode(false);
toggleBottomBlock(blockName);
};
const state = completed
? PlayingState.Completed
: playing
? PlayingState.Playing
: PlayingState.Paused;
return (
<div className={styles.controls}>
<Timeline />
{!fullscreen && (
<div className={cn(styles.buttons, "!px-2")}>
<div className="flex items-center">
<PlayerControls
skip={skip}
sessionTz={sessionTz}
speed={speed}
disabled={disabled}
backTenSeconds={backTenSeconds}
forthTenSeconds={forthTenSeconds}
toggleSpeed={(speedIndex) => player.toggleSpeed(speedIndex)}
toggleSkip={() => player.toggleSkip()}
playButton={<PlayButton state={state} togglePlay={player.togglePlay} iconSize={36} />}
skipIntervals={SKIP_INTERVALS}
setSkipInterval={changeSkipInterval}
currentInterval={skipInterval}
startedAt={session.startedAt}
/>
<div className={cn("mx-2")} />
</div>
<div className="flex gap-2 items-center h-full">
{uxtestingStore.hideDevtools && uxtestingStore.isUxt() ? null : (
<DevtoolsButtons
showStorageRedux={showStorageRedux}
toggleBottomTools={toggleBottomTools}
bottomBlock={bottomBlock}
disabled={disabled}
/>
)}
<FullScreenButton
size={16}
onClick={props.fullscreenOn}
customClasses={"rounded hover:bg-gray-light-shade color-gray-medium"}
/>
</div>
</div>
)}
</div>
);
}
interface IDevtoolsButtons {
showStorageRedux: boolean;
toggleBottomTools: (blockName: number) => void;
bottomBlock: number;
disabled: boolean;
}
const DevtoolsButtons = observer(
({ showStorageRedux, toggleBottomTools, bottomBlock, disabled }: IDevtoolsButtons) => {
const { aiSummaryStore } = useStore();
const { store, player } = React.useContext(PlayerContext);
// @ts-ignore
const originStr = window.env.ORIGIN || window.location.origin;
const isSaas = /app\.openreplay\.com/.test(originStr);
const { inspectorMode, currentTab, tabStates } = store.get();
const disableButtons = disabled;
const profilesList = tabStates[currentTab]?.profilesList || [];
const graphqlList = tabStates[currentTab]?.graphqlList || [];
const logRedCount = tabStates[currentTab]?.logMarkedCountNow || 0;
const resourceRedCount = tabStates[currentTab]?.resourceMarkedCountNow || 0;
const stackRedCount = tabStates[currentTab]?.stackMarkedCountNow || 0;
const exceptionsList = tabStates[currentTab]?.exceptionsList || [];
const storageType = store.get().tabStates[currentTab]
? selectStorageType(store.get().tabStates[currentTab])
: StorageType.NONE;
const profilesCount = profilesList.length;
const graphqlCount = graphqlList.length;
const showGraphql = graphqlCount > 0;
const showProfiler = profilesCount > 0;
const showExceptions = exceptionsList.length > 0;
const showStorage = storageType !== STORAGE_TYPES.NONE || showStorageRedux;
const showSummary = () => {
player.pause();
if (bottomBlock !== OVERVIEW) {
toggleBottomTools(OVERVIEW);
}
aiSummaryStore.setToggleSummary(!aiSummaryStore.toggleSummary);
};
return (
<>
{isSaas ? (
<SummaryButton
onClick={showSummary}
withToggle={bottomBlock === OVERVIEW}
toggleValue={aiSummaryStore.toggleSummary}
/>
) : null}
<ControlButton
popover={
<div className={"flex items-center gap-2"}>
<LaunchXRaShortcut />
<div>Get a quick overview on the issues in this session.</div>
</div>
}
label={"X-Ray"}
onClick={() => toggleBottomTools(OVERVIEW)}
active={bottomBlock === OVERVIEW && !inspectorMode}
/>
<ControlButton
popover={
<div className={"flex gap-2 items-center"}>
<LaunchConsoleShortcut />
<div>Launch Console</div>
</div>
}
disabled={disableButtons}
onClick={() => toggleBottomTools(CONSOLE)}
active={bottomBlock === CONSOLE && !inspectorMode}
label="Console"
hasErrors={logRedCount > 0 || showExceptions}
/>
<ControlButton
popover={
<div className={"flex gap-2 items-center"}>
<LaunchNetworkShortcut />
<div>Launch Network</div>
</div>
}
disabled={disableButtons}
onClick={() => toggleBottomTools(NETWORK)}
active={bottomBlock === NETWORK && !inspectorMode}
label="Network"
hasErrors={resourceRedCount > 0}
/>
<ControlButton
popover={
<div className={"flex gap-2 items-center"}>
<LaunchPerformanceShortcut />
<div>Launch Performance</div>
</div>
}
disabled={disableButtons}
onClick={() => toggleBottomTools(PERFORMANCE)}
active={bottomBlock === PERFORMANCE && !inspectorMode}
label="Performance"
/>
{showGraphql && (
<ControlButton
disabled={disableButtons}
onClick={() => toggleBottomTools(GRAPHQL)}
active={bottomBlock === GRAPHQL && !inspectorMode}
label="Graphql"
/>
)}
{showStorage && (
<ControlButton
popover={
<div className={"flex gap-2 items-center"}>
<LaunchStateShortcut />
<div>Launch State</div>
</div>
}
disabled={disableButtons}
onClick={() => toggleBottomTools(STORAGE)}
active={bottomBlock === STORAGE && !inspectorMode}
label={getStorageName(storageType) as string}
/>
)}
<ControlButton
popover={
<div className={"flex gap-2 items-center"}>
<LaunchEventsShortcut />
<div>Launch Events</div>
</div>
}
disabled={disableButtons}
onClick={() => toggleBottomTools(STACKEVENTS)}
active={bottomBlock === STACKEVENTS && !inspectorMode}
label="Events"
hasErrors={stackRedCount > 0}
/>
{showProfiler && (
<ControlButton
disabled={disableButtons}
onClick={() => toggleBottomTools(PROFILER)}
active={bottomBlock === PROFILER && !inspectorMode}
label="Profiler"
/>
)}
</>
);
}
);
export function SummaryButton({
onClick,
withToggle,
onToggle,
toggleValue
}: {
onClick?: () => void,
withToggle?: boolean,
onToggle?: () => void,
toggleValue?: boolean
}) {
const [isHovered, setHovered] = React.useState(false);
return (
<div style={gradientButton} onClick={onClick}>
<div
style={isHovered ? onHoverFillStyle : fillStyle}
onMouseEnter={() => setHovered(true)}
onMouseLeave={() => setHovered(false)}
>
{withToggle ? (
<Switch
size={"small"}
checked={toggleValue}
onChange={onToggle}
/>
) : null}
<Icon name={"sparkles"} size={16} />
<div className={"font-semibold text-main"}>Summary AI</div>
</div>
</div>
);
}
const gradientButton = {
border: "double 1px transparent",
borderRadius: "60px",
background:
"linear-gradient(#f6f6f6, #f6f6f6), linear-gradient(to right, #394EFF 0%, #3EAAAF 100%)",
backgroundOrigin: "border-box",
backgroundClip: "content-box, border-box",
cursor: "pointer"
};
const onHoverFillStyle = {
width: "100%",
height: "100%",
display: "flex",
borderRadius: "60px",
gap: 2,
alignItems: "center",
padding: "2px 8px",
background: "linear-gradient(156deg, #E3E6FF 0%, #E4F3F4 69.48%)"
};
const fillStyle = {
width: "100%",
height: "100%",
display: "flex",
borderRadius: "60px",
gap: 2,
alignItems: "center",
padding: "2px 8px"
};
const ControlPlayer = observer(Controls);
export default connect(
(state: any) => {
const permissions = state.getIn(["user", "account", "permissions"]) || [];
const isEnterprise = state.getIn(["user", "account", "edition"]) === "ee";
return {
disabledRedux: isEnterprise && !permissions.includes("DEV_TOOLS"),
fullscreen: state.getIn(["components", "player", "fullscreen"]),
bottomBlock: state.getIn(["components", "player", "bottomBlock"]),
showStorageRedux: !state.getIn(["components", "player", "hiddenHints", "storage"]),
showStackRedux: !state.getIn(["components", "player", "hiddenHints", "stack"]),
session: state.getIn(["sessions", "current"]),
totalAssistSessions: state.getIn(["liveSearch", "total"]),
skipInterval: state.getIn(["components", "player", "skipInterval"]),
previousSessionId: state.getIn(["sessions", "previousId"]),
nextSessionId: state.getIn(["sessions", "nextId"]),
siteId: state.getIn(["site", "siteId"])
};
},
{
fullscreenOn,
fullscreenOff,
toggleBottomBlock,
fetchSessions,
changeSkipInterval
}
)(ControlPlayer);
// shouldComponentUpdate(nextProps) {
// if (
// nextProps.fullscreen !== props.fullscreen ||
// nextProps.bottomBlock !== props.bottomBlock ||
// nextProps.live !== props.live ||
// nextProps.livePlay !== props.livePlay ||
// nextProps.playing !== props.playing ||
// nextProps.completed !== props.completed ||
// nextProps.skip !== props.skip ||
// nextProps.skipToIssue !== props.skipToIssue ||
// nextProps.speed !== props.speed ||
// nextProps.disabled !== props.disabled ||
// nextProps.fullscreenDisabled !== props.fullscreenDisabled ||
// // nextProps.inspectorMode !== props.inspectorMode ||
// // nextProps.logCount !== props.logCount ||
// nextProps.logRedCount !== props.logRedCount ||
// nextProps.showExceptions !== props.showExceptions ||
// nextProps.resourceRedCount !== props.resourceRedCount ||
// nextProps.fetchRedCount !== props.fetchRedCount ||
// nextProps.showStack !== props.showStack ||
// nextProps.stackCount !== props.stackCount ||
// nextProps.stackRedCount !== props.stackRedCount ||
// nextProps.profilesCount !== props.profilesCount ||
// nextProps.storageCount !== props.storageCount ||
// nextProps.storageType !== props.storageType ||
// nextProps.showStorage !== props.showStorage ||
// nextProps.showProfiler !== props.showProfiler ||
// nextProps.showGraphql !== props.showGraphql ||
// nextProps.showFetch !== props.showFetch ||
// nextProps.fetchCount !== props.fetchCount ||
// nextProps.graphqlCount !== props.graphqlCount ||
// nextProps.liveTimeTravel !== props.liveTimeTravel ||
// nextProps.skipInterval !== props.skipInterval
// )
// return true;
// return false;
// }