dbff58ebdd
Add get_gathering_state() returning the iceGatheringState of the connection. Add get_signaling_state() returning the signalingState of the connection. Improve JS library.
482 lines
15 KiB
JavaScript
482 lines
15 KiB
JavaScript
/*************************************************************************/
|
|
/* library_godot_webrtc.js */
|
|
/*************************************************************************/
|
|
/* This file is part of: */
|
|
/* GODOT ENGINE */
|
|
/* https://godotengine.org */
|
|
/*************************************************************************/
|
|
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
|
|
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
|
|
/* */
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
/* a copy of this software and associated documentation files (the */
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
/* the following conditions: */
|
|
/* */
|
|
/* The above copyright notice and this permission notice shall be */
|
|
/* included in all copies or substantial portions of the Software. */
|
|
/* */
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
/*************************************************************************/
|
|
|
|
const GodotRTCDataChannel = {
|
|
// Our socket implementation that forwards events to C++.
|
|
$GodotRTCDataChannel__deps: ['$IDHandler', '$GodotRuntime'],
|
|
$GodotRTCDataChannel: {
|
|
connect: function (p_id, p_on_open, p_on_message, p_on_error, p_on_close) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
|
|
ref.binaryType = 'arraybuffer';
|
|
ref.onopen = function (event) {
|
|
p_on_open();
|
|
};
|
|
ref.onclose = function (event) {
|
|
p_on_close();
|
|
};
|
|
ref.onerror = function (event) {
|
|
p_on_error();
|
|
};
|
|
ref.onmessage = function (event) {
|
|
let buffer;
|
|
let is_string = 0;
|
|
if (event.data instanceof ArrayBuffer) {
|
|
buffer = new Uint8Array(event.data);
|
|
} else if (event.data instanceof Blob) {
|
|
GodotRuntime.error('Blob type not supported');
|
|
return;
|
|
} else if (typeof event.data === 'string') {
|
|
is_string = 1;
|
|
const enc = new TextEncoder('utf-8');
|
|
buffer = new Uint8Array(enc.encode(event.data));
|
|
} else {
|
|
GodotRuntime.error('Unknown message type');
|
|
return;
|
|
}
|
|
const len = buffer.length * buffer.BYTES_PER_ELEMENT;
|
|
const out = GodotRuntime.malloc(len);
|
|
HEAPU8.set(buffer, out);
|
|
p_on_message(out, len, is_string);
|
|
GodotRuntime.free(out);
|
|
};
|
|
},
|
|
|
|
close: function (p_id) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
ref.onopen = null;
|
|
ref.onmessage = null;
|
|
ref.onerror = null;
|
|
ref.onclose = null;
|
|
ref.close();
|
|
},
|
|
|
|
get_prop: function (p_id, p_prop, p_def) {
|
|
const ref = IDHandler.get(p_id);
|
|
return (ref && ref[p_prop] !== undefined) ? ref[p_prop] : p_def;
|
|
},
|
|
},
|
|
|
|
godot_js_rtc_datachannel_ready_state_get__sig: 'ii',
|
|
godot_js_rtc_datachannel_ready_state_get: function (p_id) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return 3; // CLOSED
|
|
}
|
|
|
|
switch (ref.readyState) {
|
|
case 'connecting':
|
|
return 0;
|
|
case 'open':
|
|
return 1;
|
|
case 'closing':
|
|
return 2;
|
|
case 'closed':
|
|
default:
|
|
return 3;
|
|
}
|
|
},
|
|
|
|
godot_js_rtc_datachannel_send__sig: 'iiiii',
|
|
godot_js_rtc_datachannel_send: function (p_id, p_buffer, p_length, p_raw) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return 1;
|
|
}
|
|
|
|
const bytes_array = new Uint8Array(p_length);
|
|
for (let i = 0; i < p_length; i++) {
|
|
bytes_array[i] = GodotRuntime.getHeapValue(p_buffer + i, 'i8');
|
|
}
|
|
|
|
if (p_raw) {
|
|
ref.send(bytes_array.buffer);
|
|
} else {
|
|
const string = new TextDecoder('utf-8').decode(bytes_array);
|
|
ref.send(string);
|
|
}
|
|
return 0;
|
|
},
|
|
|
|
godot_js_rtc_datachannel_is_ordered__sig: 'ii',
|
|
godot_js_rtc_datachannel_is_ordered: function (p_id) {
|
|
return GodotRTCDataChannel.get_prop(p_id, 'ordered', true);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_id_get__sig: 'ii',
|
|
godot_js_rtc_datachannel_id_get: function (p_id) {
|
|
return GodotRTCDataChannel.get_prop(p_id, 'id', 65535);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_max_packet_lifetime_get__sig: 'ii',
|
|
godot_js_rtc_datachannel_max_packet_lifetime_get: function (p_id) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return 65535;
|
|
}
|
|
if (ref['maxPacketLifeTime'] !== undefined) {
|
|
return ref['maxPacketLifeTime'];
|
|
} else if (ref['maxRetransmitTime'] !== undefined) {
|
|
// Guess someone didn't appreciate the standardization process.
|
|
return ref['maxRetransmitTime'];
|
|
}
|
|
return 65535;
|
|
},
|
|
|
|
godot_js_rtc_datachannel_max_retransmits_get__sig: 'ii',
|
|
godot_js_rtc_datachannel_max_retransmits_get: function (p_id) {
|
|
return GodotRTCDataChannel.get_prop(p_id, 'maxRetransmits', 65535);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_is_negotiated__sig: 'ii',
|
|
godot_js_rtc_datachannel_is_negotiated: function (p_id) {
|
|
return GodotRTCDataChannel.get_prop(p_id, 'negotiated', 65535);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_get_buffered_amount__sig: 'ii',
|
|
godot_js_rtc_datachannel_get_buffered_amount: function (p_id) {
|
|
return GodotRTCDataChannel.get_prop(p_id, 'bufferedAmount', 0);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_label_get__sig: 'ii',
|
|
godot_js_rtc_datachannel_label_get: function (p_id) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref || !ref.label) {
|
|
return 0;
|
|
}
|
|
return GodotRuntime.allocString(ref.label);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_protocol_get__sig: 'ii',
|
|
godot_js_rtc_datachannel_protocol_get: function (p_id) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref || !ref.protocol) {
|
|
return 0;
|
|
}
|
|
return GodotRuntime.allocString(ref.protocol);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_destroy__sig: 'vi',
|
|
godot_js_rtc_datachannel_destroy: function (p_id) {
|
|
GodotRTCDataChannel.close(p_id);
|
|
IDHandler.remove(p_id);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_connect__sig: 'viiiiii',
|
|
godot_js_rtc_datachannel_connect: function (p_id, p_ref, p_on_open, p_on_message, p_on_error, p_on_close) {
|
|
const onopen = GodotRuntime.get_func(p_on_open).bind(null, p_ref);
|
|
const onmessage = GodotRuntime.get_func(p_on_message).bind(null, p_ref);
|
|
const onerror = GodotRuntime.get_func(p_on_error).bind(null, p_ref);
|
|
const onclose = GodotRuntime.get_func(p_on_close).bind(null, p_ref);
|
|
GodotRTCDataChannel.connect(p_id, onopen, onmessage, onerror, onclose);
|
|
},
|
|
|
|
godot_js_rtc_datachannel_close__sig: 'vi',
|
|
godot_js_rtc_datachannel_close: function (p_id) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
GodotRTCDataChannel.close(p_id);
|
|
},
|
|
};
|
|
|
|
autoAddDeps(GodotRTCDataChannel, '$GodotRTCDataChannel');
|
|
mergeInto(LibraryManager.library, GodotRTCDataChannel);
|
|
|
|
const GodotRTCPeerConnection = {
|
|
$GodotRTCPeerConnection__deps: ['$IDHandler', '$GodotRuntime', '$GodotRTCDataChannel'],
|
|
$GodotRTCPeerConnection: {
|
|
// Enums
|
|
ConnectionState: {
|
|
'new': 0,
|
|
'connecting': 1,
|
|
'connected': 2,
|
|
'disconnected': 3,
|
|
'failed': 4,
|
|
'closed': 5,
|
|
},
|
|
|
|
ConnectionStateCompat: {
|
|
// Using values from IceConnectionState for browsers that do not support ConnectionState (notably Firefox).
|
|
'new': 0,
|
|
'checking': 1,
|
|
'connected': 2,
|
|
'completed': 2,
|
|
'disconnected': 3,
|
|
'failed': 4,
|
|
'closed': 5,
|
|
},
|
|
|
|
IceGatheringState: {
|
|
'new': 0,
|
|
'gathering': 1,
|
|
'complete': 2,
|
|
},
|
|
|
|
SignalingState: {
|
|
'stable': 0,
|
|
'have-local-offer': 1,
|
|
'have-remote-offer': 2,
|
|
'have-local-pranswer': 3,
|
|
'have-remote-pranswer': 4,
|
|
'closed': 5,
|
|
},
|
|
|
|
// Callbacks
|
|
create: function (config, onConnectionChange, onSignalingChange, onIceGatheringChange, onIceCandidate, onDataChannel) {
|
|
let conn = null;
|
|
try {
|
|
conn = new RTCPeerConnection(config);
|
|
} catch (e) {
|
|
GodotRuntime.error(e);
|
|
return 0;
|
|
}
|
|
|
|
const id = IDHandler.add(conn);
|
|
|
|
if ('connectionState' in conn && conn['connectionState'] !== undefined) {
|
|
// Use "connectionState" if supported
|
|
conn.onconnectionstatechange = function (event) {
|
|
if (!IDHandler.get(id)) {
|
|
return;
|
|
}
|
|
onConnectionChange(GodotRTCPeerConnection.ConnectionState[conn.connectionState] || 0);
|
|
};
|
|
} else {
|
|
// Fall back to using "iceConnectionState" when "connectionState" is not supported (notably Firefox).
|
|
conn.oniceconnectionstatechange = function (event) {
|
|
if (!IDHandler.get(id)) {
|
|
return;
|
|
}
|
|
onConnectionChange(GodotRTCPeerConnection.ConnectionStateCompat[conn.iceConnectionState] || 0);
|
|
};
|
|
}
|
|
conn.onicegatheringstatechange = function (event) {
|
|
if (!IDHandler.get(id)) {
|
|
return;
|
|
}
|
|
onIceGatheringChange(GodotRTCPeerConnection.IceGatheringState[conn.iceGatheringState] || 0);
|
|
};
|
|
conn.onsignalingstatechange = function (event) {
|
|
if (!IDHandler.get(id)) {
|
|
return;
|
|
}
|
|
onSignalingChange(GodotRTCPeerConnection.SignalingState[conn.signalingState] || 0);
|
|
};
|
|
conn.onicecandidate = function (event) {
|
|
if (!IDHandler.get(id)) {
|
|
return;
|
|
}
|
|
const c = event.candidate;
|
|
if (!c || !c.candidate) {
|
|
return;
|
|
}
|
|
const candidate_str = GodotRuntime.allocString(c.candidate);
|
|
const mid_str = GodotRuntime.allocString(c.sdpMid);
|
|
onIceCandidate(mid_str, c.sdpMLineIndex, candidate_str);
|
|
GodotRuntime.free(candidate_str);
|
|
GodotRuntime.free(mid_str);
|
|
};
|
|
conn.ondatachannel = function (event) {
|
|
if (!IDHandler.get(id)) {
|
|
return;
|
|
}
|
|
const cid = IDHandler.add(event.channel);
|
|
onDataChannel(cid);
|
|
};
|
|
return id;
|
|
},
|
|
|
|
destroy: function (p_id) {
|
|
const conn = IDHandler.get(p_id);
|
|
if (!conn) {
|
|
return;
|
|
}
|
|
conn.onconnectionstatechange = null;
|
|
conn.oniceconnectionstatechange = null;
|
|
conn.onicegatheringstatechange = null;
|
|
conn.onsignalingstatechange = null;
|
|
conn.onicecandidate = null;
|
|
conn.ondatachannel = null;
|
|
IDHandler.remove(p_id);
|
|
},
|
|
|
|
onsession: function (p_id, callback, session) {
|
|
if (!IDHandler.get(p_id)) {
|
|
return;
|
|
}
|
|
const type_str = GodotRuntime.allocString(session.type);
|
|
const sdp_str = GodotRuntime.allocString(session.sdp);
|
|
callback(type_str, sdp_str);
|
|
GodotRuntime.free(type_str);
|
|
GodotRuntime.free(sdp_str);
|
|
},
|
|
|
|
onerror: function (p_id, callback, error) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
GodotRuntime.error(error);
|
|
callback();
|
|
},
|
|
},
|
|
|
|
godot_js_rtc_pc_create__sig: 'iiiiiiii',
|
|
godot_js_rtc_pc_create: function (p_config, p_ref, p_on_connection_state_change, p_on_ice_gathering_state_change, p_on_signaling_state_change, p_on_ice_candidate, p_on_datachannel) {
|
|
const wrap = function (p_func) {
|
|
return GodotRuntime.get_func(p_func).bind(null, p_ref);
|
|
};
|
|
return GodotRTCPeerConnection.create(
|
|
JSON.parse(GodotRuntime.parseString(p_config)),
|
|
wrap(p_on_connection_state_change),
|
|
wrap(p_on_signaling_state_change),
|
|
wrap(p_on_ice_gathering_state_change),
|
|
wrap(p_on_ice_candidate),
|
|
wrap(p_on_datachannel)
|
|
);
|
|
},
|
|
|
|
godot_js_rtc_pc_close__sig: 'vi',
|
|
godot_js_rtc_pc_close: function (p_id) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
ref.close();
|
|
},
|
|
|
|
godot_js_rtc_pc_destroy__sig: 'vi',
|
|
godot_js_rtc_pc_destroy: function (p_id) {
|
|
GodotRTCPeerConnection.destroy(p_id);
|
|
},
|
|
|
|
godot_js_rtc_pc_offer_create__sig: 'viiii',
|
|
godot_js_rtc_pc_offer_create: function (p_id, p_obj, p_on_session, p_on_error) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
const onsession = GodotRuntime.get_func(p_on_session).bind(null, p_obj);
|
|
const onerror = GodotRuntime.get_func(p_on_error).bind(null, p_obj);
|
|
ref.createOffer().then(function (session) {
|
|
GodotRTCPeerConnection.onsession(p_id, onsession, session);
|
|
}).catch(function (error) {
|
|
GodotRTCPeerConnection.onerror(p_id, onerror, error);
|
|
});
|
|
},
|
|
|
|
godot_js_rtc_pc_local_description_set__sig: 'viiiii',
|
|
godot_js_rtc_pc_local_description_set: function (p_id, p_type, p_sdp, p_obj, p_on_error) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
const type = GodotRuntime.parseString(p_type);
|
|
const sdp = GodotRuntime.parseString(p_sdp);
|
|
const onerror = GodotRuntime.get_func(p_on_error).bind(null, p_obj);
|
|
ref.setLocalDescription({
|
|
'sdp': sdp,
|
|
'type': type,
|
|
}).catch(function (error) {
|
|
GodotRTCPeerConnection.onerror(p_id, onerror, error);
|
|
});
|
|
},
|
|
|
|
godot_js_rtc_pc_remote_description_set__sig: 'viiiiii',
|
|
godot_js_rtc_pc_remote_description_set: function (p_id, p_type, p_sdp, p_obj, p_session_created, p_on_error) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
const type = GodotRuntime.parseString(p_type);
|
|
const sdp = GodotRuntime.parseString(p_sdp);
|
|
const onerror = GodotRuntime.get_func(p_on_error).bind(null, p_obj);
|
|
const onsession = GodotRuntime.get_func(p_session_created).bind(null, p_obj);
|
|
ref.setRemoteDescription({
|
|
'sdp': sdp,
|
|
'type': type,
|
|
}).then(function () {
|
|
if (type !== 'offer') {
|
|
return Promise.resolve();
|
|
}
|
|
return ref.createAnswer().then(function (session) {
|
|
GodotRTCPeerConnection.onsession(p_id, onsession, session);
|
|
});
|
|
}).catch(function (error) {
|
|
GodotRTCPeerConnection.onerror(p_id, onerror, error);
|
|
});
|
|
},
|
|
|
|
godot_js_rtc_pc_ice_candidate_add__sig: 'viiii',
|
|
godot_js_rtc_pc_ice_candidate_add: function (p_id, p_mid_name, p_mline_idx, p_sdp) {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return;
|
|
}
|
|
const sdpMidName = GodotRuntime.parseString(p_mid_name);
|
|
const sdpName = GodotRuntime.parseString(p_sdp);
|
|
ref.addIceCandidate(new RTCIceCandidate({
|
|
'candidate': sdpName,
|
|
'sdpMid': sdpMidName,
|
|
'sdpMlineIndex': p_mline_idx,
|
|
}));
|
|
},
|
|
|
|
godot_js_rtc_pc_datachannel_create__deps: ['$GodotRTCDataChannel'],
|
|
godot_js_rtc_pc_datachannel_create__sig: 'iiii',
|
|
godot_js_rtc_pc_datachannel_create: function (p_id, p_label, p_config) {
|
|
try {
|
|
const ref = IDHandler.get(p_id);
|
|
if (!ref) {
|
|
return 0;
|
|
}
|
|
|
|
const label = GodotRuntime.parseString(p_label);
|
|
const config = JSON.parse(GodotRuntime.parseString(p_config));
|
|
|
|
const channel = ref.createDataChannel(label, config);
|
|
return IDHandler.add(channel);
|
|
} catch (e) {
|
|
GodotRuntime.error(e);
|
|
return 0;
|
|
}
|
|
},
|
|
};
|
|
|
|
autoAddDeps(GodotRTCPeerConnection, '$GodotRTCPeerConnection');
|
|
mergeInto(LibraryManager.library, GodotRTCPeerConnection);
|