diff --git a/examples/spotlight_sendrecv/index.html b/examples/spotlight_sendrecv/index.html index d8e8a82a..8378c140 100644 --- a/examples/spotlight_sendrecv/index.html +++ b/examples/spotlight_sendrecv/index.html @@ -11,8 +11,8 @@

Spotlight Sendrecv test

sendrecv1

- -
+ +
@@ -20,8 +20,8 @@

sendrecv1

sendrecv2

- -
+ +
@@ -30,7 +30,7 @@

sendrecv2

- + \ No newline at end of file diff --git a/examples/spotlight_sendrecv/main.mjs b/examples/spotlight_sendrecv/main.mjs deleted file mode 100644 index 7989e281..00000000 --- a/examples/spotlight_sendrecv/main.mjs +++ /dev/null @@ -1,107 +0,0 @@ -import Sora from '../../dist/sora.mjs' - -const SORA_SIGNALING_URL = import.meta.env.VITE_SORA_SIGNALING_URL -const SORA_CHANNEL_ID_PREFIX = import.meta.env.VITE_SORA_CHANNEL_ID_PREFIX || '' -const SORA_CHANNEL_ID_SUFFIX = import.meta.env.VITE_SORA_CHANNEL_ID_SUFFIX || '' -const ACCESS_TOKEN = import.meta.env.VITE_ACCESS_TOKEN || '' - -const channelId = `${SORA_CHANNEL_ID_PREFIX}spotlight${SORA_CHANNEL_ID_SUFFIX}` -const debug = false -const sora = Sora.connection(SORA_SIGNALING_URL, debug) -const metadata = { access_token: ACCESS_TOKEN } -const options = { - audio: true, - video: true, - simulcast: true, - spotlight: true, - spotlightNumber: 1, -} - -const sendrecv1 = sora.sendrecv(channelId, metadata, options) - -sendrecv1.on('notify', (event) => { - if (event.event_type === 'connection.created' && sendrecv1.connectionId === event.connection_id) { - const connectionIdElement = document.querySelector('#sendrecv1-connection-id') - connectionIdElement.textContent = event.connection_id - } -}) - -sendrecv1.on('track', (event) => { - const stream = event.streams[0] - if (!stream) return - const remoteVideoId = `sendrecv1-remotevideo-${stream.id}` - const remoteVideos = document.querySelector('#sendrecv1-remote-videos') - if (!remoteVideos.querySelector(`#${remoteVideoId}`)) { - const remoteVideo = document.createElement('video') - remoteVideo.id = remoteVideoId - remoteVideo.style.border = '1px solid red' - remoteVideo.autoplay = true - remoteVideo.playsinline = true - remoteVideo.controls = true - remoteVideo.srcObject = stream - remoteVideos.appendChild(remoteVideo) - } -}) - -sendrecv1.on('removetrack', (event) => { - const remoteVideo = document.querySelector(`#sendrecv1-remotevideo-${event.target.id}`) - if (remoteVideo) { - document.querySelector('#sendrecv1-remote-videos').removeChild(remoteVideo) - } -}) - -const sendrecv2 = sora.sendrecv(channelId, metadata, options) -sendrecv2.on('notify', (event) => { - if (event.event_type === 'connection.created' && sendrecv2.connectionId === event.connection_id) { - const connectionIdElement = document.querySelector('#sendrecv2-connection-id') - connectionIdElement.textContent = event.connection_id - } -}) - -sendrecv2.on('track', (event) => { - const stream = event.streams[0] - if (!stream) return - const remoteVideoId = `sendrecv2-remotevideo-${stream.id}` - const remoteVideos = document.querySelector('#sendrecv2-remote-videos') - if (!remoteVideos.querySelector(`#${remoteVideoId}`)) { - const remoteVideo = document.createElement('video') - remoteVideo.id = remoteVideoId - remoteVideo.style.border = '1px solid red' - remoteVideo.autoplay = true - remoteVideo.playsinline = true - remoteVideo.controls = true - remoteVideo.srcObject = stream - remoteVideos.appendChild(remoteVideo) - } -}) -sendrecv2.on('removetrack', (event) => { - const remoteVideo = document.querySelector(`#sendrecv2-remotevideo-${event.target.id}`) - if (remoteVideo) { - remoteVideo.srcObject = null - document.querySelector('#sendrecv2-remote-videos').removeChild(remoteVideo) - } -}) - -document.querySelector('#start-sendrecv1').addEventListener('click', async () => { - // sendrecv1 - const mediaStream = await navigator.mediaDevices.getUserMedia({ audio: true, video: true }) - - document.querySelector('#sendrecv1-local-video').srcObject = mediaStream - await sendrecv1.connect(mediaStream) -}) -document.querySelector('#start-sendrecv2').addEventListener('click', async () => { - // sendrecv2 - const mediaStream = await navigator.mediaDevices.getUserMedia({ audio: true, video: true }) - document.querySelector('#sendrecv2-local-video').srcObject = mediaStream - await sendrecv2.connect(mediaStream) -}) -document.querySelector('#stop-sendrecv1').addEventListener('click', async () => { - await sendrecv1.disconnect() - document.querySelector('#sendrecv1-local-video').srcObject = null - document.querySelector('#sendrecv1-remote-videos').innerHTML = null -}) -document.querySelector('#stop-sendrecv2').addEventListener('click', async () => { - sendrecv2.disconnect() - document.querySelector('#sendrecv2-local-video').srcObject = null - document.querySelector('#sendrecv2-remote-videos').innerHTML = null -}) diff --git a/examples/spotlight_sendrecv/main.mts b/examples/spotlight_sendrecv/main.mts new file mode 100644 index 00000000..ba149236 --- /dev/null +++ b/examples/spotlight_sendrecv/main.mts @@ -0,0 +1,148 @@ +import Sora, { + type SoraConnection, + type ConnectionPublisher, + type SignalingNotifyMessage, +} from '../../dist/sora' + +document.addEventListener('DOMContentLoaded', async () => { + const SORA_SIGNALING_URL = import.meta.env.VITE_SORA_SIGNALING_URL + const SORA_CHANNEL_ID_PREFIX = import.meta.env.VITE_SORA_CHANNEL_ID_PREFIX || '' + const SORA_CHANNEL_ID_SUFFIX = import.meta.env.VITE_SORA_CHANNEL_ID_SUFFIX || '' + const ACCESS_TOKEN = import.meta.env.VITE_ACCESS_TOKEN || '' + + const sendrecv1 = new SoraClient( + 'sendrecv1', + SORA_SIGNALING_URL, + SORA_CHANNEL_ID_PREFIX, + SORA_CHANNEL_ID_SUFFIX, + ACCESS_TOKEN, + ) + + const sendrecv2 = new SoraClient( + 'sendrecv2', + SORA_SIGNALING_URL, + SORA_CHANNEL_ID_PREFIX, + SORA_CHANNEL_ID_SUFFIX, + ACCESS_TOKEN, + ) + + document.querySelector('#sendrecv1-start')?.addEventListener('click', async () => { + // sendrecv1 + const stream = await navigator.mediaDevices.getUserMedia({ audio: true, video: true }) + await sendrecv1.connect(stream) + }) + document.querySelector('#sendrecv1-stop')?.addEventListener('click', async () => { + await sendrecv1.disconnect() + }) + + document.querySelector('#sendrecv2-start')?.addEventListener('click', async () => { + // sendrecv2 + const stream = await navigator.mediaDevices.getUserMedia({ audio: true, video: true }) + await sendrecv2.connect(stream) + }) + document.querySelector('#sendrecv2-stop')?.addEventListener('click', async () => { + await sendrecv2.disconnect() + }) +}) + +class SoraClient { + // sendrecv1 or sendrecv2 + private label: string + + private debug = false + + private channelId: string + private metadata: { access_token: string } + private options: object + + private sora: SoraConnection + private connection: ConnectionPublisher + + constructor( + label: string, + signalingUrl: string, + channelIdPrefix: string, + channelIdSuffix: string, + accessToken: string, + ) { + this.label = label + + this.sora = Sora.connection(signalingUrl, this.debug) + this.channelId = `${channelIdPrefix}sendrecv${channelIdSuffix}` + this.metadata = { access_token: accessToken } + this.options = { + audio: true, + video: true, + simulcast: true, + spotlight: true, + spotlightNumber: 1, + } + + this.connection = this.sora.sendrecv(this.channelId, this.metadata, this.options) + + this.connection.on('notify', this.onnotify.bind(this)) + this.connection.on('track', this.ontrack.bind(this)) + this.connection.on('removetrack', this.onremovetrack.bind(this)) + } + + async connect(stream: MediaStream) { + await this.connection.connect(stream) + const localVideo = document.querySelector(`#${this.label}-local-video`) + if (localVideo) { + localVideo.srcObject = stream + } + } + + async disconnect() { + await this.connection.disconnect() + + // お掃除 + const localVideo = document.querySelector(`#${this.label}-local-video`) + if (localVideo) { + localVideo.srcObject = null + } + // お掃除 + const remoteVideos = document.querySelector(`#${this.label}-remote-videos`) + if (remoteVideos) { + remoteVideos.innerHTML = '' + } + } + + private onnotify(event: SignalingNotifyMessage): void { + if ( + event.event_type === 'connection.created' && + this.connection.connectionId === event.connection_id + ) { + const connectionIdElement = document.querySelector(`#${this.label}-connection-id`) + if (connectionIdElement) { + connectionIdElement.textContent = event.connection_id + } + } + } + + private ontrack(event: RTCTrackEvent): void { + const stream = event.streams[0] + const remoteVideoId = `${this.label}-remote-video-${stream.id}` + const remoteVideos = document.querySelector(`#${this.label}-remote-videos`) + if (remoteVideos && !remoteVideos.querySelector(`#${remoteVideoId}`)) { + const remoteVideo = document.createElement('video') + remoteVideo.id = remoteVideoId + remoteVideo.style.border = '1px solid red' + remoteVideo.autoplay = true + remoteVideo.playsInline = true + remoteVideo.controls = true + remoteVideo.width = 160 + remoteVideo.height = 120 + remoteVideo.srcObject = stream + remoteVideos.appendChild(remoteVideo) + } + } + + private onremovetrack(event: MediaStreamTrackEvent): void { + const target = event.target as MediaStream + const remoteVideo = document.querySelector(`#${this.label}-remote-video-${target.id}`) + if (remoteVideo) { + document.querySelector(`#${this.label}-remote-videos`)?.removeChild(remoteVideo) + } + } +} diff --git a/tests/spotlight_sendrecv.spec.ts b/tests/spotlight_sendrecv.spec.ts index e002a38f..ad7e5470 100644 --- a/tests/spotlight_sendrecv.spec.ts +++ b/tests/spotlight_sendrecv.spec.ts @@ -3,8 +3,8 @@ import { test } from '@playwright/test' test('spotlight sendrecv x2', async ({ page }) => { await page.goto('http://localhost:9000/spotlight_sendrecv/') - await page.click('#start-sendrecv1') - await page.click('#start-sendrecv2') + await page.click('#sendrecv1-start') + await page.click('#sendrecv2-start') // #sendrecv1-connection-id 要素が存在し、その内容が空でないことを確認するまで待つ await page.waitForSelector('#sendrecv1-connection-id:not(:empty)') @@ -20,6 +20,6 @@ test('spotlight sendrecv x2', async ({ page }) => { const sendrecv2ConnectionId = await page.$eval('#sendrecv2-connection-id', (el) => el.textContent) console.log(`sendrecv2 connectionId=${sendrecv2ConnectionId}`) - await page.click('#stop-sendrecv1') - await page.click('#stop-sendrecv2') + await page.click('#sendrecv1-stop') + await page.click('#sendrecv2-stop') })