webrtc/public/client.js

185 lines
8.1 KiB
JavaScript
Raw Normal View History

2023-09-28 23:49:26 +08:00
import { List, ListItem } from './weigets.js'
export default class ClientList {
2023-10-01 01:22:39 +08:00
constructor({ channels = {}, EventListeners = {}, name: username }) {
2023-09-29 20:20:00 +08:00
this.channels = channels
2023-09-29 17:31:37 +08:00
this.EventListeners = EventListeners
2023-09-28 23:49:26 +08:00
const protocol = window.location.protocol === 'https:' ? 'wss' : 'ws'
const host = window.location.host
this.clientlist = []
this.ul = List({})
document.body.appendChild(this.ul)
2023-09-30 22:45:01 +08:00
2023-09-30 23:01:30 +08:00
// 连接 WebSocket
const linkStart = () => {
2023-10-01 01:22:39 +08:00
const websocket = new WebSocket(`${protocol}://${host}/webrtc/music?name=${username}`)
2023-09-30 23:01:30 +08:00
websocket.onmessage = async event => {
const data = JSON.parse(event.data)
2023-10-01 14:19:38 +08:00
const webrtc_init = async () => {
2023-10-01 12:47:39 +08:00
const webrtc = new RTCPeerConnection({
iceServers: [{
2023-10-01 13:12:18 +08:00
urls: 'turn:satori.love:3478',
2023-10-01 20:36:21 +08:00
username: 'x-username',
2023-10-02 00:30:47 +08:00
credential: 'x-password'
2023-10-01 12:47:39 +08:00
}]
})
2023-09-30 23:01:30 +08:00
webrtc.onicecandidate = event => {
if (event.candidate) {
websocket.send(JSON.stringify({
type: 'candidate',
id: data.id,
candidate: event.candidate
}))
}
}
webrtc.ondatachannel = ({ channel }) => {
2023-10-02 00:42:29 +08:00
console.log('对方建立', channel.label, '数据通道')
2023-10-01 02:42:59 +08:00
const client = this.clientlist.find(x => x.id === data.id)
const option = this.channels[channel.label]
channel.onopen = event => {
2023-10-02 00:42:29 +08:00
console.log('对方打开', channel.label, '数据通道')
2023-10-01 02:42:59 +08:00
if (option && option.onopen) {
option.onopen(event, client)
}
}
2023-09-30 23:01:30 +08:00
channel.onmessage = event => {
2023-10-02 00:53:17 +08:00
//console.log('对方发送', channel.label, '数据消息')
2023-10-01 02:42:59 +08:00
if (option && option.onmessage) {
option.onmessage(event, client)
}
}
channel.onclose = event => {
2023-10-02 00:42:29 +08:00
console.log('对方关闭', channel.label, '数据通道')
2023-10-01 02:42:59 +08:00
if (option && option.onclose) {
option.onclose(event, client)
}
}
channel.onerror = event => {
2023-10-02 00:42:29 +08:00
console.log('对方通道', channel.label, '发生错误')
2023-10-01 02:42:59 +08:00
if (option && option.onerror) {
option.onerror(event, client)
2023-09-30 23:01:30 +08:00
}
2023-09-29 17:31:37 +08:00
}
2023-09-29 01:12:20 +08:00
}
2023-09-30 23:01:30 +08:00
webrtc.oniceconnectionstatechange = event => {
2023-10-01 02:42:59 +08:00
console.log('WebRTC ICE 连接状态更改:', webrtc.iceConnectionState)
2023-09-30 23:01:30 +08:00
}
2023-10-01 02:42:59 +08:00
const channels = Object.entries(this.channels).map(([name, callback]) => {
const channel = webrtc.createDataChannel(name, { reliable: true })
return channel
})
return { webrtc, channels }
2023-09-29 01:12:20 +08:00
}
2023-09-30 23:01:30 +08:00
if (data.type === 'list') {
//console.log('取得在线对端列表:', data)
2023-10-01 14:19:38 +08:00
const { webrtc, channels } = await webrtc_init()
2023-09-30 23:01:30 +08:00
//console.log('发送给对方 offer')
const offer = await webrtc.createOffer()
await webrtc.setLocalDescription(offer)
this.clientlist.push({ id: data.id, name: data.name, webrtc, channels })
websocket.send(JSON.stringify({ type: 'offer', id: data.id, offer }))
this.add(data)
return
2023-09-29 01:12:20 +08:00
}
2023-09-30 23:01:30 +08:00
if (data.type === 'push') {
//console.log('新上线客户端:', data)
return this.add(data)
}
if (data.type === 'pull') {
//console.log('移除客户端:', data)
return this.remove(data)
}
if (data.type === 'offer') {
//console.log('收到对方 offer', data)
2023-10-01 14:19:38 +08:00
const { webrtc, channels } = await webrtc_init()
2023-09-30 23:01:30 +08:00
this.clientlist.push({ id: data.id, name: data.name, webrtc, channels })
//console.log('发送给对方 answer')
await webrtc.setRemoteDescription(data.offer)
const answer = await webrtc.createAnswer()
await webrtc.setLocalDescription(answer)
websocket.send(JSON.stringify({ type: 'answer', id: data.id, answer }))
return
}
if (data.type === 'answer') {
//console.log('收到对方 answer', data)
const pc = this.clientlist.find(client => client.id === data.id).webrtc
await pc.setRemoteDescription(data.answer)
return
}
if (data.type === 'candidate') {
2023-10-02 00:37:02 +08:00
console.log(data.name, '发来 candidate 候选通道')
2023-09-30 23:01:30 +08:00
const pc = this.clientlist.find(client => client.id === data.id).webrtc
await pc.addIceCandidate(data.candidate)
return
}
console.log('收到未知数据:', data)
2023-09-28 23:49:26 +08:00
}
2023-10-01 12:55:42 +08:00
websocket.onclose = async event => {
2023-09-30 23:01:30 +08:00
console.log('WebSocket 断线重连...')
2023-10-01 12:55:42 +08:00
await new Promise(resolve => setTimeout(resolve, 10000))
// this.websocket = linkStart()
// 调试模式: 直接刷新页面重载
window.location.reload()
2023-09-28 23:49:26 +08:00
}
2023-09-30 23:01:30 +08:00
return websocket
2023-09-28 23:49:26 +08:00
}
2023-09-30 23:01:30 +08:00
this.websocket = linkStart()
2023-09-28 23:49:26 +08:00
}
2023-09-29 21:21:26 +08:00
setChannel(name, option) {
this.channels[name] = option
}
2023-09-28 23:49:26 +08:00
add(item) {
this.ul.appendChild(ListItem({
id: item.id,
innerText: item.name ?? item.id,
onclick: event => {
},
chidren: []
}))
}
remove(item) {
this.clientlist = this.clientlist.filter(client => client.id !== item.id)
this.ul.removeChild(document.getElementById(item.id))
}
2023-09-29 01:12:20 +08:00
// 添加回调函数
on(name, callback) {
this.EventListeners[name] = callback
}
// 执行回调函数
_on(name, ...args) {
if (this.EventListeners[name]) {
this.EventListeners[name](...args)
}
}
2023-10-02 00:30:47 +08:00
// 通过指定通道发送数据(单播)
sendto(id, name, data) {
//console.log('发送数据:', data, '到通道:', name, '到客户端:', id)
const client = this.clientlist.find(client => client.id === id)
if (!client) {
console.log('客户端不存在:', id)
return
}
client.channels.filter(ch => ch.label === name).forEach(async ch => {
// 等待 datachannel 打开(临时解决方案)
while (ch.readyState !== 'open') {
await new Promise(resolve => setTimeout(resolve, 100))
}
ch.send(data)
})
}
2023-09-29 01:12:20 +08:00
// 通过指定通道发送数据(广播)
send(name, data) {
2023-09-30 16:19:33 +08:00
//console.log('广播数据:', data, '到通道:', name, '到所有客户端')
2023-09-29 01:12:20 +08:00
this.clientlist.forEach(client => {
2023-10-01 01:22:39 +08:00
//console.log('发送数据到客户端:', client.id, client.name, '通道:', name, '数据:', data)
client.channels.filter(ch => ch.label === name).forEach(async ch => {
// 等待 datachannel 打开(临时解决方案)
while (ch.readyState !== 'open') {
await new Promise(resolve => setTimeout(resolve, 100))
}
2023-09-30 16:19:33 +08:00
ch.send(data)
})
2023-09-29 01:12:20 +08:00
})
}
2023-09-28 23:49:26 +08:00
}