diff --git a/packages/timeline-state-resolver-types/src/integrations/vmix.ts b/packages/timeline-state-resolver-types/src/integrations/vmix.ts index 1826497c9..01ffbef07 100644 --- a/packages/timeline-state-resolver-types/src/integrations/vmix.ts +++ b/packages/timeline-state-resolver-types/src/integrations/vmix.ts @@ -45,6 +45,7 @@ export enum VMixCommand { LIST_ADD = 'LIST_ADD', LIST_REMOVE_ALL = 'LIST_REMOVE_ALL', RESTART_INPUT = 'RESTART_INPUT', + BROWSER_NAVIGATE = 'BROWSER_NAVIGATE', } export type TimelineContentVMixAny = @@ -188,6 +189,9 @@ export interface TimelineContentVMixInput extends TimelineContentVMixBase { /** If media should start from the beginning or resume from where it left off */ restart?: boolean + + /** The URL for Browser input */ + url?: string } export interface TimelineContentVMixOutput extends TimelineContentVMixBase { @@ -325,4 +329,5 @@ export enum VMixInputType { Flash = 'Flash', PowerPoint = 'PowerPoint', List = 'List', + Browser = 'Browser', } diff --git a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/connection.spec.ts b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/connection.spec.ts index 6c6201fa8..869e1652b 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/connection.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/connection.spec.ts @@ -92,4 +92,19 @@ describe('VMixCommandSender', () => { value: 1.5, }) }) + + it('sends url', async () => { + const { sender, mockConnection } = createTestee() + await sender.sendCommand({ + command: VMixCommand.BROWSER_NAVIGATE, + input: 5, + value: 'https://example.com', + }) + + expect(mockConnection.sendCommandFunction).toHaveBeenCalledTimes(1) + expect(mockConnection.sendCommandFunction).toHaveBeenLastCalledWith('BrowserNavigate', { + input: 5, + value: 'https%3A%2F%2Fexample.com', + }) + }) }) diff --git a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vMixStateDiffer.spec.ts b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vMixStateDiffer.spec.ts index f2782ceb9..ed39f469b 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vMixStateDiffer.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vMixStateDiffer.spec.ts @@ -146,4 +146,26 @@ describe('VMixStateDiffer', () => { cropBottom: 0.8, }) }) + + it('sets browser url', () => { + const differ = createTestee() + + const oldState = makeMockFullState() + const newState = makeMockFullState() + + oldState.reportedState.existingInputs['99'] = differ.getDefaultInputState(99) + + newState.reportedState.existingInputs['99'] = differ.getDefaultInputState(99) + const url = 'https://example.com' + newState.reportedState.existingInputs['99'].url = url + + const commands = differ.getCommandsToAchieveState(Date.now(), oldState, newState) + + expect(commands.length).toBe(1) + expect(commands[0].command).toMatchObject({ + command: VMixCommand.BROWSER_NAVIGATE, + input: '99', + value: url, + }) + }) }) diff --git a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vMixTimelineStateConverter.spec.ts b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vMixTimelineStateConverter.spec.ts index 3d88cef0b..53d1e2289 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vMixTimelineStateConverter.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vMixTimelineStateConverter.spec.ts @@ -204,6 +204,26 @@ describe('VMixTimelineStateConverter', () => { }) expect(result.reportedState.mixes[0]?.program).toEqual(2) }) + it('supports url', () => { + const converter = createTestee() + const url = 'https://example.com' + const result = converter.getVMixStateFromTimelineState( + wrapInTimelineState({ + inp0: wrapInTimelineObject('inp0', { + deviceType: DeviceType.VMIX, + url, + type: TimelineContentTypeVMix.INPUT, + }), + }), + { + inp0: wrapInMapping({ + mappingType: MappingVmixType.Input, + index: '1', + }), + } + ) + expect(result.reportedState.existingInputs['1'].url).toEqual(url) + }) // TODO: maybe we can't trust the defaults when adding an input? Make this test pass eventually // it('tracks audio state for mapped inputs added by us', () => { diff --git a/packages/timeline-state-resolver/src/integrations/vmix/connection.ts b/packages/timeline-state-resolver/src/integrations/vmix/connection.ts index 67ebfed0d..d3ed5a855 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/connection.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/connection.ts @@ -260,6 +260,8 @@ export class VMixCommandSender { return this.listRemoveAll(command.input) case VMixCommand.RESTART_INPUT: return this.restart(command.input) + case VMixCommand.BROWSER_NAVIGATE: + return this.browserNavigate(command.input, command.value) default: throw new Error(`vmixAPI: Command ${((command || {}) as any).command} not implemented`) } @@ -467,6 +469,10 @@ export class VMixCommandSender { return this.sendCommandFunction(`Restart`, { input }) } + public async browserNavigate(input: string | number, value: string): Promise { + return this.sendCommandFunction(`BrowserNavigate`, { input, value: encodeURIComponent(value) }) + } + private async sendCommandFunction(func: string, args: SentCommandArgs) { return this.vMixConnection.sendCommandFunction(func, args) } diff --git a/packages/timeline-state-resolver/src/integrations/vmix/vMixCommands.ts b/packages/timeline-state-resolver/src/integrations/vmix/vMixCommands.ts index 9c67d1c21..f3d1fe474 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/vMixCommands.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/vMixCommands.ts @@ -203,6 +203,11 @@ export interface VMixStateCommandRestart extends VMixStateCommandBase { command: VMixCommand.RESTART_INPUT input: string | number } +export interface VMixStateCommandBrowserNavigate extends VMixStateCommandBase { + command: VMixCommand.BROWSER_NAVIGATE + input: string | number + value: string +} export type VMixStateCommand = | VMixStateCommandPreviewInput | VMixStateCommandTransition @@ -248,6 +253,7 @@ export type VMixStateCommand = | VMixStateCommandListAdd | VMixStateCommandListRemoveAll | VMixStateCommandRestart + | VMixStateCommandBrowserNavigate export enum CommandContext { None = 'none', diff --git a/packages/timeline-state-resolver/src/integrations/vmix/vMixStateDiffer.ts b/packages/timeline-state-resolver/src/integrations/vmix/vMixStateDiffer.ts index 71948d452..9ba436064 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/vMixStateDiffer.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/vMixStateDiffer.ts @@ -78,6 +78,7 @@ export interface VMixInput { layers?: VMixLayers listFilePaths?: string[] restart?: boolean + url?: string } export interface VMixInputAudio { @@ -583,6 +584,17 @@ export class VMixStateDiffer { timelineId: '', }) } + if (input.url !== undefined && oldInput.url !== input.url) { + commands.push({ + command: { + command: VMixCommand.BROWSER_NAVIGATE, + input: key, + value: input.url, + }, + context: CommandContext.None, + timelineId: '', + }) + } return { preTransitionCommands, postTransitionCommands } } diff --git a/packages/timeline-state-resolver/src/integrations/vmix/vMixTimelineStateConverter.ts b/packages/timeline-state-resolver/src/integrations/vmix/vMixTimelineStateConverter.ts index 45fb7d9d8..c56e387d3 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/vMixTimelineStateConverter.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/vMixTimelineStateConverter.ts @@ -148,6 +148,7 @@ export class VMixTimelineStateConverter { (content.overlays ? this._convertDeprecatedInputOverlays(content.overlays) : undefined), listFilePaths: content.listFilePaths, restart: content.restart, + url: content.url, }, { key: mapping.options.index, filePath: content.filePath }, layerName