UX improvements

* When clicking on empty space in the snapshot editor, the selection reverts to the sheet containing the scene
This commit is contained in:
Aria Minaei 2021-07-31 15:10:08 +02:00
parent 0f7d918547
commit 2daa270879
6 changed files with 65 additions and 43 deletions

View file

@ -0,0 +1,24 @@
import React, {useLayoutEffect} from 'react'
import {useThree} from '@react-three/fiber'
import type {ISheet} from '@theatre/core'
import {bindToCanvas} from './store'
const Wrapper: React.FC<{
getSheet: () => ISheet
}> = (props) => {
const {scene, gl} = useThree((s) => ({scene: s.scene, gl: s.gl}))
useLayoutEffect(() => {
const sheet = props.getSheet()
if (!sheet || sheet.type !== 'Theatre_Sheet_PublicAPI') {
throw new Error(
`getSheet() in <Wrapper getSheet={getSheet}> has returned an invalid value`,
)
}
bindToCanvas({sheet})({gl, scene})
}, [scene, gl])
return <>{props.children}</>
}
export default Wrapper

View file

@ -1,4 +1,4 @@
import {useLayoutEffect} from 'react' import {useCallback, useLayoutEffect} from 'react'
import React from 'react' import React from 'react'
import {Canvas} from '@react-three/fiber' import {Canvas} from '@react-three/fiber'
import {useEditorStore} from '../store' import {useEditorStore} from '../store'
@ -100,11 +100,12 @@ const SnapshotEditor: React.FC<{object: ISheetObject<$FixMe>; paneId: string}> =
const snapshotEditorSheet = props.object.sheet const snapshotEditorSheet = props.object.sheet
const paneId = props.paneId const paneId = props.paneId
const [editorObject, sceneSnapshot, createSnapshot] = useEditorStore( const [editorObject, sceneSnapshot, createSnapshot, sheet] = useEditorStore(
(state) => [ (state) => [
state.editorObject, state.editorObject,
state.sceneSnapshot, state.sceneSnapshot,
state.createSnapshot, state.createSnapshot,
state.sheet,
], ],
shallow, shallow,
) )
@ -124,6 +125,10 @@ const SnapshotEditor: React.FC<{object: ISheetObject<$FixMe>; paneId: string}> =
} }
}, [editorOpen]) }, [editorOpen])
const onPointerMissed = useCallback(() => {
if (sheet !== null) studio.__experimental_setSelection([sheet])
}, [sheet])
if (!editorObject) return <></> if (!editorObject) return <></>
return ( return (
@ -154,9 +159,7 @@ const SnapshotEditor: React.FC<{object: ISheetObject<$FixMe>; paneId: string}> =
shadowMap shadowMap
dpr={[1, 2]} dpr={[1, 2]}
fog={'red'} fog={'red'}
onPointerMissed={() => onPointerMissed={onPointerMissed}
studio.__experimental_setSelection([])
}
> >
<EditorScene <EditorScene
snapshotEditorSheet={snapshotEditorSheet} snapshotEditorSheet={snapshotEditorSheet}

View file

@ -1,35 +1,13 @@
import SnapshotEditor from './components/SnapshotEditor' import SnapshotEditor from './components/SnapshotEditor'
export {default as EditorHelper} from './components/EditorHelper'
export type {EditorHelperProps} from './components/EditorHelper'
export {default as editable} from './components/editable'
export type {EditableState, BindFunction} from './store'
import studio from '@theatre/studio' import studio from '@theatre/studio'
import Toolbar from './components/Toolbar/Toolbar' import Toolbar from './components/Toolbar/Toolbar'
import {types} from '@theatre/core' import {types} from '@theatre/core'
import React, {useLayoutEffect} from 'react'
import {useThree} from '@react-three/fiber'
import type {ISheet} from '@theatre/core'
import {bindToCanvas} from './store'
export const Wrapper: React.FC<{ export {default as EditorHelper} from './components/EditorHelper'
getSheet: () => ISheet export type {EditorHelperProps} from './components/EditorHelper'
}> = (props) => { export {default as editable} from './components/editable'
const {scene, gl} = useThree((s) => ({scene: s.scene, gl: s.gl})) export type {EditableState, BindFunction} from './store'
export {default as Wrapper} from './Wrapper'
useLayoutEffect(() => {
const sheet = props.getSheet()
if (!sheet || sheet.type !== 'Theatre_Sheet_PublicAPI') {
throw new Error(
`getSheet() in <Wrapper getSheet={getSheet}> has returned an invalid value`,
)
}
bindToCanvas({sheet})({gl, scene})
}, [scene, gl])
return <>{props.children}</>
}
if (process.env.NODE_ENV === 'development') { if (process.env.NODE_ENV === 'development') {
studio.extend({ studio.extend({

View file

@ -10,11 +10,19 @@ import type {$IntentionalAny} from '@theatre/shared/utils/types'
const publicAPIToPrivateAPIMap = new WeakMap() const publicAPIToPrivateAPIMap = new WeakMap()
export function privateAPI(pub: IProject): Project export function privateAPI<
export function privateAPI(pub: ISheet): Sheet P extends IProject | ISheet | ISheetObject<$IntentionalAny> | ISequence,
export function privateAPI(pub: ISheetObject<$IntentionalAny>): SheetObject >(
export function privateAPI(pub: ISequence): Sequence pub: P,
export function privateAPI(pub: {}): unknown { ): P extends IProject
? Project
: P extends ISheet
? Sheet
: P extends ISheetObject<$IntentionalAny>
? SheetObject
: P extends ISequence
? Sequence
: never {
return publicAPIToPrivateAPIMap.get(pub) return publicAPIToPrivateAPIMap.get(pub)
} }

View file

@ -5,6 +5,8 @@ import type Sequence from './Sequence'
import type {IPlaybackDirection, IPlaybackRange} from './Sequence' import type {IPlaybackDirection, IPlaybackRange} from './Sequence'
export interface ISequence { export interface ISequence {
readonly type: 'Theatre_Sequence_PublicAPI'
/** /**
* Starts playback of a sequence. * Starts playback of a sequence.
* Returns a promise that either resolves to true when the playback completes, * Returns a promise that either resolves to true when the playback completes,
@ -24,7 +26,11 @@ export interface ISequence {
time: number time: number
} }
export default class TheatreSequence { export default class TheatreSequence implements ISequence {
get type(): 'Theatre_Sequence_PublicAPI' {
return 'Theatre_Sequence_PublicAPI'
}
/** /**
* @internal * @internal
*/ */

View file

@ -1,4 +1,4 @@
import type {ISheetObject} from '@theatre/core' import type {ISheet, ISheetObject} from '@theatre/core'
import studioTicker from '@theatre/studio/studioTicker' import studioTicker from '@theatre/studio/studioTicker'
import type {IDerivation, Pointer} from '@theatre/dataverse' import type {IDerivation, Pointer} from '@theatre/dataverse'
import {prism} from '@theatre/dataverse' import {prism} from '@theatre/dataverse'
@ -7,7 +7,10 @@ import type {$FixMe, $IntentionalAny, VoidFn} from '@theatre/shared/utils/types'
import type {IScrub} from '@theatre/studio/Scrub' import type {IScrub} from '@theatre/studio/Scrub'
import type {Studio} from '@theatre/studio/Studio' import type {Studio} from '@theatre/studio/Studio'
import {isSheetObjectPublicAPI} from '@theatre/shared/instanceTypes' import {
isSheetObjectPublicAPI,
isSheetPublicAPI,
} from '@theatre/shared/instanceTypes'
import {getOutlineSelection} from './selectors' import {getOutlineSelection} from './selectors'
import type SheetObject from '@theatre/core/sheetObjects/SheetObject' import type SheetObject from '@theatre/core/sheetObjects/SheetObject'
import getStudio from './getStudio' import getStudio from './getStudio'
@ -68,7 +71,7 @@ export interface IStudio {
scrub(): IScrub scrub(): IScrub
debouncedScrub(threshhold: number): Pick<IScrub, 'capture'> debouncedScrub(threshhold: number): Pick<IScrub, 'capture'>
__experimental_setSelection(selection: Array<ISheetObject>): void __experimental_setSelection(selection: Array<ISheetObject | ISheet>): void
__experimental_onSelectionChange( __experimental_onSelectionChange(
fn: (s: Array<ISheetObject>) => void, fn: (s: Array<ISheetObject>) => void,
): VoidFunction ): VoidFunction
@ -136,9 +139,9 @@ export default class TheatreStudio implements IStudio {
return this._getSelectionDerivation().getValue() return this._getSelectionDerivation().getValue()
} }
__experimental_setSelection(selection: Array<ISheetObject>): void { __experimental_setSelection(selection: Array<ISheetObject | ISheet>): void {
const sanitizedSelection = [...selection] const sanitizedSelection = [...selection]
.filter((s) => isSheetObjectPublicAPI(s)) .filter((s) => isSheetObjectPublicAPI(s) || isSheetPublicAPI(s))
.map((s) => getStudio().corePrivateAPI!(s)) .map((s) => getStudio().corePrivateAPI!(s))
getStudio().transaction(({stateEditors}) => { getStudio().transaction(({stateEditors}) => {