Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat/UI useredit panel #22

Merged
merged 93 commits into from
Dec 13, 2024
Merged
Show file tree
Hide file tree
Changes from 79 commits
Commits
Show all changes
93 commits
Select commit Hold shift + click to select a range
2b4c08c
wip: initial draft implementation of userEditPanel
olzzon Oct 14, 2024
c1e8e10
wip: userEditPanel grouping
olzzon Oct 15, 2024
fb54ad2
wip: userEditPanel cleanup remove schema as it's replaces by schemas
olzzon Oct 16, 2024
7fd9bf1
wip: userEditPanel styling and use <select> for list
olzzon Oct 16, 2024
040dbc4
wip: userEditPanel - make groupType optional to allow empty slots
olzzon Oct 16, 2024
d49b265
wip: userEditPanel - convert to FC
olzzon Oct 16, 2024
d93d0a9
wip: userEditPanel - uncomment context menu
olzzon Oct 17, 2024
9cbc324
wip: UserEditPanel - Add DefaultUserOperationsTypes for revert button…
olzzon Oct 17, 2024
0643ff3
wip: UserEditPanel - Default revert button on panel
olzzon Oct 17, 2024
9d46129
wip: UserEditPanel - tracker on part and segment
olzzon Oct 17, 2024
27a2680
wip: UserEditPanel - only render Segment or Part in panel
olzzon Oct 21, 2024
298d10c
wip: UserEditPanel - initial header for part and segment
olzzon Oct 21, 2024
526211e
wip: UserEditPanel - comment for the usage of getTimePosition
olzzon Oct 21, 2024
a673847
wip: UserEditPanel - add Icons in header
olzzon Oct 21, 2024
5324fa3
wip: UserEditPanel - add button type in UserEditingDefinitions
olzzon Oct 21, 2024
c8b7e79
wip: UserEditPanel - rename revert button to "Revert Changes"
olzzon Oct 21, 2024
a1fc06b
wip: UserEditPanel - add support for SVG icons in grouping (e.g for s…
olzzon Oct 22, 2024
c76ec35
wip: UserEditPanel - center revert button
olzzon Oct 22, 2024
c6d0bf5
wip: Renaming UserEditPanel to PropertiesPanel
olzzon Oct 22, 2024
163cb46
wip: UI selection - Context for SelectedElements
olzzon Oct 23, 2024
e037d59
wip: UI selection - refactor from useState() to useReducer()
olzzon Oct 24, 2024
7fe136f
wip: UI selection - use type for SelectedElement
olzzon Oct 24, 2024
061d61f
wip: UI selection -Segment not updating upon selection
olzzon Oct 24, 2024
2a252e8
feat: unit tests for SelectedElementsContext
olzzon Oct 24, 2024
1710bb4
wip: UI selection - add isSelected to withSegmentTiming
olzzon Oct 25, 2024
c00b2fd
wip: fix props withSelection
olzzon Oct 25, 2024
3a3b9ff
wip: fix props withSelection
olzzon Oct 25, 2024
02b9a41
wip: remove componentDidUpdate - leftover from previous debugging
olzzon Oct 25, 2024
030163e
fix: added listSelectedElements() instead of exposing the selectedEle…
olzzon Oct 25, 2024
ad46b36
fix: add assertNever() to selectionReducer
olzzon Oct 25, 2024
e760534
fix: remove id and use ElementId as reference in context provider
olzzon Oct 25, 2024
73753f8
fix: clean up isSelected should parse boolean not function as props
olzzon Oct 25, 2024
14e03fe
chore: update package with @testing-library/react
olzzon Oct 25, 2024
cb43189
fix: tests for SelecedElementsContext updated and type fixed
olzzon Oct 25, 2024
875a438
fix: listSelectedElements is af function not an array
olzzon Oct 25, 2024
382179c
wip: implement useSelection in PropertiesPanel
olzzon Oct 25, 2024
3222f65
wip: useSelection on Parts
olzzon Oct 25, 2024
ef2c47d
fix: missing rundown id in propertiespanel actions
olzzon Oct 25, 2024
2ad708a
feat: add enabling of userediting in settings
olzzon Oct 31, 2024
7a79afa
feat: doubleclick on part selection for properties panel
olzzon Oct 31, 2024
2169357
fix: properties panel crash when no selection was made
olzzon Oct 31, 2024
a6a431c
feat: segment selection
olzzon Oct 31, 2024
8dc7700
feat: dbl click anywhere in segment header to select properties
olzzon Oct 31, 2024
48521c6
feat: select part by double clicking a piece
olzzon Oct 31, 2024
8220392
feat: properties panel commit button for pending changes
olzzon Oct 31, 2024
3aaea87
feat: select/deselect an element - fix flickering upen commit changes
olzzon Nov 1, 2024
07bd3ec
fix: render all userEditOperations when part is selected
olzzon Nov 1, 2024
ba03df0
feat: move Properties panel out of notification center logic
olzzon Nov 4, 2024
52a0d54
feat: use context.consumer instead of wrapping
olzzon Nov 4, 2024
8bdc2cd
feat: refactor userEditing structure
olzzon Nov 6, 2024
662681e
feat: properties panel with refactored data structure.
olzzon Nov 6, 2024
091af40
feat: Schema selection implemented in commit button logic
olzzon Nov 6, 2024
4d58ce8
feat: implement Layer colors in group selectors
olzzon Nov 6, 2024
4ef35fd
feat: only parse the selected source on useraction commit
olzzon Nov 7, 2024
525cd59
feat: simplify structure for userEditing source
olzzon Nov 7, 2024
3a5bda0
fix: rerender group selection when selecting new part
olzzon Nov 7, 2024
5e547ce
fix: read the SourceLayerType into UserEditingDefinition
olzzon Nov 7, 2024
3978c2b
feat: selected element indicator
olzzon Nov 7, 2024
6470d43
feat: properties panel styling - is edited indication
olzzon Nov 8, 2024
b674e2d
wip: add close propterties icon (sketch)
olzzon Nov 8, 2024
4bd57d6
wip: properties panel css dim commit+revert when not active
olzzon Nov 8, 2024
cb2455b
wip: properties panel clean up first iteration selector
olzzon Nov 8, 2024
b4d75dd
wip: properties panel add close upper right
olzzon Nov 8, 2024
15d77e8
fix: crash UI - remove transiongroup
olzzon Nov 8, 2024
7b65130
feat: properties panel animate in and shrink rundown view
olzzon Nov 8, 2024
8666da3
fix: cleanup using notificationpanel for properties panel
olzzon Nov 8, 2024
631ef97
wip: properties panel, rightbar icon color
olzzon Nov 8, 2024
ad78549
fix: re-run only if part.segmentId has changed
olzzon Nov 9, 2024
d3e0ca7
wip: tests for properties panel
olzzon Nov 9, 2024
5fe974f
wip: properties panel tests - useFaketimers and implement mock useTra…
olzzon Nov 11, 2024
b86bc21
wip: properties panel - element selection styling
olzzon Nov 11, 2024
e6d4c94
wip: properties panel styling edit pencil icon
olzzon Nov 11, 2024
e0969ef
wip: properties panel close when notification is open
olzzon Nov 12, 2024
6a10475
wip: properties panel selection glow on segment header
olzzon Nov 12, 2024
61c73a0
feat: StyledSchemaFormInPlace schema component
olzzon Nov 13, 2024
958d8e3
fix: Properties panel - StyledSchemaFormInPlace schema for normal form
olzzon Nov 13, 2024
7f5fa09
feat: Properties panel is edited pencil styling
olzzon Nov 13, 2024
d623cc6
feat: Properties header styling
olzzon Nov 13, 2024
75b1930
wip: Properties tests - more mocking to get correct rendering in test…
olzzon Nov 13, 2024
91dae54
chore: fix unit tests
mint-dewit Nov 21, 2024
77bf227
feat: refactor form action to properties field
mint-dewit Nov 27, 2024
aa35b1b
chore: update styles
mint-dewit Dec 2, 2024
ea2fb4a
chore: add actions back into property panel
mint-dewit Dec 2, 2024
a671baa
chore: support translations for properties panel
mint-dewit Dec 5, 2024
53488d2
chore: fix some build issues
mint-dewit Dec 5, 2024
ffdde73
Merge pull request #38 from bbc/chore/ui-useredit-properties
mint-dewit Dec 5, 2024
80ae9d8
feat: allow editing piece properties
mint-dewit Dec 6, 2024
50d869e
chore: various review comments
mint-dewit Dec 10, 2024
e4cf90e
chore: fix tests
mint-dewit Dec 11, 2024
91e9431
chore: merge bbc-release52 into feat/ui-useredit-panel
mint-dewit Dec 11, 2024
8f9df89
chore: merge bbc-release52 into feat/useredit-panel
mint-dewit Dec 12, 2024
dedebee
chore: remove unused parameter
mint-dewit Dec 12, 2024
287f007
chore: removed commented styles
mint-dewit Dec 13, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions meteor/server/api/rest/v1/typeConversion.ts
Original file line number Diff line number Diff line change
Expand Up @@ -321,6 +321,7 @@ export function studioSettingsFrom(apiStudioSettings: APIStudioSettings): IStudi
enableQuickLoop: apiStudioSettings.enableQuickLoop,
forceQuickLoopAutoNext: forceQuickLoopAutoNextFrom(apiStudioSettings.forceQuickLoopAutoNext),
fallbackPartDuration: apiStudioSettings.fallbackPartDuration ?? DEFAULT_FALLBACK_PART_DURATION,
enableUserEdits: apiStudioSettings.enableUserEdits,
}
}

