From 70a58a3d36738a2648b2e7f77812677f50d522cf Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mateusz=20Burzy=C5=84ski?= Date: Mon, 15 Aug 2022 14:25:31 +0200 Subject: [PATCH 1/2] Replace `connect` with hooks in 3 components --- src/ui/components/Timeline/EventMarker.tsx | 25 ++------ src/ui/components/Timeline/MessageMarker.tsx | 24 +++----- .../WorkspaceSettingsModal.tsx | 59 +++++++------------ 3 files changed, 35 insertions(+), 73 deletions(-) diff --git a/src/ui/components/Timeline/EventMarker.tsx b/src/ui/components/Timeline/EventMarker.tsx index 56712937964..f8d8d9e01c5 100644 --- a/src/ui/components/Timeline/EventMarker.tsx +++ b/src/ui/components/Timeline/EventMarker.tsx @@ -1,17 +1,13 @@ import React from "react"; -import { connect, ConnectedProps } from "react-redux"; import { selectors } from "ui/reducers"; -import { UIState } from "ui/state"; import { ReplayEvent } from "ui/state/app"; +import { useAppSelector } from "ui/setup/hooks"; import Marker from "./Marker"; -function EventMarker({ - event, - currentTime, - isPrimaryHighlighted, - zoomRegion, - overlayWidth, -}: EventMarkerProps) { +export default function EventMarker({ event, isPrimaryHighlighted }: EventMarkerProps) { + const zoomRegion = useAppSelector(selectors.getZoomRegion); + const currentTime = useAppSelector(selectors.getCurrentTime); + const overlayWidth = useAppSelector(selectors.getTimelineDimensions).width; return ( ({ - zoomRegion: selectors.getZoomRegion(state), - currentTime: selectors.getCurrentTime(state), - overlayWidth: selectors.getTimelineDimensions(state).width, -})); - -type PropsFromRedux = ConnectedProps; -type EventMarkerProps = PropsFromRedux & { +type EventMarkerProps = { event: ReplayEvent; isPrimaryHighlighted: boolean; }; - -export default connector(EventMarker); diff --git a/src/ui/components/Timeline/MessageMarker.tsx b/src/ui/components/Timeline/MessageMarker.tsx index aa5330fd4e5..903d8e55030 100644 --- a/src/ui/components/Timeline/MessageMarker.tsx +++ b/src/ui/components/Timeline/MessageMarker.tsx @@ -1,20 +1,19 @@ import React from "react"; -import { connect, ConnectedProps } from "react-redux"; -import { actions } from "ui/actions"; +import { useAppSelector } from "ui/setup/hooks"; import { selectors } from "ui/reducers"; -import { UIState } from "ui/state"; import Marker from "./Marker"; -function MessageMarker({ +export default function MessageMarker({ message, - currentTime, isPrimaryHighlighted, isSecondaryHighlighted, - zoomRegion, - overlayWidth, }: MessageMarkerProps) { const { executionPoint, executionPointTime, frame, pauseId, executionPointHasFrames } = message; + const zoomRegion = useAppSelector(selectors.getZoomRegion); + const currentTime = useAppSelector(selectors.getCurrentTime); + const overlayWidth = useAppSelector(selectors.getTimelineDimensions).width; + return ( ({ - zoomRegion: selectors.getZoomRegion(state), - currentTime: selectors.getCurrentTime(state), - overlayWidth: selectors.getTimelineDimensions(state).width, -})); - -type PropsFromRedux = ConnectedProps; -type MessageMarkerProps = PropsFromRedux & { +type MessageMarkerProps = { message: any; isPrimaryHighlighted: boolean; isSecondaryHighlighted: boolean; }; - -export default connector(MessageMarker); diff --git a/src/ui/components/shared/WorkspaceSettingsModal/WorkspaceSettingsModal.tsx b/src/ui/components/shared/WorkspaceSettingsModal/WorkspaceSettingsModal.tsx index 52492fc6453..0eb2a8ac2fa 100644 --- a/src/ui/components/shared/WorkspaceSettingsModal/WorkspaceSettingsModal.tsx +++ b/src/ui/components/shared/WorkspaceSettingsModal/WorkspaceSettingsModal.tsx @@ -1,9 +1,8 @@ -import React, { ChangeEvent, useEffect, useState } from "react"; -import { connect, ConnectedProps } from "react-redux"; +import React, { ChangeEvent, useState } from "react"; import * as actions from "ui/actions/app"; import hooks from "ui/hooks"; +import { useAppSelector, useAppDispatch } from "ui/setup/hooks"; import * as selectors from "ui/reducers/app"; -import { UIState } from "ui/state"; import { WorkspaceUser } from "ui/types"; import { validateEmail } from "ui/utils/helpers"; import { TextInput } from "../Forms"; @@ -130,7 +129,6 @@ const settings: Settings< settings?: any; isAdmin: boolean; workspaceId: string; - hideModal: PropsFromRedux["hideModal"]; } > = [ { @@ -146,13 +144,13 @@ const settings: Settings< { title: "Team Members", icon: "group", - component: function TeamMembers({ isAdmin, workspaceId, settings, ...rest }) { + component: function TeamMembers({ isAdmin, workspaceId }) { const { members } = hooks.useGetWorkspaceMembers(workspaceId); return ( -
+
{`Manage members here so that everyone who belongs to this team can see each other's replays.`}
- +
{`Members`}
@@ -180,7 +178,8 @@ const settings: Settings< { title: "Delete Team", icon: "cancel", - component: function DeleteTeam({ hideModal, workspaceId }) { + component: function DeleteTeam({ workspaceId }) { + const dispatch = useAppDispatch(); const redirectToTeam = useRedirectToTeam(true); const updateDefaultWorkspace = hooks.useUpdateDefaultWorkspace(); const deleteWorkspace = hooks.useDeleteWorkspace(); @@ -196,7 +195,7 @@ const settings: Settings< deleteWorkspace({ variables: { workspaceId: workspaceId, shouldDeleteRecordings: true }, }); - hideModal(); + dispatch(actions.hideModal()); updateDefaultWorkspace({ variables: { workspaceId: null } }); redirectToTeam("me"); } @@ -221,24 +220,24 @@ const settings: Settings< }, ]; -function WorkspaceSettingsModal({ view, ...rest }: PropsFromRedux) { +export default function WorkspaceSettingsModal() { + const selectedTab = useAppSelector(state => { + const opts = selectors.getModalOptions(state); + const view = opts && "view" in opts ? opts.view : null; + return ( + view && + { + billing: "Billing", + members: "Team Members", + api: "API Keys", + }[view] + ); + }); const workspaceId = useGetTeamIdFromRoute(); - const [selectedTab, setTab] = useState(); const { members } = hooks.useGetWorkspaceMembers(workspaceId); const { workspace } = hooks.useGetWorkspace(workspaceId); const { userId: localUserId } = hooks.useGetUserId(); - useEffect(() => { - if (view) { - const views: Record = { - billing: "Billing", - members: "Team Members", - api: "API Keys", - }; - setTab(views[view]); - } - }, [view]); - if (!(workspaceId && workspace)) { return null; } @@ -263,7 +262,7 @@ function WorkspaceSettingsModal({ view, ...rest }: PropsFromRedux) { ); } - -const connector = connect( - (state: UIState) => { - const opts = selectors.getModalOptions(state); - const view = opts && "view" in opts ? opts.view : null; - return { view }; - }, - { - hideModal: actions.hideModal, - } -); -export type PropsFromRedux = ConnectedProps; - -export default connector(WorkspaceSettingsModal); From cfee84da5e0f11e81118dd5e50c4be57c315d344 Mon Sep 17 00:00:00 2001 From: Mark Erikson Date: Wed, 24 Aug 2022 14:41:13 -0400 Subject: [PATCH 2/2] Clean up component logic a bit --- src/ui/components/Timeline/EventMarker.tsx | 1 + .../WorkspaceSettingsModal.tsx | 29 ++++++++++--------- 2 files changed, 16 insertions(+), 14 deletions(-) diff --git a/src/ui/components/Timeline/EventMarker.tsx b/src/ui/components/Timeline/EventMarker.tsx index f8d8d9e01c5..779bdbcedc1 100644 --- a/src/ui/components/Timeline/EventMarker.tsx +++ b/src/ui/components/Timeline/EventMarker.tsx @@ -4,6 +4,7 @@ import { ReplayEvent } from "ui/state/app"; import { useAppSelector } from "ui/setup/hooks"; import Marker from "./Marker"; +// TODO This component doesn't appear to be used right now? export default function EventMarker({ event, isPrimaryHighlighted }: EventMarkerProps) { const zoomRegion = useAppSelector(selectors.getZoomRegion); const currentTime = useAppSelector(selectors.getCurrentTime); diff --git a/src/ui/components/shared/WorkspaceSettingsModal/WorkspaceSettingsModal.tsx b/src/ui/components/shared/WorkspaceSettingsModal/WorkspaceSettingsModal.tsx index 0eb2a8ac2fa..3055c5f9b19 100644 --- a/src/ui/components/shared/WorkspaceSettingsModal/WorkspaceSettingsModal.tsx +++ b/src/ui/components/shared/WorkspaceSettingsModal/WorkspaceSettingsModal.tsx @@ -1,4 +1,4 @@ -import React, { ChangeEvent, useState } from "react"; +import React, { ChangeEvent, useState, useMemo } from "react"; import * as actions from "ui/actions/app"; import hooks from "ui/hooks"; import { useAppSelector, useAppDispatch } from "ui/setup/hooks"; @@ -28,9 +28,11 @@ export function WorkspaceMembers({ members: WorkspaceUser[]; isAdmin: boolean; }) { - const sortedMembers = members.sort( - (a: WorkspaceUser, b: WorkspaceUser) => Number(b.pending) - Number(a.pending) - ); + const sortedMembers = useMemo(() => { + return members + .slice() + .sort((a: WorkspaceUser, b: WorkspaceUser) => Number(b.pending) - Number(a.pending)); + }, [members]); const canLeave = members.length > 1; const canAdminLeave = canLeave && members.filter(a => a.roles?.includes("admin")).length > 1; @@ -61,7 +63,7 @@ type WorkspaceFormProps = { members?: WorkspaceUser[]; }; -function WorkspaceForm({ members }: WorkspaceFormProps) { +function WorkspaceForm({ members = [] }: WorkspaceFormProps) { const workspaceId = useGetTeamIdFromRoute(); const [inputValue, setInputValue] = useState(""); const [errorMessage, setErrorMessage] = useState(null); @@ -71,7 +73,7 @@ function WorkspaceForm({ members }: WorkspaceFormProps) { setIsLoading(false); }); - const memberEmails = (members || []).filter(m => m.email).map(m => m.email!); + const memberEmails = members.filter(m => m.email).map(m => m.email!); const onChange = (e: ChangeEvent) => { setInputValue(e.target.value); }; @@ -220,18 +222,17 @@ const settings: Settings< }, ]; +const tabNameForView = { + billing: "Billing", + members: "Team Members", + api: "API Keys", +} as const; + export default function WorkspaceSettingsModal() { const selectedTab = useAppSelector(state => { const opts = selectors.getModalOptions(state); const view = opts && "view" in opts ? opts.view : null; - return ( - view && - { - billing: "Billing", - members: "Team Members", - api: "API Keys", - }[view] - ); + return view && tabNameForView[view as keyof typeof tabNameForView]; }); const workspaceId = useGetTeamIdFromRoute(); const { members } = hooks.useGetWorkspaceMembers(workspaceId);