openreplay/frontend/app/layout/SideMenu.tsx
Delirium 1326bb2eae
feat spot: init commit for extension (#2452)
* feat spot: init commit for extension

* nvmrc

* fix login flow

* Spots Gridview Updates (#2422)

* feat ui: login flow for spot extension

* spot list, store and service created

* some fixing for header

* start work on single spot

* spot player start

* header for player, comments, icons, etc

* split stuff into compoennts, create player state manager

* player controls, activity panel etc etc

* comments, empty page, rename and stuff

* interval buttons etc

* access modal

* pubkey support

* fix tooltip

* limit 10 -> 9

* hls lib instead of videojs

* some warnings

* fix date display for exp

* change public links

* display more client data

* fix cleaning, init comment

* map network to replay player network ev

* stream support, console panel, close panels on X

* fixing streaming, destroy on leave

* fix autoplay

* show notification on spot login

* fix spot login

* backup player added, fix audio issue

* show thumbnail when no video, add spot roles

* add poster thumbnail

* some fixes to video check

* fix events jump

* fix play btn

* try catch over pubkey

* feat ui: login flow for spot extension

* spot list, store and service created

* some fixing for header

* start work on single spot

* spot player start

* header for player, comments, icons, etc

* split stuff into compoennts, create player state manager

* player controls, activity panel etc etc

* comments, empty page, rename and stuff

* interval buttons etc

* access modal

* pubkey support

* fix tooltip

* limit 10 -> 9

* hls lib instead of videojs

* some warnings

* fix date display for exp

* change public links

* display more client data

* fix cleaning, init comment

* map network to replay player network ev

* stream support, console panel, close panels on X

* fixing streaming, destroy on leave

* fix autoplay

* show notification on spot login

* fix spot login

* backup player added, fix audio issue

* show thumbnail when no video, add spot roles

* add poster thumbnail

* some fixes to video check

* fix events jump

* fix play btn

* try catch over pubkey

* icons

* Various updates

* Update SVG.tsx

* Update SideMenu.tsx

* SpotList & Menu updates

* feat ui: login flow for spot extension

* spot list, store and service created

* some fixing for header

* start work on single spot

* spot player start

* header for player, comments, icons, etc

* split stuff into compoennts, create player state manager

* player controls, activity panel etc etc

* comments, empty page, rename and stuff

* interval buttons etc

* access modal

* pubkey support

* fix tooltip

* limit 10 -> 9

* hls lib instead of videojs

* some warnings

* fix date display for exp

* change public links

* display more client data

* fix cleaning, init comment

* map network to replay player network ev

* stream support, console panel, close panels on X

* fixing streaming, destroy on leave

* fix autoplay

* show notification on spot login

* fix spot login

* backup player added, fix audio issue

* show thumbnail when no video, add spot roles

* add poster thumbnail

* some fixes to video check

* fix events jump

* fix play btn

* try catch over pubkey

* icons

* spot login pinging

* Spot List & Player Updates

* move spot login flow to login comp, use separate spot login path for unique jwt

* invalidate spot jwt on logout

* add visual data on page load event

* typo fix

* Spot Listing improvements post review.

* Update SpotListItem.tsx

* Improved Spot List and Item Details

* Minor improvements

* More improvements

* Public player header improvements

* Moved formatExpirationTime to utils

* fixes after merge

---------

Co-authored-by: nick-delirium <nikita@openreplay.com>

* set sso link to <a>?

* some small perf fixes

* login duck reformat...

* Update frontend.yaml

* add observer to spot list header

* split list header

* update spotjwt param in router

* fix toast in router

* fix async fetch, move ctx

* capture space btn ev

* fix header link

* public sharing error msg

* fix err msg for unsuccessful rec start

* fix list alignment

* Caching assets. Finally!!!

* fix typing in comment field

* add pubkey to comments, fix console jump btn

* no content comp

* change refresh token logic

* move thumbnail ts

* move thumbnail ts

* fix tab change

* switch up toggler

* early exit if no jwt present

* regenerate icons

* fix location str

* fix ctx

* change thumnail res, return autoplay for video player

* parse links in console rows, fix injected method parse?

* remove ts from js

* fix console parsing order?

* fixes for autoplay

* xray for spot player

* move to spot list after login;
esc to cancel;
fix signup link;
move ux commit

* kb sc for skipping; xray for spot ext

* track aborted requests

* tooltip for readability

* fixing empty state

* New blank state + various minor improvements (#2471)

* New blank state + various minor improvements

* apres merge

---------

Co-authored-by: nick-delirium <nikita@openreplay.com>

* rm temp v

* init or card

* empty state debug

* empty state debug

* empty state debug

* fix initor img

* spotonly scope support

* Improved Spot dead-end pages (#2475)

* Improved Spot dead-end pages

* Initiate OpenReplay Setup and some more

* get scope changes

* fix crash

* scope upgrade/downgrade

* scope setup flow

* ping for backend

* upgrade wxt deps

* cancel ping int on expiration

* check rec status

* fix ping

* check video processing state

* check video processing state

* fix xray close, network highlight, fcp rounding

* update wxt, move open spot stuff to settings

* fix some history issues

* fix spot login flow

* fix spot login again

* fix spot login again

* don't send two requests

* limit messages for logged users

* limit messages for logged users

* fix public ignore

* microphone stuff

* microphone stuff

* Various improvements (#2509)

* Various improvements

- Updated icons in mic settings
- Included prefix in Spot title
- Save recording notification has been updated
- Other minor UI improvements

* Inline declaration of spot name field, and settings UI

* str f

---------

Co-authored-by: nick-delirium <nikita@openreplay.com>

* UI changes in player header, spot list (#2510)

* Added UI elements in player page

- Badge with counts for comments
- Download and Delete dropdown in player
- Spot selection -- UI improvement

* Minor copy updates

* completing changes

---------

Co-authored-by: nick-delirium <nikita@openreplay.com>

* rm cmt

* fix cellmeasurer

* thumbnail dur

* fix download

* Minor fixes (#2512)

- Spot delete confirmation
- Spot comments UI update
- Minor copy updates

* limit number of notif messages

* add spot title to doc title, add cache groups for webpack

* drop mic controls from recording popup view

* fix for webpack compress

* fix for auto mic pickup

* change status banners

* move svgs around, remove undefined check

* refactor svgs

* fix timetable scaling

* fix error popup

* self contain css

* pre-select spot on spot onboarding

---------

Co-authored-by: Sudheer Salavadi <connect.uxmaster@gmail.com>
Co-authored-by: Rajesh Rajendran <rjshrjndrn@users.noreply.github.com>
2024-08-29 13:35:58 +02:00

378 lines
12 KiB
TypeScript

import { Divider, Menu, Tag, Typography } from 'antd';
import cn from 'classnames';
import React from 'react';
import { connect } from 'react-redux';
import { RouteComponentProps, withRouter } from 'react-router-dom';
import SupportModal from 'App/layout/SupportModal';
import * as routes from 'App/routes';
import {
CLIENT_DEFAULT_TAB,
CLIENT_TABS,
bookmarks,
client,
fflags,
notes,
sessions,
withSiteId,
} from 'App/routes';
import { MODULES } from 'Components/Client/Modules';
import { setActiveTab } from 'Duck/search';
import { Icon } from 'UI';
import SVG from 'UI/SVG';
import { getScope } from 'App/duck/user';
import InitORCard from './InitORCard';
import SpotToOpenReplayPrompt from './SpotToOpenReplayPrompt';
import {
MENU,
PREFERENCES_MENU,
categories as main_menu,
preferences,
spotOnlyCats,
} from './data';
const { Text } = Typography;
const TabToUrlMap = {
all: sessions() as '/sessions',
bookmark: bookmarks() as '/bookmarks',
notes: notes() as '/notes',
flags: fflags() as '/feature-flags',
};
interface Props extends RouteComponentProps {
siteId?: string;
modules: string[];
setActiveTab: (tab: any) => void;
activeTab: string;
isEnterprise: boolean;
isCollapsed?: boolean;
spotOnly?: boolean;
account: any;
}
function SideMenu(props: Props) {
const {
activeTab,
siteId,
modules,
location,
account,
isEnterprise,
isCollapsed,
spotOnly,
} = props;
const isPreferencesActive = location.pathname.includes('/client/');
const [supportOpen, setSupportOpen] = React.useState(false);
const isAdmin = account.admin || account.superAdmin;
const [isModalVisible, setIsModalVisible] = React.useState(false);
const handleModalOpen = () => {
setIsModalVisible(true);
};
const handleModalClose = () => {
setIsModalVisible(false);
};
let menu: any[] = React.useMemo(() => {
const sourceMenu = isPreferencesActive ? preferences : main_menu;
return sourceMenu
.filter((cat) => {
if (spotOnly) {
return spotOnlyCats.includes(cat.key);
}
return true;
})
.map((category) => {
const updatedItems = category.items
.filter((item) => {
if (spotOnly) {
return spotOnlyCats.includes(item.key);
}
return true;
})
.map((item) => {
if (isEnterprise) {
if (item.key === MENU.BOOKMARKS) {
return { ...item, hidden: true };
}
if (item.key === MENU.VAULT) {
return { ...item, hidden: false };
}
} else {
if (item.key === MENU.VAULT) {
return { ...item, hidden: true };
}
if (item.key === MENU.BOOKMARKS) {
return { ...item, hidden: false };
}
}
if (item.hidden) return item;
const isHidden = [
item.key === MENU.RECOMMENDATIONS &&
modules.includes(MODULES.RECOMMENDATIONS),
item.key === MENU.FEATURE_FLAGS &&
modules.includes(MODULES.FEATURE_FLAGS),
item.key === MENU.NOTES && modules.includes(MODULES.NOTES),
item.key === MENU.LIVE_SESSIONS &&
modules.includes(MODULES.ASSIST),
item.key === MENU.SESSIONS &&
modules.includes(MODULES.OFFLINE_RECORDINGS),
item.key === MENU.ALERTS && modules.includes(MODULES.ALERTS),
item.isAdmin && !isAdmin,
item.isEnterprise && !isEnterprise,
].some((cond) => cond);
return { ...item, hidden: isHidden };
});
const allItemsHidden = updatedItems.every((item) => item.hidden);
return {
...category,
items: updatedItems,
hidden: allItemsHidden,
};
});
}, [isAdmin, isEnterprise, isPreferencesActive, modules, spotOnly]);
React.useEffect(() => {
const currentLocation = location.pathname;
const tab = Object.keys(TabToUrlMap).find((tab: keyof typeof TabToUrlMap) =>
currentLocation.includes(TabToUrlMap[tab])
);
if (tab && tab !== activeTab) {
props.setActiveTab({ type: tab });
}
}, [location.pathname]);
const menuRoutes: any = {
[MENU.EXIT]: () =>
props.history.push(withSiteId(routes.sessions(), siteId)),
[MENU.SESSIONS]: () => withSiteId(routes.sessions(), siteId),
[MENU.BOOKMARKS]: () => withSiteId(routes.bookmarks(), siteId),
[MENU.VAULT]: () => withSiteId(routes.bookmarks(), siteId),
[MENU.NOTES]: () => withSiteId(routes.notes(), siteId),
[MENU.LIVE_SESSIONS]: () => withSiteId(routes.assist(), siteId),
[MENU.DASHBOARDS]: () => withSiteId(routes.dashboard(), siteId),
[MENU.CARDS]: () => withSiteId(routes.metrics(), siteId),
[MENU.ALERTS]: () => withSiteId(routes.alerts(), siteId),
[MENU.FEATURE_FLAGS]: () => withSiteId(routes.fflags(), siteId),
[MENU.PREFERENCES]: () => client(CLIENT_DEFAULT_TAB),
[MENU.USABILITY_TESTS]: () => withSiteId(routes.usabilityTesting(), siteId),
[MENU.SPOTS]: () => withSiteId(routes.spotsList(), siteId),
[PREFERENCES_MENU.ACCOUNT]: () => client(CLIENT_TABS.PROFILE),
[PREFERENCES_MENU.SESSION_LISTING]: () =>
client(CLIENT_TABS.SESSIONS_LISTING),
[PREFERENCES_MENU.INTEGRATIONS]: () => client(CLIENT_TABS.INTEGRATIONS),
[PREFERENCES_MENU.METADATA]: () => client(CLIENT_TABS.CUSTOM_FIELDS),
[PREFERENCES_MENU.WEBHOOKS]: () => client(CLIENT_TABS.WEBHOOKS),
[PREFERENCES_MENU.PROJECTS]: () => client(CLIENT_TABS.SITES),
[PREFERENCES_MENU.ROLES_ACCESS]: () => client(CLIENT_TABS.MANAGE_ROLES),
[PREFERENCES_MENU.AUDIT]: () => client(CLIENT_TABS.AUDIT),
[PREFERENCES_MENU.TEAM]: () => client(CLIENT_TABS.MANAGE_USERS),
[PREFERENCES_MENU.NOTIFICATIONS]: () => client(CLIENT_TABS.NOTIFICATIONS),
[PREFERENCES_MENU.BILLING]: () => client(CLIENT_TABS.BILLING),
[PREFERENCES_MENU.MODULES]: () => client(CLIENT_TABS.MODULES),
};
const handleClick = (item: any) => {
if (item.key === MENU.SUPPORT) {
setSupportOpen(true);
return;
}
const handler = menuRoutes[item.key];
if (handler) {
const route = handler();
pushTo(route);
}
};
const isMenuItemActive = (key: string) => {
const { pathname } = location;
const activeRoute = menuRoutes[key];
if (activeRoute && !key.includes('exit')) {
const route = activeRoute();
return pathname === route;
}
return false;
};
const pushTo = (path: string) => {
props.history.push(path);
};
return (
<>
<Menu
mode="inline"
onClick={handleClick}
style={{ marginTop: '8px', border: 'none' }}
selectedKeys={menu.flatMap((category) =>
category.items
.filter((item: any) => isMenuItemActive(item.key))
.map((item) => item.key)
)}
>
{menu.map((category, index) => (
<React.Fragment key={category.key}>
{!category.hidden && (
<>
{index > 0 && <Divider style={{ margin: '6px 0' }} />}
{category.items
.filter((item: any) => !item.hidden)
.map((item: any) => {
const isActive = isMenuItemActive(item.key);
if (item.key === MENU.EXIT) {
return (
<Menu.Item
key={item.key}
style={{ paddingLeft: '20px' }}
icon={
<Icon
name={item.icon}
size={16}
color={isActive ? 'teal' : ''}
/>
}
className={cn('!rounded-lg hover-fill-teal')}
>
{item.label}
</Menu.Item>
);
}
if (item.key === MENU.SPOTS) {
return (
<Menu.Item
key={item.key}
icon={
<Icon
name={item.icon}
size={16}
color={isActive ? 'teal' : ''}
/>
}
style={{ paddingLeft: '20px' }}
className={cn('!rounded-lg hover-fill-teal !pe-0')}
itemIcon={
item.leading ? (
<Icon
name={item.leading}
size={16}
color={isActive ? 'teal' : ''}
/>
) : null
}
>
<div
style={{
display: 'flex',
justifyContent: 'space-between',
alignItems: 'center',
}}
>
{item.label}
<Tag
color="cyan"
bordered={false}
className="text-xs"
>
{' '}
Beta{' '}
</Tag>
</div>
</Menu.Item>
);
}
return item.children ? (
<Menu.SubMenu
key={item.key}
title={
<Text className={cn('ml-5 !rounded')}>
{item.label}
</Text>
}
icon={<SVG name={item.icon} size={16} />}
>
{item.children.map((child: any) => (
<Menu.Item
className={cn('ml-8', {
'ant-menu-item-selected !bg-active-dark-blue':
isMenuItemActive(child.key),
})}
key={child.key}
>
{child.label}
</Menu.Item>
))}
</Menu.SubMenu>
) : (
<Menu.Item
key={item.key}
icon={
<Icon
name={item.icon}
size={16}
color={isActive ? 'teal' : ''}
className={'hover-fill-teal'}
/>
}
style={{ paddingLeft: '20px' }}
className={cn('!rounded-lg hover-fill-teal')}
itemIcon={
item.leading ? (
<Icon
name={item.leading}
size={16}
color={isActive ? 'teal' : ''}
/>
) : null
}
>
{item.label}
</Menu.Item>
);
})}
</>
)}
</React.Fragment>
))}
</Menu>
{spotOnly && !isPreferencesActive ? (
<>
<InitORCard onOpenModal={handleModalOpen} />
<SpotToOpenReplayPrompt
isVisible={isModalVisible}
onCancel={handleModalClose}
/>
</>
) : null}
<SupportModal onClose={() => setSupportOpen(false)} open={supportOpen} />
</>
);
}
export default withRouter(
connect(
(state: any) => ({
modules: state.getIn(['user', 'account', 'settings', 'modules']) || [],
activeTab: state.getIn(['search', 'activeTab', 'type']),
isEnterprise: state.getIn(['user', 'account', 'edition']) === 'ee',
account: state.getIn(['user', 'account']),
spotOnly: getScope(state) === 'spot',
}),
{ setActiveTab }
)(SideMenu)
);