diff --git a/index.html b/index.html
index 72b3c34..da3f7e8 100644
--- a/index.html
+++ b/index.html
@@ -487,6 +487,7 @@
Loading...
+
@@ -523,6 +524,7 @@ Loading...
};
const DEFAULT_URL = 'https://google.com';
const PARCEL_SIZE = 8;
+const BROWSER_SIZE = 1920;
const TOKEN_PRICE = 5e15;
const DEFAULT_SKIN_URL = 'skin.png';
const RAY_COLOR = 0x44c2ff;
@@ -4370,7 +4372,9 @@
};
mesh.getIntersectionCandidates = () => mesh.visible ?
- [mesh, keyboardMesh].concat(appIconMeshes.children)
+ [mesh, keyboardMesh]
+ .concat(appIconMeshes.children)
+ .concat(browserMesh || [])
: [];
return mesh;
};
@@ -5782,6 +5786,9 @@
intersectionSpec.onclose(i);
} else if (type === 'screen') {
screenMesh.click(intersectionSpec);
+ } else if (type === 'browser') {
+ const {mesh: browserMesh} = intersectionSpec;
+ browserMesh.click();
} else if (type === 'map') {
window.document.xrOffset.position[0] += intersectionSpec.x*7 - camera.position.x;
window.document.xrOffset.position[2] += intersectionSpec.y*7 - camera.position.z;
@@ -6603,14 +6610,13 @@
{'urls': 'stun:stun.l.google.com:19302'},
],
};
-
const peerConnection = new RTCPeerConnection(peerConnectionConfig);
peerConnection.ontrack = e => {
console.log('got track', e);
};
let skinMesh = null;
- const sendChannel = peerConnection.createDataChannel('sendChannel');
+ const sendChannel = peerConnection.createDataChannel('pose');
let pingInterval = 0;
let updateInterval = 0;
sendChannel.onopen = () => {
@@ -6736,6 +6742,9 @@
const connectionId = Math.random().toString(36).replace(/[^a-z]+/g, '').substr(0, 5);
const peerConnections = [];
+let browserPeerConnection = null;
+let browserDataChannel = null;
+let browserMesh = null;
const _rtcConnect = () => {
const s = new WebSocket(LAMBDA_URLS.presence);
s.onopen = () => {
@@ -6745,9 +6754,111 @@
method: 'init',
connectionId,
}));
+
+ const peerConnectionConfig = {
+ iceServers: [
+ {'urls': 'stun:stun.stunprotocol.org:3478'},
+ {'urls': 'stun:stun.l.google.com:19302'},
+ ],
+ };
+ browserPeerConnection = new RTCPeerConnection(peerConnectionConfig);
+ const dataChannel = browserPeerConnection.createDataChannel('browser', {id: 0, negotiated: true});
+ dataChannel.onopen = () => {
+ console.log('got browser data channel', dataChannel);
+ };
+ browserDataChannel = dataChannel;
+ browserPeerConnection.ontrack = e => {
+ console.log('got browser track', e, e.streams.length);
+
+ const video = document.createElement('video');
+ video.style.width = '100%';
+ video.srcObject = e.streams[0];
+ video.play();
+
+ const s = 2;
+ const r = s*4/(2*Math.PI);
+ const geometry = new THREE.CylinderBufferGeometry(r, r, s, 10, 10, true, Math.PI*3/2+Math.PI/4, Math.PI/2)
+ .applyMatrix(new THREE.Matrix4().makeScale(1, 1, -1))
+ .applyMatrix(new THREE.Matrix4().makeTranslation(0, 0, r));
+
+ const texture = new THREE.VideoTexture(video);
+ texture.minFilter = THREE.LinearFilter;
+ texture.magFilter = THREE.LinearFilter;
+ texture.format = THREE.RGBFormat;
+
+ const material = new THREE.MeshBasicMaterial({
+ map: texture,
+ side: THREE.DoubleSide,
+ });
+
+ const mesh = new THREE.Mesh(geometry, material);
+ mesh.frustumCulled = false;
+ const lastCoords = [NaN, NaN];
+ mesh.intersect = ray => {
+ localRaycaster.ray.copy(ray);
+ const intersection = localRaycaster.intersectObject(mesh)[0];
+ if (intersection) {
+ const {uv, distance} = intersection;
+ uv.y = 1-uv.y;
+ uv.x *= BROWSER_SIZE;
+ uv.y *= BROWSER_SIZE;
+ const {x, y} = uv;
+ if (x !== lastCoords[0] || y !== lastCoords[1]) {
+ mesh.mousemove(x, y);
+
+ lastCoords[0] = x;
+ lastCoords[1] = y;
+ }
+ return {
+ type: 'browser',
+ x,
+ y,
+ distance,
+ mesh,
+ cancel: true,
+ };
+ } else {
+ return null;
+ }
+ };
+ mesh.mousemove = (x, y) => {
+ console.log('mouse move', x, y, dataChannel && dataChannel.readyState);
+ if (dataChannel && dataChannel.readyState === 'open') {
+ dataChannel.send(JSON.stringify({
+ method: 'mousemove',
+ x,
+ y: y + 160*(1 - y/BROWSER_SIZE),
+ }));
+ }
+ };
+ mesh.click = () => {
+ if (dataChannel && dataChannel.readyState === 'open') {
+ dataChannel.send(JSON.stringify({
+ method: 'click',
+ }));
+ }
+ };
+ mesh.position.y = 2;
+ scene.add(mesh);
+ browserMesh = mesh;
+
+ e.track.onended = () => {
+ console.log('track ended!!!!!!!!!!!!!!!!!!!!!!!!!', e.track);
+ };
+
+ /* document.body.querySelector('header').style.display = 'none';
+ document.body.querySelector('#canvas').style.display = 'none';
+ document.body.appendChild(video); */
+ };
+ s.send(JSON.stringify({
+ method: 'requestBrowser',
+ url: 'https://google.com/',
+ connectionId,
+ }));
+ peerConnections.push(browserPeerConnection);
};
s.onmessage = e => {
- // console.log('got message', e.data);
+ console.log('got message', e.data);
const data = JSON.parse(e.data);
const {method} = data;
@@ -6837,6 +6948,29 @@
} else {
console.warn('no such peer connection', peerConnectionId, peerConnections.map(peerConnection => peerConnection.connectionId));
}
+ } else if (method === 'respondBrowser') {
+ const {src: peerConnectionId, offer} = data;
+ browserPeerConnection.connectionId = peerConnectionId;
+ browserPeerConnection.onicecandidate = e => {
+ s.send(JSON.stringify({
+ dst: peerConnectionId,
+ src: connectionId,
+ method: 'iceCandidate',
+ candidate: e.candidate,
+ }));
+ };
+ browserPeerConnection.setRemoteDescription(offer);
+ browserPeerConnection.createAnswer()
+ .then(answer => {
+ browserPeerConnection.setLocalDescription(answer);
+
+ s.send(JSON.stringify({
+ dst: peerConnectionId,
+ src: connectionId,
+ method: 'answer',
+ answer,
+ }));
+ });
}
};
s.onclose = () => {
@@ -6859,11 +6993,5 @@
})();
-
-
-
-
-
-