Expand All @@ -340,6 +341,7 @@ export function APIStudioSettingsFrom(settings: IStudioSettings): APIStudioSetti
enableQuickLoop: settings.enableQuickLoop,
forceQuickLoopAutoNext: APIForceQuickLoopAutoNextFrom(settings.forceQuickLoopAutoNext),
fallbackPartDuration: settings.fallbackPartDuration,
enableUserEdits: settings.enableUserEdits,
}
}

Expand Down
1 change: 1 addition & 0 deletions meteor/server/lib/rest/v1/studios.ts
Original file line number Diff line number Diff line change
Expand Up @@ -186,4 +186,5 @@ export interface APIStudioSettings {
forceQuickLoopAutoNext?: 'disabled' | 'enabled_when_valid_duration' | 'enabled_forcing_min_duration'
minimumTakeSpan?: number
fallbackPartDuration?: number
enableUserEdits?: boolean
}
28 changes: 26 additions & 2 deletions packages/blueprints-integration/src/ingest.ts
Original file line number Diff line number Diff line change
Expand Up @@ -124,11 +124,35 @@ export interface UserOperationTarget {
pieceExternalId: string | undefined
}

export type DefaultUserOperations = {
id: '__sofie-move-segment' // Future: define properly
export enum DefaultUserOperationsTypes {
REVERT_SEGMENT = 'revert-segment',
mint-dewit marked this conversation as resolved.
Show resolved Hide resolved
REVERT_PART = 'revert-part',
REVERT_RUNDOWN = 'revert-rundown',
}

export interface DefaultUserOperationRevertRundown {
id: DefaultUserOperationsTypes.REVERT_RUNDOWN
payload: Record<string, never>
}

export interface DefaultUserOperationRevertSegment {
id: DefaultUserOperationsTypes.REVERT_SEGMENT
payload: Record<string, never>
}

export interface DefaultUserOperationRevertPart {
id: DefaultUserOperationsTypes.REVERT_PART
}

export type DefaultUserOperations =
| {
id: '__sofie-move-segment' // Future: define properly
mint-dewit marked this conversation as resolved.
Show resolved Hide resolved
payload: Record<string, never>
}
| DefaultUserOperationRevertRundown
| DefaultUserOperationRevertSegment
| DefaultUserOperationRevertPart

export interface UserOperationChange<TCustomBlueprintOperations extends { id: string } = never> {
/** Indicate that this change is from user operations */
source: IngestChangeType.User
Expand Down
50 changes: 47 additions & 3 deletions packages/blueprints-integration/src/userEditing.ts
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
import { JSONBlob } from '@sofie-automation/shared-lib/dist/lib/JSONBlob'
import type { ITranslatableMessage } from './translations'
import { JSONSchema } from '@sofie-automation/shared-lib/dist/lib/JSONSchemaTypes'
import { SourceLayerType } from './content'

/**
* Description of a user performed editing operation allowed on an document
*/
export type UserEditingDefinition = UserEditingDefinitionAction | UserEditingDefinitionForm
export type UserEditingDefinition =
| UserEditingDefinitionAction
| UserEditingDefinitionForm
| UserEditingDefinitionSourceLayerForm

/**
* A simple 'action' that can be performed
Expand All @@ -16,10 +20,14 @@ export interface UserEditingDefinitionAction {
id: string
/** Label to show to the user for this operation */
label: ITranslatableMessage
/** Icon to show to when this action is 'active' */
/** Icon to show when this action is 'active' */
svgIcon?: string
/** Icon to show when this action is 'disabled' */
svgIconInactive?: string
/** Whether this action should be indicated as being active */
isActive?: boolean
/** Button Type */
buttonType?: UserEditingButtonType
}

/**
Expand All @@ -37,9 +45,45 @@ export interface UserEditingDefinitionForm {
currentValues: Record<string, any>
}

/**
* A form based operation where the user first selects the type
* of form they want to use (i.e. Camera form vs VT form)
*/
export interface UserEditingDefinitionSourceLayerForm {
type: UserEditingType.SOURCE_LAYER_FORM
/** Id of this operation */
id: string
/** Label to show to the user for this operation */
label: ITranslatableMessage
/** The json schemas describing the form to display */
schemas: Record<string, UserEditingSourceLayer>
/** Current values to populate the form with */
currentValues: {
type: SourceLayerType
mint-dewit marked this conversation as resolved.
Show resolved Hide resolved
value: Record<string, any>
}
}

export enum UserEditingType {
/** Action */
ACTION = 'action',
/** Form of selections */
/** Form */
FORM = 'form',
/** Forms that the user has to select a sourceLayerType first */
SOURCE_LAYER_FORM = 'sourceLayerForm',
}

export interface UserEditingSourceLayer {
sourceLayerLabel: string
sourceLayerType: SourceLayerType
schema: JSONBlob<JSONSchema>
}

export enum UserEditingButtonType {
/** Button */
BUTTON = 'button',
/** Icon */
SWITCH = 'switch',
/** Hidden */
HIDDEN = 'hidden',
}
5 changes: 5 additions & 0 deletions packages/corelib/src/dataModel/Studio.ts
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,11 @@ export interface IStudioSettings {
* Default: 3000
*/
fallbackPartDuration?: number

/**
* Doubleclick changes behaviour as selector for userediting
*/
enableUserEdits?: boolean
}

export type StudioLight = Omit<DBStudio, 'mappingsWithOverrides' | 'blueprintConfigWithOverrides'>
Expand Down
44 changes: 41 additions & 3 deletions packages/corelib/src/dataModel/UserEditingDefinitions.ts
Original file line number Diff line number Diff line change
@@ -1,20 +1,37 @@
import type { UserEditingType, JSONBlob, JSONSchema } from '@sofie-automation/blueprints-integration'
import type {
UserEditingType,
JSONBlob,
JSONSchema,
UserEditingSourceLayer,
UserEditingButtonType,
SourceLayerType,
} from '@sofie-automation/blueprints-integration'
import type { ITranslatableMessage } from '../TranslatableMessage'

export type CoreUserEditingDefinition = CoreUserEditingDefinitionAction | CoreUserEditingDefinitionForm
export type CoreUserEditingDefinition =
| CoreUserEditingDefinitionAction
| CoreUserEditingDefinitionForm
| CoreUserEditingDefinitionSourceLayerForm

export interface CoreUserEditingDefinitionAction {
type: UserEditingType.ACTION
/** Id of this operation */
id: string
/** Label to show to the user for this operation */
label: ITranslatableMessage
/** Icon to show to when this action is 'active' */
/** Icon to show when this action is 'active' */
svgIcon?: string
/** Icon to show when this action is 'disabled' */
svgIconInactive?: string
/** Whether this action should be indicated as being active */
isActive?: boolean
//** Button Type */
buttonType?: UserEditingButtonType
}

/**
* A simple form based operation
*/
export interface CoreUserEditingDefinitionForm {
type: UserEditingType.FORM
/** Id of this operation */
Expand All @@ -28,3 +45,24 @@ export interface CoreUserEditingDefinitionForm {
/** Translation namespaces to use when rendering this form */
translationNamespaces: string[]
}

/**
* A form based operation where the user first selects the type
* of form they want to use (i.e. Camera form vs VT form)
*/
export interface CoreUserEditingDefinitionSourceLayerForm {
type: UserEditingType.SOURCE_LAYER_FORM
/** Id of this operation */
id: string
/** Label to show to the user for this operation */
label: ITranslatableMessage
/** Sourcelayer Type */
schemas: Record<string, UserEditingSourceLayer>
/** Translation namespaces to use when rendering this form */
translationNamespaces: string[]
/** Current values to populate the form with */
currentValues: {
type: SourceLayerType
value: Record<string, any>
}
}
23 changes: 23 additions & 0 deletions packages/job-worker/src/blueprints/context/lib.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import {
CoreUserEditingDefinition,
CoreUserEditingDefinitionAction,
CoreUserEditingDefinitionForm,
CoreUserEditingDefinitionSourceLayerForm,
} from '@sofie-automation/corelib/dist/dataModel/UserEditingDefinitions'
import { DBSegment } from '@sofie-automation/corelib/dist/dataModel/Segment'
import { assertNever, clone, Complete, literal, omit } from '@sofie-automation/corelib/dist/lib'
Expand Down Expand Up @@ -56,6 +57,7 @@ import {
UserEditingDefinition,
UserEditingDefinitionAction,
UserEditingDefinitionForm,
UserEditingDefinitionSourceLayerForm,
UserEditingType,
} from '@sofie-automation/blueprints-integration/dist/userEditing'
import type { PlayoutMutatablePart } from '../../playout/model/PlayoutPartInstanceModel'
Expand Down Expand Up @@ -510,7 +512,9 @@ function translateUserEditsToBlueprint(
id: userEdit.id,
label: omit(userEdit.label, 'namespaces'),
svgIcon: userEdit.svgIcon,
svgIconInactive: userEdit.svgIconInactive,
isActive: userEdit.isActive,
buttonType: userEdit.buttonType,
} satisfies Complete<UserEditingDefinitionAction>
case UserEditingType.FORM:
return {
Expand All @@ -520,6 +524,14 @@ function translateUserEditsToBlueprint(
schema: clone(userEdit.schema),
currentValues: clone(userEdit.currentValues),
} satisfies Complete<UserEditingDefinitionForm>
case UserEditingType.SOURCE_LAYER_FORM:
return {
type: UserEditingType.SOURCE_LAYER_FORM,
id: userEdit.id,
label: omit(userEdit.label, 'namespaces'),
schemas: clone(userEdit.schemas),
currentValues: clone(userEdit.currentValues),
} satisfies Complete<UserEditingDefinitionSourceLayerForm>
default:
assertNever(userEdit)
return undefined
Expand All @@ -543,7 +555,9 @@ export function translateUserEditsFromBlueprint(
id: userEdit.id,
label: wrapTranslatableMessageFromBlueprints(userEdit.label, blueprintIds),
svgIcon: userEdit.svgIcon,
svgIconInactive: userEdit.svgIconInactive,
isActive: userEdit.isActive,
buttonType: userEdit.buttonType,
} satisfies Complete<CoreUserEditingDefinitionAction>
case UserEditingType.FORM:
return {
Expand All @@ -554,6 +568,15 @@ export function translateUserEditsFromBlueprint(
currentValues: clone(userEdit.currentValues),
translationNamespaces: unprotectStringArray(blueprintIds),
} satisfies Complete<CoreUserEditingDefinitionForm>
case UserEditingType.SOURCE_LAYER_FORM:
return {
type: UserEditingType.SOURCE_LAYER_FORM,
id: userEdit.id,
label: wrapTranslatableMessageFromBlueprints(userEdit.label, blueprintIds),
schemas: clone(userEdit.schemas),
currentValues: clone(userEdit.currentValues),
translationNamespaces: unprotectStringArray(blueprintIds),
} satisfies Complete<CoreUserEditingDefinitionSourceLayerForm>
default:
assertNever(userEdit)
return undefined
Expand Down
10 changes: 7 additions & 3 deletions packages/webui/jest.config.cjs
Original file line number Diff line number Diff line change
@@ -1,7 +1,11 @@
module.exports = {
setupFilesAfterEnv: ['./src/__mocks__/_setupMocks.ts', '<rootDir>/src/client/__tests__/jest-setup.cjs'],
setupFilesAfterEnv: [
'./src/__mocks__/_setupMocks.ts',
'<rootDir>/src/client/__tests__/jest-setup.cjs',
'@testing-library/jest-dom',
],
globals: {},
moduleFileExtensions: ['js', 'ts'],
moduleFileExtensions: ['js', 'ts', 'tsx'],
moduleNameMapper: {
'meteor/(.*)': '<rootDir>/src/meteor/$1',
},
Expand All @@ -15,7 +19,7 @@ module.exports = {
'^.+\\.(js|jsx)$': ['babel-jest', { presets: ['@babel/preset-env'] }],
},
transformIgnorePatterns: ['node_modules/(?!(nanoid)/)', '\\.pnp\\.[^\\/]+$'],
testMatch: ['**/__tests__/**/*.(spec|test).(ts|js)'],
testMatch: ['**/__tests__/**/*.(spec|test).(ts|tsx|js)'],
testPathIgnorePatterns: ['integrationTests'],
testEnvironment: 'jsdom',
// coverageThreshold: {
Expand Down
4 changes: 4 additions & 0 deletions packages/webui/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@
"@sofie-automation/meteor-lib": "1.52.0-in-development",
"@sofie-automation/shared-lib": "1.52.0-in-development",
"@sofie-automation/sorensen": "^1.4.3",
"@testing-library/user-event": "^14.5.2",
"@types/sinon": "^10.0.20",
"classnames": "^2.5.1",
"cubic-spline": "^3.0.3",
Expand Down Expand Up @@ -84,6 +85,9 @@
},
"devDependencies": {
"@babel/preset-env": "^7.24.8",
"@testing-library/dom": "^10.4.0",
"@testing-library/jest-dom": "^6.6.3",
"@testing-library/react": "^16.0.1",
"@types/classnames": "^2.3.1",
"@types/deep-extend": "^0.6.2",
"@types/react": "^18.3.3",
Expand Down
1 change: 1 addition & 0 deletions packages/webui/src/client/__tests__/jest-setup.cjs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
/* eslint-disable node/no-unpublished-require */
require('@testing-library/jest-dom')

// used by code creating XML with the DOM API to return an XML string
global.XMLSerializer = require('@xmldom/xmldom').XMLSerializer
Expand Down
2 changes: 1 addition & 1 deletion packages/webui/src/client/lib/Components/DropdownInput.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -112,7 +112,7 @@ export function DropdownInputControl<TValue>({
const newOptions = [
...options,
{
name: 'Value: ' + value,
name: String(value),
value: value,
i: options.length,
},
Expand Down
Loading
Loading