Add video call functionality to Whimsical boards for real-time collaboration
Size
21.0 KB
Version
1.1.10
Created
Nov 7, 2025
Updated
about 1 month ago
1// ==UserScript==
2// @name Whimsical Video Call
3// @description Add video call functionality to Whimsical boards for real-time collaboration
4// @version 1.1.10
5// @match https://*.whimsical.com/*
6// @icon 
7// @require https://cdn.jsdelivr.net/npm/peerjs@1.5.2/dist/peerjs.min.js
8// ==/UserScript==
9(function() {
10 'use strict';
11
12 console.log('Whimsical Video Call extension loaded');
13
14 let localStream = null;
15 let peer = null;
16 let currentCall = null;
17 let roomId = null;
18 let isCallActive = false;
19 let myPeerId = null;
20 let isFirstPeer = false;
21
22 // Debounce function for MutationObserver
23 function debounce(func, wait) {
24 let timeout;
25 return function executedFunction(...args) {
26 const later = () => {
27 clearTimeout(timeout);
28 func(...args);
29 };
30 clearTimeout(timeout);
31 timeout = setTimeout(later, wait);
32 };
33 }
34
35 // Initialize the extension
36 async function init() {
37 console.log('Initializing Whimsical Video Call');
38
39 // Wait for the page to be fully loaded
40 if (document.readyState === 'loading') {
41 document.addEventListener('DOMContentLoaded', addVideoCallButton);
42 } else {
43 addVideoCallButton();
44 }
45
46 // Watch for dynamic changes
47 const observer = new MutationObserver(debounce(() => {
48 if (!document.querySelector('#whimsical-video-call-btn')) {
49 addVideoCallButton();
50 }
51 }, 500));
52
53 observer.observe(document.body, {
54 childList: true,
55 subtree: true
56 });
57 }
58
59 // Add video call button
60 function addVideoCallButton() {
61 if (document.querySelector('#whimsical-video-call-btn')) {
62 console.log('Video call button already exists');
63 return;
64 }
65
66 console.log('Adding video call button to bottom-left corner');
67
68 const videoCallBtn = document.createElement('button');
69 videoCallBtn.id = 'whimsical-video-call-btn';
70 videoCallBtn.innerHTML = `
71 <svg width="20" height="20" viewBox="0 0 16 16" fill="white">
72 <path d="M12 0C14.2091 0 16 1.79086 16 4V10C16 12.2091 14.2091 14 12 14H11.6182L11.8945 14.5527C12.1415 15.0467 11.9412 15.6475 11.4473 15.8945C10.9533 16.1415 10.3525 15.9412 10.1055 15.4473L9.38184 14H6.61816L5.89453 15.4473C5.64754 15.9412 5.04671 16.1415 4.55273 15.8945C4.05876 15.6475 3.85848 15.0467 4.10547 14.5527L4.38184 14H4C1.79086 14 0 12.2091 0 10V4C0 1.79086 1.79086 0 4 0H12ZM4 2C2.89543 2 2 2.89543 2 4V10C2 11.1046 2.89543 12 4 12H12C13.1046 12 14 11.1046 14 10V4C14 2.89543 13.1046 2 12 2H4ZM5.5 8C5.77614 8 6 8.22386 6 8.5V9.5C6 9.77614 5.77614 10 5.5 10H4.5C4.22386 10 4 9.77614 4 9.5V8.5C4 8.22386 4.22386 8 4.5 8H5.5ZM5.5 4C5.77614 4 6 4.22386 6 4.5V5.5C6 5.77614 5.77614 6 5.5 6H4.5C4.22386 6 4 5.77614 4 5.5V4.5C4 4.22386 4.22386 4 4.5 4H5.5ZM9.5 4C9.77614 4 10 4.22386 10 4.5V5.5C10 5.77614 9.77614 6 9.5 6H8.5C8.22386 6 8 5.77614 8 5.5V4.5C8 4.22386 8.22386 4 8.5 4H9.5Z"/>
73 </svg>
74 <span id="video-call-btn-text">Video Call</span>
75 `;
76 videoCallBtn.style.cssText = `
77 position: fixed;
78 bottom: 20px;
79 left: 20px;
80 z-index: 999998;
81 background: #730FC3;
82 color: white;
83 border: none;
84 border-radius: 50px;
85 padding: 14px 24px;
86 font-size: 15px;
87 font-weight: 600;
88 cursor: pointer;
89 display: inline-flex;
90 align-items: center;
91 justify-content: center;
92 gap: 8px;
93 box-shadow: 0 4px 12px rgba(115, 15, 195, 0.4);
94 transition: all 0.2s ease;
95 font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
96 `;
97
98 videoCallBtn.addEventListener('click', toggleVideoCall);
99
100 videoCallBtn.addEventListener('mouseenter', () => {
101 if (!isCallActive) {
102 videoCallBtn.style.transform = 'scale(1.05)';
103 videoCallBtn.style.boxShadow = '0 6px 16px rgba(115, 15, 195, 0.5)';
104 }
105 });
106
107 videoCallBtn.addEventListener('mouseleave', () => {
108 videoCallBtn.style.transform = 'scale(1)';
109 videoCallBtn.style.boxShadow = '0 4px 12px rgba(115, 15, 195, 0.4)';
110 });
111
112 document.body.appendChild(videoCallBtn);
113 console.log('Video call button added successfully');
114 }
115
116 // Toggle video call on/off
117 async function toggleVideoCall() {
118 if (isCallActive) {
119 endCall();
120 } else {
121 await startCall();
122 }
123 }
124
125 // Start video call
126 async function startCall() {
127 try {
128 console.log('Starting video call...');
129
130 // Get room ID from URL
131 const urlParts = window.location.pathname.split('-');
132 roomId = 'whimsical-' + urlParts[urlParts.length - 1];
133 console.log('Room ID:', roomId);
134
135 // Request camera and microphone access
136 localStream = await navigator.mediaDevices.getUserMedia({
137 video: true,
138 audio: true
139 });
140
141 console.log('Got local stream');
142
143 // Create video call UI
144 createVideoCallUI();
145
146 // Display local video
147 const localVideo = document.getElementById('local-video');
148 if (localVideo) {
149 localVideo.srcObject = localStream;
150 }
151
152 // Initialize PeerJS
153 initializePeer();
154
155 // Update button state
156 isCallActive = true;
157 updateButtonState();
158
159 console.log('Video call started successfully');
160
161 } catch (error) {
162 console.error('Error starting video call:', error);
163 alert('Could not access camera/microphone. Please check permissions.');
164 }
165 }
166
167 // Initialize PeerJS
168 function initializePeer() {
169 console.log('Initializing PeerJS...');
170
171 // Try to be the first peer with the room ID
172 peer = new Peer(roomId, {
173 debug: 2
174 });
175
176 peer.on('open', (id) => {
177 myPeerId = id;
178 isFirstPeer = true;
179 console.log('I am the FIRST peer in the room');
180 console.log('My peer ID:', myPeerId);
181 console.log('Waiting for others to join...');
182 });
183
184 peer.on('call', (call) => {
185 console.log('Receiving call from:', call.peer);
186
187 // Answer the call with our stream
188 call.answer(localStream);
189
190 call.on('stream', (remoteStream) => {
191 console.log('Received remote stream');
192 displayRemoteStream(remoteStream);
193 });
194
195 call.on('close', () => {
196 console.log('Call closed');
197 });
198
199 currentCall = call;
200 });
201
202 peer.on('error', (err) => {
203 console.error('PeerJS error:', err);
204
205 // If peer ID is taken, it means someone else is already in the room
206 if (err.type === 'unavailable-id') {
207 console.log('Room already occupied, I am the SECOND peer');
208 isFirstPeer = false;
209
210 // Destroy the failed peer
211 if (peer) {
212 peer.destroy();
213 }
214
215 // Create a new peer with a unique ID
216 const myUniquePeerId = roomId + '-peer2-' + Math.random().toString(36).substr(2, 9);
217 peer = new Peer(myUniquePeerId, {
218 debug: 2
219 });
220
221 peer.on('open', (id) => {
222 myPeerId = id;
223 console.log('My peer ID:', myPeerId);
224 console.log('Calling the first peer:', roomId);
225
226 // Call the first peer
227 setTimeout(() => {
228 callPeer(roomId);
229 }, 1000);
230 });
231
232 peer.on('call', (call) => {
233 console.log('Receiving call from:', call.peer);
234
235 // Answer the call with our stream
236 call.answer(localStream);
237
238 call.on('stream', (remoteStream) => {
239 console.log('Received remote stream');
240 displayRemoteStream(remoteStream);
241 });
242
243 call.on('close', () => {
244 console.log('Call closed');
245 });
246
247 currentCall = call;
248 });
249
250 peer.on('error', (err2) => {
251 console.error('Second peer error:', err2);
252 });
253 }
254 });
255
256 peer.on('disconnected', () => {
257 console.log('Peer disconnected');
258 });
259 }
260
261 // Call a specific peer
262 function callPeer(peerId) {
263 console.log('Calling peer:', peerId);
264
265 const call = peer.call(peerId, localStream);
266
267 if (!call) {
268 console.error('Failed to create call');
269 return;
270 }
271
272 call.on('stream', (remoteStream) => {
273 console.log('Received remote stream from call');
274 displayRemoteStream(remoteStream);
275 });
276
277 call.on('close', () => {
278 console.log('Call closed');
279 });
280
281 call.on('error', (err) => {
282 console.error('Call error:', err);
283 });
284
285 currentCall = call;
286 }
287
288 // Display remote stream
289 function displayRemoteStream(stream) {
290 const remoteVideo = document.getElementById('remote-video');
291 const waitingMessage = document.getElementById('waiting-message');
292
293 if (remoteVideo) {
294 remoteVideo.srcObject = stream;
295 if (waitingMessage) {
296 waitingMessage.style.display = 'none';
297 }
298 }
299 }
300
301 // End video call
302 function endCall() {
303 console.log('Ending video call...');
304
305 // Stop local stream
306 if (localStream) {
307 localStream.getTracks().forEach(track => track.stop());
308 localStream = null;
309 }
310
311 // Close current call
312 if (currentCall) {
313 currentCall.close();
314 currentCall = null;
315 }
316
317 // Destroy peer
318 if (peer) {
319 peer.destroy();
320 peer = null;
321 }
322
323 // Remove video call UI
324 const videoCallContainer = document.getElementById('whimsical-video-call-container');
325 if (videoCallContainer) {
326 videoCallContainer.remove();
327 }
328
329 // Update button state
330 isCallActive = false;
331 updateButtonState();
332
333 console.log('Video call ended');
334 }
335
336 // Update button state
337 function updateButtonState() {
338 const btn = document.getElementById('whimsical-video-call-btn');
339 const btnText = document.getElementById('video-call-btn-text');
340
341 if (btn && btnText) {
342 if (isCallActive) {
343 btnText.textContent = 'End Call';
344 btn.style.backgroundColor = '#dc3545';
345 btn.style.color = 'white';
346 } else {
347 btnText.textContent = 'Video Call';
348 btn.style.backgroundColor = '';
349 btn.style.color = '';
350 }
351 }
352 }
353
354 // Create video call UI
355 function createVideoCallUI() {
356 if (document.getElementById('whimsical-video-call-container')) {
357 return;
358 }
359
360 const container = document.createElement('div');
361 container.id = 'whimsical-video-call-container';
362 container.innerHTML = `
363 <div id="video-call-header">
364 <span>Video Call</span>
365 <button id="minimize-video-call" title="Minimize">−</button>
366 <button id="maximize-video-call" title="Maximize">□</button>
367 <button id="close-video-call" title="End Call">×</button>
368 </div>
369 <div id="video-call-content">
370 <div id="remote-video-container">
371 <video id="remote-video" autoplay playsinline></video>
372 <div id="waiting-message">Waiting for others to join...</div>
373 </div>
374 <div id="local-video-container">
375 <video id="local-video" autoplay playsinline muted></video>
376 </div>
377 <div id="video-call-controls">
378 <button id="toggle-mic" class="control-btn" title="Mute/Unmute">
379 <svg width="20" height="20" viewBox="0 0 24 24" fill="white">
380 <path d="M12 14c1.66 0 3-1.34 3-3V5c0-1.66-1.34-3-3-3S9 3.34 9 5v6c0 1.66 1.34 3 3 3z"/>
381 <path d="M17 11c0 2.76-2.24 5-5 5s-5-2.24-5-5H5c0 3.53 2.61 6.43 6 6.92V21h2v-3.08c3.39-.49 6-3.39 6-6.92h-2z"/>
382 </svg>
383 </button>
384 <button id="toggle-camera" class="control-btn" title="Turn Camera On/Off">
385 <svg width="20" height="20" viewBox="0 0 24 24" fill="white">
386 <path d="M17 10.5V7c0-.55-.45-1-1-1H4c-.55 0-1 .45-1 1v10c0 .55.45 1 1 1h12c.55 0 1-.45 1-1v-3.5l4 4v-11l-4 4z"/>
387 </svg>
388 </button>
389 </div>
390 </div>
391 `;
392
393 const style = document.createElement('style');
394 style.textContent = `
395 #whimsical-video-call-container {
396 position: fixed;
397 bottom: 20px;
398 right: 20px;
399 width: 320px;
400 background: #1a1a1a;
401 border-radius: 12px;
402 box-shadow: 0 8px 32px rgba(0, 0, 0, 0.4);
403 z-index: 999999;
404 overflow: hidden;
405 font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
406 transition: all 0.3s ease;
407 }
408
409 #whimsical-video-call-container.minimized #video-call-content {
410 display: none;
411 }
412
413 #whimsical-video-call-container.maximized {
414 width: 90vw;
415 height: 90vh;
416 bottom: 5vh;
417 right: 5vw;
418 border-radius: 16px;
419 }
420
421 #whimsical-video-call-container.maximized #remote-video-container {
422 height: calc(90vh - 120px);
423 }
424
425 #whimsical-video-call-container.maximized #local-video-container {
426 width: 200px;
427 height: 150px;
428 bottom: 90px;
429 right: 24px;
430 }
431
432 #video-call-header {
433 background: #2d2d2d;
434 color: white;
435 padding: 10px 12px;
436 display: flex;
437 align-items: center;
438 justify-content: space-between;
439 font-weight: 600;
440 font-size: 13px;
441 }
442
443 #video-call-header button {
444 background: none;
445 border: none;
446 color: white;
447 font-size: 18px;
448 cursor: pointer;
449 padding: 0 6px;
450 opacity: 0.8;
451 transition: opacity 0.2s;
452 }
453
454 #video-call-header button:hover {
455 opacity: 1;
456 }
457
458 #video-call-content {
459 position: relative;
460 background: #000;
461 }
462
463 #remote-video-container {
464 position: relative;
465 width: 100%;
466 height: 240px;
467 background: #000;
468 display: flex;
469 align-items: center;
470 justify-content: center;
471 transition: height 0.3s ease;
472 }
473
474 #remote-video {
475 width: 100%;
476 height: 100%;
477 object-fit: cover;
478 }
479
480 #waiting-message {
481 position: absolute;
482 color: white;
483 font-size: 13px;
484 text-align: center;
485 padding: 20px;
486 }
487
488 #local-video-container {
489 position: absolute;
490 bottom: 60px;
491 right: 12px;
492 width: 100px;
493 height: 75px;
494 border-radius: 8px;
495 overflow: hidden;
496 border: 2px solid white;
497 box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
498 transition: all 0.3s ease;
499 }
500
501 #local-video {
502 width: 100%;
503 height: 100%;
504 object-fit: cover;
505 transform: scaleX(-1);
506 }
507
508 #video-call-controls {
509 position: absolute;
510 bottom: 12px;
511 left: 50%;
512 transform: translateX(-50%);
513 display: flex;
514 gap: 10px;
515 }
516
517 .control-btn {
518 width: 42px;
519 height: 42px;
520 border-radius: 50%;
521 background: rgba(255, 255, 255, 0.2);
522 border: none;
523 cursor: pointer;
524 display: flex;
525 align-items: center;
526 justify-content: center;
527 transition: background 0.2s;
528 }
529
530 .control-btn:hover {
531 background: rgba(255, 255, 255, 0.3);
532 }
533
534 .control-btn.muted {
535 background: #dc3545;
536 }
537
538 .control-btn svg {
539 width: 18px;
540 height: 18px;
541 }
542 `;
543
544 document.head.appendChild(style);
545 document.body.appendChild(container);
546
547 // Add event listeners
548 document.getElementById('minimize-video-call').addEventListener('click', () => {
549 container.classList.toggle('minimized');
550 });
551
552 document.getElementById('maximize-video-call').addEventListener('click', () => {
553 container.classList.toggle('maximized');
554 const btn = document.getElementById('maximize-video-call');
555 if (container.classList.contains('maximized')) {
556 btn.textContent = '▢';
557 btn.title = 'Restore';
558 } else {
559 btn.textContent = '□';
560 btn.title = 'Maximize';
561 }
562 });
563
564 document.getElementById('close-video-call').addEventListener('click', endCall);
565
566 document.getElementById('toggle-mic').addEventListener('click', () => {
567 if (localStream) {
568 const audioTrack = localStream.getAudioTracks()[0];
569 if (audioTrack) {
570 audioTrack.enabled = !audioTrack.enabled;
571 document.getElementById('toggle-mic').classList.toggle('muted', !audioTrack.enabled);
572 }
573 }
574 });
575
576 document.getElementById('toggle-camera').addEventListener('click', () => {
577 if (localStream) {
578 const videoTrack = localStream.getVideoTracks()[0];
579 if (videoTrack) {
580 videoTrack.enabled = !videoTrack.enabled;
581 document.getElementById('toggle-camera').classList.toggle('muted', !videoTrack.enabled);
582 }
583 }
584 });
585 }
586
587 // Start the extension
588 init();
589
590})();