From 4dc398d3e55eb01ad4db1a4c8fbf8f19cf266e32 Mon Sep 17 00:00:00 2001 From: Federico Ruggi <1081051+ruggi@users.noreply.github.com> Date: Wed, 10 Jan 2024 17:22:20 +0100 Subject: [PATCH] Input/controls constraints for viewers (#4713) * disable inputs for non-editors * update snaps * update snaps * fix guideline test * snaps * update tests with store hook * useIsMyProject * revert --- .../canvas/controls/new-canvas-controls.tsx | 65 ++++++++++++------- .../components/canvas/design-panel-root.tsx | 4 +- .../components/editor/global-shortcuts.tsx | 3 + .../editor/store/collaborative-editing.ts | 8 +++ .../inspector/controls/color-control.tsx | 4 ++ .../controls/color-picker-swatches.tsx | 9 ++- .../inspector/controls/color-picker.tsx | 12 ++++ .../inspector/controls/option-control.tsx | 5 +- .../inspector/controls/slider-control.tsx | 5 +- editor/src/components/inspector/inspector.tsx | 24 ++++--- .../background-picker.tsx | 5 ++ editor/src/components/user-bar.tsx | 10 ++- ...performance-regression-tests.spec.tsx.snap | 14 ++-- .../performance-regression-tests.spec.tsx | 8 +-- editor/src/uuiui/inputs/number-input.tsx | 19 +++++- .../inputs/string-input.spec.browser2.tsx | 62 +++++++++++++----- editor/src/uuiui/inputs/string-input.tsx | 5 +- .../uuiui/widgets/popup-list/popup-list.tsx | 6 +- 18 files changed, 192 insertions(+), 76 deletions(-) diff --git a/editor/src/components/canvas/controls/new-canvas-controls.tsx b/editor/src/components/canvas/controls/new-canvas-controls.tsx index 9dacd48d7ad5..1d705eff25fc 100644 --- a/editor/src/components/canvas/controls/new-canvas-controls.tsx +++ b/editor/src/components/canvas/controls/new-canvas-controls.tsx @@ -66,9 +66,10 @@ import { import { useSelectionArea } from './selection-area-hooks' import { RemixSceneLabelControl } from './select-mode/remix-scene-label' import { NO_OP } from '../../../core/shared/utils' +import { useIsMyProject } from '../../editor/store/collaborative-editing' +import { useStatus } from '../../../../liveblocks.config' import { MultiplayerWrapper } from '../../../utils/multiplayer-wrapper' import { MultiplayerPresence } from '../multiplayer-presence' -import { useStatus } from '../../../../liveblocks.config' export const CanvasControlsContainerID = 'new-canvas-controls-container' @@ -501,6 +502,8 @@ const NewCanvasControlsInner = (props: NewCanvasControlsInnerProps) => { }) } + const isMyProject = useIsMyProject() + const resizeStatus = getResizeStatus() return ( @@ -531,42 +534,56 @@ const NewCanvasControlsInner = (props: NewCanvasControlsInnerProps) => { {when( resizeStatus !== 'disabled', <> - {inspectorFocusedControls.map((c) => ( - - ))} - {inspectorHoveredControls.map((c) => ( - - ))} - {when( - isSelectMode(editorMode), - , - )} - {when(isSelectMode(editorMode) && !anyStrategyActive, )} - {when(isSelectMode(editorMode), )} {renderHighlightControls()} - {renderTextEditableControls()} {unless(dragInteractionActive, )} - {when(isSelectMode(editorMode), )} + {when( - isSelectOrInsertMode(editorMode) && - !EP.multiplePathsAllWithTheSameUID(localSelectedViews), + isMyProject, <> - {strategyControls.map((c) => ( + {inspectorFocusedControls.map((c) => ( + + ))} + {inspectorHoveredControls.map((c) => ( ))} + {when( + isSelectMode(editorMode), + , + )} + {when(isSelectMode(editorMode) && !anyStrategyActive, )} + {when(isSelectMode(editorMode), )} + {renderTextEditableControls()} + {when(isSelectMode(editorMode), )} + {when( + isSelectOrInsertMode(editorMode) && + !EP.multiplePathsAllWithTheSameUID(localSelectedViews), + <> + {strategyControls.map((c) => ( + + ))} + , + )} + {when(isSelectMode(editorMode), )} + , )} - {when(isSelectMode(editorMode), )} - {when( roomStatus === 'connected', diff --git a/editor/src/components/canvas/design-panel-root.tsx b/editor/src/components/canvas/design-panel-root.tsx index c50b27f8da25..124a88253e5e 100644 --- a/editor/src/components/canvas/design-panel-root.tsx +++ b/editor/src/components/canvas/design-panel-root.tsx @@ -20,7 +20,7 @@ import { import { ConsoleAndErrorsPane } from '../code-editor/console-and-errors-pane' import { CanvasStrategyInspector } from './canvas-strategies/canvas-strategy-inspector' import { getQueryParam } from '../../common/env-vars' -import { unless, when } from '../../utils/react-conditionals' +import { when } from '../../utils/react-conditionals' import { InsertMenuPane } from '../navigator/insert-menu-pane' import { VariablesMenuPane } from '../navigator/variables-menu-pane' import { useDispatch } from '../editor/store/dispatch-context' @@ -31,9 +31,7 @@ import { SettingsPane } from '../navigator/left-pane/settings-pane' import { MenuTab } from '../../uuiui/menu-tab' import { FlexRow } from 'utopia-api' import type { StoredPanel } from './stored-layout' -import { MultiplayerPresence } from './multiplayer-presence' import { useRoom, useStatus } from '../../../liveblocks.config' -import { MultiplayerWrapper } from '../../utils/multiplayer-wrapper' import { isFeatureEnabled } from '../../utils/feature-switches' import { CommentsPane } from '../inspector/comments-pane' import { EditorModes, isCommentMode } from '../editor/editor-modes' diff --git a/editor/src/components/editor/global-shortcuts.tsx b/editor/src/components/editor/global-shortcuts.tsx index 90d532d60996..8ac03db4f830 100644 --- a/editor/src/components/editor/global-shortcuts.tsx +++ b/editor/src/components/editor/global-shortcuts.tsx @@ -686,6 +686,9 @@ export function handleKeyDown( ) }, [INSERT_DIV_SHORTCUT]: () => { + if (!allowedToEdit) { + return [] + } if (!isSelectMode(editor.mode) && !isInsertMode(editor.mode)) { return [] } diff --git a/editor/src/components/editor/store/collaborative-editing.ts b/editor/src/components/editor/store/collaborative-editing.ts index 2882d09a650b..ba2200e3ac08 100644 --- a/editor/src/components/editor/store/collaborative-editing.ts +++ b/editor/src/components/editor/store/collaborative-editing.ts @@ -790,3 +790,11 @@ export function useAllowedToEditProject(): boolean { 'useAllowedToEditProject', ) } + +export function useIsMyProject(): boolean { + return useEditorState( + Substores.projectServerState, + (store) => store.projectServerState.isMyProject === 'yes', + 'useIsMyProject', + ) +} diff --git a/editor/src/components/inspector/controls/color-control.tsx b/editor/src/components/inspector/controls/color-control.tsx index c0dc93c8623f..ea65f96cb403 100644 --- a/editor/src/components/inspector/controls/color-control.tsx +++ b/editor/src/components/inspector/controls/color-control.tsx @@ -8,6 +8,7 @@ import type { ControlStyles } from '../common/control-styles' import type { ControlStatus } from '../common/control-status' import { useColorTheme, UtopiaTheme } from '../../../uuiui' import Utils from '../../../utils/utils' +import { useIsMyProject } from '../../editor/store/collaborative-editing' export interface ColorControlProps { value: CSSColor @@ -81,6 +82,8 @@ export const ColorControl = React.memo((props: ColorControlProps) => { const closePopup = React.useCallback(() => setPopupOpen(false), [setPopupOpen]) + const isMyProject = useIsMyProject() + const picker = !popupOpen ? null : ( { value={props.value} onSubmitValue={props.onSubmitValue} onTransientSubmitValue={props.onTransientSubmitValue} + disabled={!isMyProject} /> ) diff --git a/editor/src/components/inspector/controls/color-picker-swatches.tsx b/editor/src/components/inspector/controls/color-picker-swatches.tsx index 455147947192..7057ec8118d2 100644 --- a/editor/src/components/inspector/controls/color-picker-swatches.tsx +++ b/editor/src/components/inspector/controls/color-picker-swatches.tsx @@ -15,6 +15,7 @@ import { useDispatch } from '../../editor/store/dispatch-context' import type { ColorSwatch } from '../../editor/store/editor-state' import { Substores, useEditorState } from '../../editor/store/store-hook' import { cssColorToChromaColorOrDefault } from '../common/css-utils' +import { useIsMyProject } from '../../editor/store/collaborative-editing' const { checkerboardBackground } = UtopiaStyles.backgrounds @@ -53,9 +54,14 @@ export const ColorPickerSwatches = React.memo((props: ColorPickerSwatchesProps) } }, [colorSwatches]) + const isMyProject = useIsMyProject() + const toggleEditing = React.useCallback(() => { + if (!isMyProject) { + return + } setEditing(!editing) - }, [editing]) + }, [editing, isMyProject]) const onAddColor = React.useCallback(() => { if (currentColor == null) { @@ -106,6 +112,7 @@ export const ColorPickerSwatches = React.memo((props: ColorPickerSwatchesProps) style={{ padding: '0 6px', }} + disabled={!isMyProject} onMouseDown={toggleEditing} > {when(editing, )} diff --git a/editor/src/components/inspector/controls/color-picker.tsx b/editor/src/components/inspector/controls/color-picker.tsx index d01eb28057d6..837564412a84 100644 --- a/editor/src/components/inspector/controls/color-picker.tsx +++ b/editor/src/components/inspector/controls/color-picker.tsx @@ -31,6 +31,7 @@ const checkerboardBackground = UtopiaStyles.backgrounds.checkerboardBackground export interface ColorPickerProps extends ColorPickerInnerProps { closePopup: () => void portalTarget?: HTMLElement + disabled: boolean } export const colorPickerWidth = 290 @@ -96,6 +97,7 @@ export interface ColorPickerInnerProps { offsetY: number id: string testId: string + disabled: boolean } function toPassedInColorType( @@ -340,6 +342,9 @@ export class ColorPickerInner extends React.Component< onMouseDownSV = (e: React.MouseEvent) => { e.stopPropagation() + if (this.props.disabled) { + return + } this.setState({ isScrubbing: true, }) @@ -399,6 +404,9 @@ export class ColorPickerInner extends React.Component< onHueSliderMouseDown = (e: React.MouseEvent) => { e.stopPropagation() + if (this.props.disabled) { + return + } if (this.HueControlRef.current != null) { this.HueOriginLeft = this.HueControlRef.current.getBoundingClientRect().left @@ -437,6 +445,10 @@ export class ColorPickerInner extends React.Component< onAlphaSliderMouseDown = (e: React.MouseEvent) => { e.stopPropagation() + if (this.props.disabled) { + return + } + if (this.AlphaControlRef.current != null) { this.setState({ isScrubbing: true, diff --git a/editor/src/components/inspector/controls/option-control.tsx b/editor/src/components/inspector/controls/option-control.tsx index df1b3518c818..a5c3ad26cee2 100644 --- a/editor/src/components/inspector/controls/option-control.tsx +++ b/editor/src/components/inspector/controls/option-control.tsx @@ -6,6 +6,7 @@ import type { DEPRECATEDControlProps, DEPRECATEDGenericControlOptions } from './ import { colorTheme } from '../../../uuiui' import type { IcnProps } from '../../../uuiui' import { UtopiaTheme, Tooltip, Icn } from '../../../uuiui' +import { useIsMyProject } from '../../editor/store/collaborative-editing' export interface DEPRECATEDOptionControlOptions extends DEPRECATEDGenericControlOptions { icon?: IcnProps @@ -60,7 +61,7 @@ export const OptionControl: React.FunctionComponent< } } - const rc = controlOptions.roundCorners + const isMyProject = useIsMyProject() const background = (() => { if (props.controlStatus === 'overridden' && props.value) { @@ -133,7 +134,7 @@ export const OptionControl: React.FunctionComponent< }} type='checkbox' checked={isChecked} - disabled={!props.controlStyles.interactive} + disabled={!props.controlStyles.interactive || !isMyProject} onChange={onSubmitValue} />
((props: InspectorProps) => { 'Inspector anyKnownElements', ) + const isMyProject = useIsMyProject() + function renderInspectorContents() { return ( @@ -378,18 +380,20 @@ export const Inspector = React.memo((props: InspectorProps) => { )} - {unless( hideAllSections, <> - + {when( + isMyProject, + , + )} {when(multiselectedContract === 'fragment', )} {unless( multiselectedContract === 'fragment', diff --git a/editor/src/components/inspector/sections/style-section/background-subsection/background-picker.tsx b/editor/src/components/inspector/sections/style-section/background-subsection/background-picker.tsx index f1e14a0a3704..85b1bcd2d32f 100644 --- a/editor/src/components/inspector/sections/style-section/background-subsection/background-picker.tsx +++ b/editor/src/components/inspector/sections/style-section/background-subsection/background-picker.tsx @@ -73,6 +73,7 @@ import { GradientStopsEditor } from './gradient-stop-editor' import { getIndexedUpdateCSSBackgroundLayerLinearGradientAngle } from './linear-gradient-layer' import { PickerImagePreview } from './picker-image-preview' import { setProp_UNSAFE } from '../../../../editor/actions/action-creators' +import { useIsMyProject } from '../../../../editor/store/collaborative-editing' const backgroundLayerOptionsByValue: { [key in CSSBackgroundLayerType]: CSSBackgroundLayerTypeSelectOption @@ -545,6 +546,8 @@ export const BackgroundPicker: React.FunctionComponent< } })() + const isMyProject = useIsMyProject() + return ( ) : ( ) ) : null} diff --git a/editor/src/components/user-bar.tsx b/editor/src/components/user-bar.tsx index 2da2ed0b3102..f8c63e1afa36 100644 --- a/editor/src/components/user-bar.tsx +++ b/editor/src/components/user-bar.tsx @@ -21,6 +21,7 @@ import { showToast, switchEditorMode } from './editor/actions/action-creators' import { EditorModes, isFollowMode } from './editor/editor-modes' import { useDispatch } from './editor/store/dispatch-context' import { Substores, useEditorState } from './editor/store/store-hook' +import { useIsMyProject } from './editor/store/collaborative-editing' const MAX_VISIBLE_OTHER_PLAYERS = 4 @@ -58,11 +59,8 @@ export const SinglePlayerUserBar = React.memo(() => { (store) => getUserPicture(store.userState.loginState), 'SinglePlayerUserBar userPicture', ) - const amIOwner = useEditorState( - Substores.projectServerState, - (store) => store.projectServerState.isMyProject === 'yes', - 'SinglePlayerUserBar amIOwner', - ) + const isMyProject = useIsMyProject() + return (
{ }} > - {amIOwner ? : null} + {isMyProject ? : null}
) }) diff --git a/editor/src/core/performance/__snapshots__/performance-regression-tests.spec.tsx.snap b/editor/src/core/performance/__snapshots__/performance-regression-tests.spec.tsx.snap index ace17ae7bcdc..121a2b457902 100644 --- a/editor/src/core/performance/__snapshots__/performance-regression-tests.spec.tsx.snap +++ b/editor/src/core/performance/__snapshots__/performance-regression-tests.spec.tsx.snap @@ -33,6 +33,7 @@ Array [ "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", + "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerPresence)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerWrapper)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", @@ -40,14 +41,14 @@ Array [ "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(SelectionAreaRectangle)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", - "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//UtopiaSpiedExoticType(Symbol(react.fragment))", - "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", "/UtopiaSpiedExoticType(Symbol(react.fragment))///div", "//div//div:data-testid='multiselect-outline'", "//div//div:data-testid='multiselect-element-outline-utopia-storyboard-uid/scene-aaa/app-entity:parent/ccc'", "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//UtopiaSpiedExoticType(Symbol(react.fragment))", + "/UtopiaSpiedExoticType(Symbol(react.fragment))/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", + "/UtopiaSpiedExoticType(Symbol(react.fragment))/UtopiaSpiedExoticType(Symbol(react.fragment))//UtopiaSpiedExoticType(Symbol(react.fragment))", "/UtopiaSpiedExoticType(Symbol(react.fragment))/Symbol(react.memo)()//Symbol(react.memo)()", "/Symbol(react.memo)()///Symbol(react.memo)(Symbol(react.forward_ref)())", "/Symbol(react.memo)()///Symbol(react.memo)(Symbol(react.forward_ref)())", @@ -645,6 +646,7 @@ Array [ "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", + "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerPresence)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerWrapper)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", @@ -652,14 +654,14 @@ Array [ "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(SelectionAreaRectangle)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", - "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//UtopiaSpiedExoticType(Symbol(react.fragment))", - "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", "/UtopiaSpiedExoticType(Symbol(react.fragment))///div", "//div//div:data-testid='multiselect-outline'", "//div//div:data-testid='multiselect-element-outline-utopia-storyboard-uid/scene-aaa/app-entity:parent/ccc'", "/div/UtopiaSpiedExoticType(Symbol(react.fragment))//UtopiaSpiedExoticType(Symbol(react.fragment))", + "/UtopiaSpiedExoticType(Symbol(react.fragment))/UtopiaSpiedExoticType(Symbol(react.fragment))//Symbol(react.memo)()", + "/UtopiaSpiedExoticType(Symbol(react.fragment))/UtopiaSpiedExoticType(Symbol(react.fragment))//UtopiaSpiedExoticType(Symbol(react.fragment))", "/UtopiaSpiedExoticType(Symbol(react.fragment))/Symbol(react.memo)()//Symbol(react.memo)()", "/Symbol(react.memo)()///Symbol(react.memo)(Symbol(react.forward_ref)())", "/Symbol(react.memo)()///Symbol(react.memo)(Symbol(react.forward_ref)())", @@ -1198,6 +1200,7 @@ Array [ "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", + "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerPresence)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerWrapper)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", @@ -1508,6 +1511,7 @@ Array [ "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", + "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerPresence)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerWrapper)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", @@ -1952,6 +1956,7 @@ Array [ "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", + "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerPresence)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerWrapper)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", @@ -2251,6 +2256,7 @@ Array [ "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)()", + "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerPresence)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/Symbol(react.memo)(MultiplayerWrapper)", "/div/div/UtopiaSpiedFunctionComponent(NewCanvasControlsInner)/UtopiaSpiedExoticType(Symbol(react.fragment))", diff --git a/editor/src/core/performance/performance-regression-tests.spec.tsx b/editor/src/core/performance/performance-regression-tests.spec.tsx index 07edcc7e4de8..4e1a16d9608d 100644 --- a/editor/src/core/performance/performance-regression-tests.spec.tsx +++ b/editor/src/core/performance/performance-regression-tests.spec.tsx @@ -65,7 +65,7 @@ describe('React Render Count Tests -', () => { const renderCountAfter = renderResult.getNumberOfRenders() // if this breaks, GREAT NEWS but update the test please :) - expect(renderCountAfter - renderCountBefore).toMatchInlineSnapshot(`704`) + expect(renderCountAfter - renderCountBefore).toMatchInlineSnapshot(`706`) expect(renderResult.getRenderInfo()).toMatchSnapshot() }) @@ -127,7 +127,7 @@ describe('React Render Count Tests -', () => { const renderCountAfter = renderResult.getNumberOfRenders() // if this breaks, GREAT NEWS but update the test please :) - expect(renderCountAfter - renderCountBefore).toMatchInlineSnapshot(`749`) + expect(renderCountAfter - renderCountBefore).toMatchInlineSnapshot(`751`) expect(renderResult.getRenderInfo()).toMatchSnapshot() }) @@ -183,7 +183,7 @@ describe('React Render Count Tests -', () => { const renderCountAfter = renderResult.getNumberOfRenders() // if this breaks, GREAT NEWS but update the test please :) - expect(renderCountAfter - renderCountBefore).toMatchInlineSnapshot(`535`) + expect(renderCountAfter - renderCountBefore).toMatchInlineSnapshot(`536`) expect(renderResult.getRenderInfo()).toMatchSnapshot() }) @@ -249,7 +249,7 @@ describe('React Render Count Tests -', () => { const renderCountAfter = renderResult.getNumberOfRenders() // if this breaks, GREAT NEWS but update the test please :) - expect(renderCountAfter - renderCountBefore).toMatchInlineSnapshot(`607`) + expect(renderCountAfter - renderCountBefore).toMatchInlineSnapshot(`608`) expect(renderResult.getRenderInfo()).toMatchSnapshot() }) }) diff --git a/editor/src/uuiui/inputs/number-input.tsx b/editor/src/uuiui/inputs/number-input.tsx index 20f0adfe78cb..db5e7cf393b0 100644 --- a/editor/src/uuiui/inputs/number-input.tsx +++ b/editor/src/uuiui/inputs/number-input.tsx @@ -50,6 +50,7 @@ import { InspectorInput, } from './base-input' import { usePropControlledStateV2 } from '../../components/inspector/common/inspector-utils' +import { useIsMyProject } from '../../components/editor/store/collaborative-editing' export type LabelDragDirection = 'horizontal' | 'vertical' @@ -188,6 +189,8 @@ export const NumberInput = React.memo( const ref = React.useRef(null) const colorTheme = useColorTheme() + const isMyProject = useIsMyProject() + const controlStyles = React.useMemo((): ControlStyles => { return { ...getControlStyles(controlStatus), @@ -553,6 +556,9 @@ export const NumberInput = React.memo( const onIncrementMouseDown = React.useCallback( (e: React.MouseEvent) => { + if (!isMyProject) { + return + } if (e.button === 0) { e.stopPropagation() setIsFauxcused(true) @@ -564,7 +570,7 @@ export const NumberInput = React.memo( }, repeatThreshold) } }, - [incrementBy, stepSize, repeatIncrement, onIncrementMouseUp], + [incrementBy, stepSize, repeatIncrement, onIncrementMouseUp, isMyProject], ) const onDecrementMouseUp = React.useCallback(() => { @@ -596,6 +602,9 @@ export const NumberInput = React.memo( const onDecrementMouseDown = React.useCallback( (e: React.MouseEvent) => { + if (!isMyProject) { + return + } if (e.button === 0) { e.stopPropagation() setIsFauxcused(true) @@ -608,11 +617,14 @@ export const NumberInput = React.memo( ) } }, - [incrementBy, stepSize, repeatIncrement, onDecrementMouseUp], + [incrementBy, stepSize, repeatIncrement, onDecrementMouseUp, isMyProject], ) const onLabelMouseDown = React.useCallback( (e: React.MouseEvent) => { + if (!isMyProject) { + return + } if (e.button === 0) { e.stopPropagation() setIsFauxcused(true) @@ -625,7 +637,7 @@ export const NumberInput = React.memo( setGlobalCursor?.(CSSCursor.ResizeEW) } }, - [scrubOnMouseMove, scrubOnMouseUp, setGlobalCursor, value], + [scrubOnMouseMove, scrubOnMouseUp, setGlobalCursor, value, isMyProject], ) const placeholder = getControlStylesAwarePlaceholder(controlStyles) @@ -678,6 +690,7 @@ export const NumberInput = React.memo( controlStyles={controlStyles} controlStatus={controlStatus} testId={testId} + disabled={!isMyProject} focused={isFocused} hasLabel={labelInner != null} roundCorners={roundCorners} diff --git a/editor/src/uuiui/inputs/string-input.spec.browser2.tsx b/editor/src/uuiui/inputs/string-input.spec.browser2.tsx index 5be069859643..862b30a9f3ff 100644 --- a/editor/src/uuiui/inputs/string-input.spec.browser2.tsx +++ b/editor/src/uuiui/inputs/string-input.spec.browser2.tsx @@ -3,6 +3,10 @@ import type { RenderResult } from '@testing-library/react' import { render } from '@testing-library/react' import React from 'react' import { StringInput } from './string-input' +import { + TestInspectorContextProvider, + getStoreHook, +} from '../../components/inspector/common/inspector.test-utils' describe('StringInput', () => { function checkPlaceholder(renderResult: RenderResult, expectedPlaceholder: string | null): void { @@ -14,36 +18,62 @@ describe('StringInput', () => { } } it('ensures that no placeholder property shows in the input field by default', () => { - const result = render() + const storeHookForTest = getStoreHook() + const result = render( + + + , + ) checkPlaceholder(result, null) }) it('ensures that the placeholder property shows in the input field', () => { + const storeHookForTest = getStoreHook() const result = render( - , + + + , ) checkPlaceholder(result, 'this is a placeholder') }) it('ensures that the unknown control styles property shows in the input field', () => { + const storeHookForTest = getStoreHook() const result = render( - , + + + , ) checkPlaceholder(result, 'Unknown') }) it('ensures that the mixed control styles property shows in the input field', () => { + const storeHookForTest = getStoreHook() const result = render( - , + + + , ) checkPlaceholder(result, 'Mixed') }) diff --git a/editor/src/uuiui/inputs/string-input.tsx b/editor/src/uuiui/inputs/string-input.tsx index 6091078de75d..670e7f9d68b3 100644 --- a/editor/src/uuiui/inputs/string-input.tsx +++ b/editor/src/uuiui/inputs/string-input.tsx @@ -10,6 +10,7 @@ import { getControlStyles } from '../../components/inspector/common/control-styl import { preventDefault, stopPropagation } from '../../components/inspector/common/inspector-utils' import { useColorTheme } from '../styles/theme' import { InspectorInputEmotionStyle, getControlStylesAwarePlaceholder } from './base-input' +import { useIsMyProject } from '../../components/editor/store/collaborative-editing' interface StringInputOptions { focusOnMount?: boolean @@ -50,8 +51,10 @@ export const StringInput = React.memo( } }, [focusOnMount, ref]) + const isMyProject = useIsMyProject() + const controlStyles: ControlStyles = getControlStyles(controlStatus) - const disabled = !controlStyles.interactive + const disabled = !controlStyles.interactive || !isMyProject const inputPropsKeyDown = inputProps.onKeyDown diff --git a/editor/src/uuiui/widgets/popup-list/popup-list.tsx b/editor/src/uuiui/widgets/popup-list/popup-list.tsx index 42772638e642..d938c2ea8771 100644 --- a/editor/src/uuiui/widgets/popup-list/popup-list.tsx +++ b/editor/src/uuiui/widgets/popup-list/popup-list.tsx @@ -27,6 +27,7 @@ import type { ControlStyles, SelectOption } from '../../../uuiui-deps' import { CommonUtils, getControlStyles } from '../../../uuiui-deps' import { SmallerIcons } from '../../../uuiui/icons' import { Tooltip } from '../../tooltip' +import { useIsMyProject } from '../../../components/editor/store/collaborative-editing' type ContainerMode = 'default' | 'showBorderOnHover' | 'noBorder' @@ -587,10 +588,13 @@ export const PopupList = React.memo( style, containerMode = 'default', controlStyles = getControlStyles('simple'), - disabled = !controlStyles.interactive, + disabled: initialDisabled, }, ref, ) => { + const isMyProject = useIsMyProject() + const disabled = initialDisabled || !controlStyles.interactive || !isMyProject + const selectOnSubmitValue = React.useCallback( (newValue: ValueType) => { if (isOptionType(newValue)) {