Invite multiple Discord users to servers with checkboxes and automation controls
Size
56.4 KB
Version
1.2.4
Created
Nov 9, 2025
Updated
about 1 month ago
1// ==UserScript==
2// @name Discord Mass User Inviter
3// @description Invite multiple Discord users to servers with checkboxes and automation controls
4// @version 1.2.4
5// @match https://*.discord.com/*
6// @icon 
7// @grant GM.getValue
8// @grant GM.setValue
9// ==/UserScript==
10(function() {
11 'use strict';
12
13 console.log('Discord Mass User Inviter - Extension loaded');
14
15 // State management
16 let isInviting = false;
17 let invitedUsers = new Set();
18 let selectedUsers = new Set();
19 let targetServerId = null;
20 let isSelecting = false; // Flag to track if selection/deselection is in progress
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 // Load saved state
36 async function loadState() {
37 try {
38 const savedInvited = await GM.getValue('invitedUsers', '[]');
39 const savedServerId = await GM.getValue('targetServerId', null);
40 invitedUsers = new Set(JSON.parse(savedInvited));
41 targetServerId = savedServerId;
42 console.log('Loaded state:', { invitedUsers: invitedUsers.size, targetServerId });
43 } catch (error) {
44 console.error('Error loading state:', error);
45 }
46 }
47
48 // Save state
49 async function saveState() {
50 try {
51 await GM.setValue('invitedUsers', JSON.stringify([...invitedUsers]));
52 await GM.setValue('targetServerId', targetServerId);
53 console.log('State saved');
54 } catch (error) {
55 console.error('Error saving state:', error);
56 }
57 }
58
59 // Add styles
60 TM_addStyle(`
61 .mass-inviter-checkbox {
62 width: 16px;
63 height: 16px;
64 margin-right: 8px;
65 cursor: pointer;
66 flex-shrink: 0;
67 }
68
69 .mass-inviter-control-panel {
70 position: fixed;
71 top: 60px;
72 right: 20px;
73 background: #2b2d31;
74 border: 1px solid #1e1f22;
75 border-radius: 8px;
76 padding: 0;
77 z-index: 10000;
78 box-shadow: 0 8px 16px rgba(0, 0, 0, 0.4);
79 min-width: 280px;
80 color: #dbdee1;
81 font-family: 'gg sans', 'Noto Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
82 }
83
84 .mass-inviter-header {
85 background: #1e1f22;
86 padding: 12px 16px;
87 border-radius: 8px 8px 0 0;
88 cursor: move;
89 display: flex;
90 justify-content: space-between;
91 align-items: center;
92 user-select: none;
93 }
94
95 .mass-inviter-header:active {
96 cursor: grabbing;
97 }
98
99 .mass-inviter-control-panel h3 {
100 margin: 0;
101 font-size: 16px;
102 font-weight: 600;
103 color: #f2f3f5;
104 }
105
106 .mass-inviter-minimize-btn {
107 background: transparent;
108 border: none;
109 color: #b5bac1;
110 font-size: 20px;
111 cursor: pointer;
112 padding: 0;
113 width: 24px;
114 height: 24px;
115 display: flex;
116 align-items: center;
117 justify-content: center;
118 border-radius: 4px;
119 transition: background-color 0.2s, color 0.2s;
120 }
121
122 .mass-inviter-minimize-btn:hover {
123 background: #4e5058;
124 color: #ffffff;
125 }
126
127 .mass-inviter-content {
128 padding: 16px;
129 max-height: 600px;
130 overflow-y: auto;
131 }
132
133 .mass-inviter-content.minimized {
134 display: none;
135 }
136
137 .mass-inviter-control-panel h3 {
138 margin: 0 0 12px 0;
139 font-size: 16px;
140 font-weight: 600;
141 color: #f2f3f5;
142 }
143
144 .mass-inviter-button {
145 width: 100%;
146 padding: 10px 16px;
147 margin: 6px 0;
148 border: none;
149 border-radius: 4px;
150 font-size: 14px;
151 font-weight: 500;
152 cursor: pointer;
153 transition: background-color 0.2s;
154 color: white;
155 }
156
157 .mass-inviter-button.primary {
158 background: #5865f2;
159 }
160
161 .mass-inviter-button.primary:hover {
162 background: #4752c4;
163 }
164
165 .mass-inviter-button.secondary {
166 background: #4e5058;
167 }
168
169 .mass-inviter-button.secondary:hover {
170 background: #6d6f78;
171 }
172
173 .mass-inviter-button.danger {
174 background: #da373c;
175 }
176
177 .mass-inviter-button.danger:hover {
178 background: #a12d30;
179 }
180
181 .mass-inviter-button.success {
182 background: #248046;
183 }
184
185 .mass-inviter-button.success:hover {
186 background: #1a6334;
187 }
188
189 .mass-inviter-button:disabled {
190 opacity: 0.5;
191 cursor: not-allowed;
192 }
193
194 .mass-inviter-input {
195 width: 100%;
196 padding: 10px;
197 margin: 6px 0;
198 background: #1e1f22;
199 border: 1px solid #4e5058;
200 border-radius: 4px;
201 color: #dbdee1;
202 font-size: 14px;
203 box-sizing: border-box;
204 }
205
206 .mass-inviter-input:focus {
207 outline: none;
208 border-color: #5865f2;
209 }
210
211 .mass-inviter-status {
212 margin: 12px 0;
213 padding: 8px;
214 background: #1e1f22;
215 border-radius: 4px;
216 font-size: 12px;
217 color: #b5bac1;
218 }
219
220 .mass-inviter-invited {
221 opacity: 0.5;
222 background: #2d7d46 !important;
223 }
224
225 .mass-inviter-label {
226 font-size: 12px;
227 font-weight: 600;
228 color: #b5bac1;
229 margin-bottom: 4px;
230 display: block;
231 }
232
233 .mass-inviter-divider {
234 height: 1px;
235 background: #4e5058;
236 margin: 12px 0;
237 }
238 `);
239
240 // Get username from member element
241 function getUsernameFromMember(memberElement) {
242 const usernameElement = memberElement.querySelector('.username__703b9, .name__703b9');
243 return usernameElement ? usernameElement.textContent.trim() : null;
244 }
245
246 // Get user ID from member element
247 function getUserIdFromMember(memberElement) {
248 const listItemId = memberElement.getAttribute('data-list-item-id');
249 return listItemId || getUsernameFromMember(memberElement);
250 }
251
252 // Check if user is already invited
253 function isUserInvited(userId) {
254 return invitedUsers.has(userId);
255 }
256
257 // Add checkbox to member
258 function addCheckboxToMember(memberElement) {
259 // Check if checkbox already exists
260 if (memberElement.querySelector('.mass-inviter-checkbox')) {
261 return;
262 }
263
264 const userId = getUserIdFromMember(memberElement);
265 if (!userId) return;
266
267 const checkbox = document.createElement('input');
268 checkbox.type = 'checkbox';
269 checkbox.className = 'mass-inviter-checkbox';
270 checkbox.dataset.userId = userId;
271
272 // Check if user was already invited
273 if (isUserInvited(userId)) {
274 memberElement.classList.add('mass-inviter-invited');
275 checkbox.checked = false;
276 checkbox.disabled = true;
277 } else {
278 checkbox.checked = selectedUsers.has(userId);
279 }
280
281 checkbox.addEventListener('change', (e) => {
282 if (e.target.checked) {
283 selectedUsers.add(userId);
284 } else {
285 selectedUsers.delete(userId);
286 }
287 updateControlPanel();
288 });
289
290 // Insert checkbox at the beginning of the member element
291 const memberInner = memberElement.querySelector('.memberInner__5d473, .layout__91a9d');
292 if (memberInner) {
293 memberInner.insertBefore(checkbox, memberInner.firstChild);
294 }
295 }
296
297 // Add checkboxes to all members
298 function addCheckboxesToAllMembers() {
299 const memberElements = document.querySelectorAll('.member__5d473[role="listitem"]');
300 console.log('Found members:', memberElements.length);
301 memberElements.forEach(addCheckboxToMember);
302 }
303
304 // Create control panel
305 function createControlPanel() {
306 // Remove existing panel if any
307 const existingPanel = document.getElementById('mass-inviter-panel');
308 if (existingPanel) {
309 existingPanel.remove();
310 }
311
312 // Get list of servers
313 const guilds = [];
314 const guildElements = document.querySelectorAll('[data-list-item-id^="guildsnav___"]');
315 guildElements.forEach(el => {
316 const guildId = el.getAttribute('data-list-item-id').replace('guildsnav___', '');
317 // Get server name from multiple possible locations
318 const hiddenVisuallySpan = el.querySelector('.hiddenVisually_b18fe2');
319 const dndContainer = el.querySelector('[data-dnd-name]');
320 const guildName = (hiddenVisuallySpan ? hiddenVisuallySpan.textContent.trim() : null) ||
321 (dndContainer ? dndContainer.getAttribute('data-dnd-name') : null) ||
322 el.getAttribute('aria-label') ||
323 'Server ' + guildId;
324 if (guildId !== 'home' && guildId !== 'create-join-button' && guildId !== 'guild-discover-button' && guildId !== 'app-download-button') {
325 guilds.push({ id: guildId, name: guildName });
326 }
327 });
328
329 const panel = document.createElement('div');
330 panel.id = 'mass-inviter-panel';
331 panel.className = 'mass-inviter-control-panel';
332
333 // Build server options HTML
334 let serverOptionsHTML = '<option value="">Select a server...</option>';
335 guilds.forEach(guild => {
336 const selected = guild.id === targetServerId ? 'selected' : '';
337 serverOptionsHTML += `<option value="${guild.id}" ${selected}>${guild.name}</option>`;
338 });
339
340 panel.innerHTML = `
341 <div class="mass-inviter-header">
342 <h3>Mass User Inviter</h3>
343 <button class="mass-inviter-minimize-btn" id="mass-inviter-minimize-btn">-</button>
344 </div>
345
346 <div class="mass-inviter-content">
347 <label class="mass-inviter-label">Target Server</label>
348 <select id="mass-inviter-server-select" class="mass-inviter-input">
349 ${serverOptionsHTML}
350 </select>
351
352 <div class="mass-inviter-divider"></div>
353
354 <button id="mass-inviter-select-all" class="mass-inviter-button secondary">
355 Select All
356 </button>
357
358 <button id="mass-inviter-deselect-all" class="mass-inviter-button secondary">
359 Deselect All
360 </button>
361
362 <button id="mass-inviter-stop-selection" class="mass-inviter-button danger" style="display: none;">
363 Stop Selection
364 </button>
365
366 <div class="mass-inviter-divider"></div>
367
368 <button id="mass-inviter-start" class="mass-inviter-button success">
369 Start Inviting
370 </button>
371
372 <button id="mass-inviter-pause" class="mass-inviter-button danger" style="display: none;">
373 Pause
374 </button>
375
376 <div class="mass-inviter-status" id="mass-inviter-status">
377 Selected: <span id="mass-inviter-selected-count">0</span> users<br>
378 Invited: <span id="mass-inviter-invited-count">${invitedUsers.size}</span> users
379 </div>
380
381 <button id="mass-inviter-reset" class="mass-inviter-button secondary">
382 Reset Invited List
383 </button>
384 </div>
385 `;
386
387 document.body.appendChild(panel);
388
389 // Add event listeners
390 document.getElementById('mass-inviter-server-select').addEventListener('change', async (e) => {
391 targetServerId = e.target.value;
392 await saveState();
393 console.log('Target server changed to:', targetServerId);
394 });
395
396 document.getElementById('mass-inviter-select-all').addEventListener('click', selectAllUsers);
397 document.getElementById('mass-inviter-deselect-all').addEventListener('click', deselectAllUsers);
398 document.getElementById('mass-inviter-stop-selection').addEventListener('click', stopSelection);
399 document.getElementById('mass-inviter-start').addEventListener('click', startInviting);
400 document.getElementById('mass-inviter-pause').addEventListener('click', pauseInviting);
401 document.getElementById('mass-inviter-reset').addEventListener('click', resetInvitedList);
402
403 // Add minimize/maximize functionality
404 const minimizeBtn = document.getElementById('mass-inviter-minimize-btn');
405 const content = panel.querySelector('.mass-inviter-content');
406 let isMinimized = false;
407
408 minimizeBtn.addEventListener('click', () => {
409 isMinimized = !isMinimized;
410 if (isMinimized) {
411 content.classList.add('minimized');
412 minimizeBtn.textContent = '+';
413 } else {
414 content.classList.remove('minimized');
415 minimizeBtn.textContent = '-';
416 }
417 });
418
419 // Add drag functionality
420 const header = panel.querySelector('.mass-inviter-header');
421 let isDragging = false;
422 let currentX;
423 let currentY;
424 let initialX;
425 let initialY;
426 let xOffset = 0;
427 let yOffset = 0;
428
429 header.addEventListener('mousedown', dragStart);
430 document.addEventListener('mousemove', drag);
431 document.addEventListener('mouseup', dragEnd);
432
433 function dragStart(e) {
434 initialX = e.clientX - xOffset;
435 initialY = e.clientY - yOffset;
436
437 if (e.target === header || e.target.tagName === 'H3') {
438 isDragging = true;
439 }
440 }
441
442 function drag(e) {
443 if (isDragging) {
444 e.preventDefault();
445
446 currentX = e.clientX - initialX;
447 currentY = e.clientY - initialY;
448
449 xOffset = currentX;
450 yOffset = currentY;
451
452 setTranslate(currentX, currentY, panel);
453 }
454 }
455
456 function dragEnd() {
457 initialX = currentX;
458 initialY = currentY;
459 isDragging = false;
460 }
461
462 function setTranslate(xPos, yPos, el) {
463 el.style.transform = `translate3d(${xPos}px, ${yPos}px, 0)`;
464 }
465
466 updateControlPanel();
467 }
468
469 // Update control panel
470 function updateControlPanel() {
471 const selectedCountElement = document.getElementById('mass-inviter-selected-count');
472 const invitedCountElement = document.getElementById('mass-inviter-invited-count');
473
474 if (selectedCountElement) {
475 selectedCountElement.textContent = selectedUsers.size;
476 }
477 if (invitedCountElement) {
478 invitedCountElement.textContent = invitedUsers.size;
479 }
480 }
481
482 // Select all users
483 async function selectAllUsers() {
484 if (isSelecting) {
485 console.log('Selection already in progress');
486 return;
487 }
488
489 console.log('Select All clicked - loading and selecting all users...');
490
491 const membersContainer = document.querySelector('.members_c8ffbb');
492 if (!membersContainer) {
493 console.log('Members container not found');
494 return;
495 }
496
497 isSelecting = true;
498
499 // Show stop button, hide select/deselect buttons
500 document.getElementById('mass-inviter-select-all').style.display = 'none';
501 document.getElementById('mass-inviter-deselect-all').style.display = 'none';
502 document.getElementById('mass-inviter-stop-selection').style.display = 'block';
503
504 // The members_c8ffbb div itself is the scrollable container
505 const scrollableContainer = membersContainer;
506 let currentMemberCount = 0;
507 let noChangeCount = 0;
508 let totalSelected = 0;
509
510 console.log('Starting continuous scroll and select...');
511 console.log('Scrollable container found:', scrollableContainer.className);
512
513 while (noChangeCount < 3 && isSelecting) {
514 // Get current member count before scrolling
515 const beforeScrollCount = document.querySelectorAll('.member__5d473[role="listitem"]').length;
516
517 // Add checkboxes to currently visible members
518 addCheckboxesToAllMembers();
519
520 // Wait for checkboxes to be added
521 await new Promise(resolve => setTimeout(resolve, 200));
522
523 // Select all currently available checkboxes
524 const checkboxes = document.querySelectorAll('.mass-inviter-checkbox:not(:disabled)');
525 let newlySelected = 0;
526 checkboxes.forEach(checkbox => {
527 if (!checkbox.checked) {
528 checkbox.checked = true;
529 selectedUsers.add(checkbox.dataset.userId);
530 newlySelected++;
531 }
532 });
533
534 if (newlySelected > 0) {
535 totalSelected += newlySelected;
536 console.log('Selected', newlySelected, 'new users. Total selected:', totalSelected);
537 updateControlPanel();
538 }
539
540 // Check if we should stop
541 if (!isSelecting) {
542 console.log('Selection stopped by user');
543 break;
544 }
545
546 // Scroll down in chunks (500px at a time)
547 const currentScrollTop = scrollableContainer.scrollTop;
548 const scrollHeight = scrollableContainer.scrollHeight;
549 const clientHeight = scrollableContainer.clientHeight;
550 const maxScroll = scrollHeight - clientHeight;
551
552 // Scroll down by 500px or to the bottom, whichever is less
553 const scrollAmount = 500;
554 const newScrollPosition = Math.min(currentScrollTop + scrollAmount, maxScroll);
555 scrollableContainer.scrollTop = newScrollPosition;
556
557 const actualScrollTop = scrollableContainer.scrollTop;
558 console.log('Scroll chunk - from:', currentScrollTop, 'to:', actualScrollTop, 'max:', maxScroll);
559
560 // Wait for new members to load
561 await new Promise(resolve => setTimeout(resolve, 500));
562
563 // Get member count after scrolling
564 const afterScrollCount = document.querySelectorAll('.member__5d473[role="listitem"]').length;
565
566 console.log('Members before scroll:', beforeScrollCount, 'after scroll:', afterScrollCount);
567
568 if (afterScrollCount === beforeScrollCount) {
569 noChangeCount++;
570 console.log('No new members loaded, attempt', noChangeCount, 'of 3');
571 } else {
572 noChangeCount = 0;
573 console.log('New members loaded! Total members:', afterScrollCount);
574 }
575
576 currentMemberCount = afterScrollCount;
577 }
578
579 // Final pass - add checkboxes and select any remaining users
580 if (isSelecting) {
581 console.log('Doing final pass...');
582 addCheckboxesToAllMembers();
583 await new Promise(resolve => setTimeout(resolve, 300));
584
585 const finalCheckboxes = document.querySelectorAll('.mass-inviter-checkbox:not(:disabled)');
586 let finalSelected = 0;
587 finalCheckboxes.forEach(checkbox => {
588 if (!checkbox.checked) {
589 checkbox.checked = true;
590 selectedUsers.add(checkbox.dataset.userId);
591 finalSelected++;
592 }
593 });
594
595 if (finalSelected > 0) {
596 totalSelected += finalSelected;
597 console.log('Final pass selected', finalSelected, 'users');
598 }
599 }
600
601 updateControlPanel();
602 console.log('Select All completed. Total users selected:', totalSelected);
603 console.log('Total members in list:', currentMemberCount);
604
605 // Scroll back to top
606 scrollableContainer.scrollTop = 0;
607
608 // Reset UI
609 isSelecting = false;
610 document.getElementById('mass-inviter-select-all').style.display = 'block';
611 document.getElementById('mass-inviter-deselect-all').style.display = 'block';
612 document.getElementById('mass-inviter-stop-selection').style.display = 'none';
613 }
614
615 // Deselect all users
616 async function deselectAllUsers() {
617 if (isSelecting) {
618 console.log('Selection already in progress');
619 return;
620 }
621
622 console.log('Deselect All clicked - loading and deselecting all users...');
623
624 const membersContainer = document.querySelector('.members_c8ffbb');
625 if (!membersContainer) {
626 console.log('Members container not found');
627 return;
628 }
629
630 isSelecting = true;
631
632 // Show stop button, hide select/deselect buttons
633 document.getElementById('mass-inviter-select-all').style.display = 'none';
634 document.getElementById('mass-inviter-deselect-all').style.display = 'none';
635 document.getElementById('mass-inviter-stop-selection').style.display = 'block';
636
637 // The members_c8ffbb div itself is the scrollable container
638 const scrollableContainer = membersContainer;
639 let currentMemberCount = 0;
640 let noChangeCount = 0;
641 let totalDeselected = 0;
642
643 console.log('Starting continuous scroll and deselect...');
644 console.log('Scrollable container found:', scrollableContainer.className);
645
646 while (noChangeCount < 3 && isSelecting) {
647 // Get current member count before scrolling
648 const beforeScrollCount = document.querySelectorAll('.member__5d473[role="listitem"]').length;
649
650 // Add checkboxes to currently visible members
651 addCheckboxesToAllMembers();
652
653 // Wait for checkboxes to be added
654 await new Promise(resolve => setTimeout(resolve, 200));
655
656 // Deselect all currently available checkboxes
657 const checkboxes = document.querySelectorAll('.mass-inviter-checkbox');
658 let newlyDeselected = 0;
659 checkboxes.forEach(checkbox => {
660 if (checkbox.checked) {
661 checkbox.checked = false;
662 selectedUsers.delete(checkbox.dataset.userId);
663 newlyDeselected++;
664 }
665 });
666
667 if (newlyDeselected > 0) {
668 totalDeselected += newlyDeselected;
669 console.log('Deselected', newlyDeselected, 'users. Total deselected:', totalDeselected);
670 updateControlPanel();
671 }
672
673 // Check if we should stop
674 if (!isSelecting) {
675 console.log('Deselection stopped by user');
676 break;
677 }
678
679 // Scroll down in chunks (500px at a time)
680 const currentScrollTop = scrollableContainer.scrollTop;
681 const scrollHeight = scrollableContainer.scrollHeight;
682 const clientHeight = scrollableContainer.clientHeight;
683 const maxScroll = scrollHeight - clientHeight;
684
685 // Scroll down by 500px or to the bottom, whichever is less
686 const scrollAmount = 500;
687 const newScrollPosition = Math.min(currentScrollTop + scrollAmount, maxScroll);
688 scrollableContainer.scrollTop = newScrollPosition;
689
690 const actualScrollTop = scrollableContainer.scrollTop;
691 console.log('Scroll chunk - from:', currentScrollTop, 'to:', actualScrollTop, 'max:', maxScroll);
692
693 // Wait for new members to load
694 await new Promise(resolve => setTimeout(resolve, 500));
695
696 // Get member count after scrolling
697 const afterScrollCount = document.querySelectorAll('.member__5d473[role="listitem"]').length;
698
699 console.log('Members before scroll:', beforeScrollCount, 'after scroll:', afterScrollCount);
700
701 if (afterScrollCount === beforeScrollCount) {
702 noChangeCount++;
703 console.log('No new members loaded, attempt', noChangeCount, 'of 3');
704 } else {
705 noChangeCount = 0;
706 console.log('New members loaded! Total members:', afterScrollCount);
707 }
708
709 currentMemberCount = afterScrollCount;
710 }
711
712 // Final pass - add checkboxes and deselect any remaining users
713 if (isSelecting) {
714 console.log('Doing final pass...');
715 addCheckboxesToAllMembers();
716 await new Promise(resolve => setTimeout(resolve, 300));
717
718 const finalCheckboxes = document.querySelectorAll('.mass-inviter-checkbox');
719 let finalDeselected = 0;
720 finalCheckboxes.forEach(checkbox => {
721 if (checkbox.checked) {
722 checkbox.checked = false;
723 selectedUsers.delete(checkbox.dataset.userId);
724 finalDeselected++;
725 }
726 });
727
728 if (finalDeselected > 0) {
729 totalDeselected += finalDeselected;
730 console.log('Final pass deselected', finalDeselected, 'users');
731 }
732 }
733
734 updateControlPanel();
735 console.log('Deselect All completed. Total users deselected:', totalDeselected);
736 console.log('Total members in list:', currentMemberCount);
737
738 // Scroll back to top
739 scrollableContainer.scrollTop = 0;
740
741 // Reset UI
742 isSelecting = false;
743 document.getElementById('mass-inviter-select-all').style.display = 'block';
744 document.getElementById('mass-inviter-deselect-all').style.display = 'block';
745 document.getElementById('mass-inviter-stop-selection').style.display = 'none';
746 }
747
748 // Stop selection/deselection process
749 function stopSelection() {
750 isSelecting = false;
751 console.log('Selection/deselection stopped by user');
752
753 // Reset UI
754 document.getElementById('mass-inviter-select-all').style.display = 'block';
755 document.getElementById('mass-inviter-deselect-all').style.display = 'block';
756 document.getElementById('mass-inviter-stop-selection').style.display = 'none';
757 }
758
759 // Scroll to load all users in the member list
760 async function scrollToLoadAllUsers() {
761 const membersContainer = document.querySelector('.members_c8ffbb');
762 if (!membersContainer) {
763 console.log('Members container not found');
764 return;
765 }
766
767 // The members_c8ffbb div itself is the scrollable container
768 const scrollableContainer = membersContainer;
769 let previousMemberCount = 0;
770 let currentMemberCount = document.querySelectorAll('.member__5d473[role="listitem"]').length;
771 let noChangeCount = 0;
772
773 console.log('Starting to load all users...');
774
775 while (noChangeCount < 3) {
776 // Scroll to bottom
777 scrollableContainer.scrollTop = scrollableContainer.scrollHeight;
778
779 // Wait for new members to load
780 await new Promise(resolve => setTimeout(resolve, 300));
781
782 previousMemberCount = currentMemberCount;
783 currentMemberCount = document.querySelectorAll('.member__5d473[role="listitem"]').length;
784
785 if (currentMemberCount === previousMemberCount) {
786 noChangeCount++;
787 } else {
788 noChangeCount = 0;
789 console.log('Loaded members:', currentMemberCount);
790 }
791 }
792
793 console.log('Finished loading all users. Total:', currentMemberCount);
794
795 // Scroll back to top
796 scrollableContainer.scrollTop = 0;
797 }
798
799 // Reset invited list
800 async function resetInvitedList() {
801 if (confirm('Are you sure you want to reset the invited users list?')) {
802 invitedUsers.clear();
803 await saveState();
804
805 // Re-enable all checkboxes
806 const checkboxes = document.querySelectorAll('.mass-inviter-checkbox');
807 checkboxes.forEach(checkbox => {
808 checkbox.disabled = false;
809 const memberElement = checkbox.closest('.member__5d473');
810 if (memberElement) {
811 memberElement.classList.remove('mass-inviter-invited');
812 }
813 });
814
815 updateControlPanel();
816 console.log('Invited list reset');
817 }
818 }
819
820 // Simulate right-click on member
821 async function rightClickMember(memberElement) {
822 return new Promise((resolve) => {
823 const rect = memberElement.getBoundingClientRect();
824 const x = rect.left + rect.width / 2;
825 const y = rect.top + rect.height / 2;
826
827 const contextMenuEvent = new MouseEvent('contextmenu', {
828 bubbles: true,
829 cancelable: true,
830 view: window,
831 clientX: x,
832 clientY: y,
833 button: 2
834 });
835
836 memberElement.dispatchEvent(contextMenuEvent);
837 console.log('Right-clicked member');
838
839 // Wait for context menu to appear
840 setTimeout(resolve, 1000);
841 });
842 }
843
844 // Click invite option in context menu
845 async function clickInviteOption() {
846 return new Promise((resolve) => {
847 setTimeout(() => {
848 // Look for "Invite to Server" option in context menu
849 const inviteMenuItem = document.getElementById('user-context-invite-to-server');
850
851 if (inviteMenuItem) {
852 // Hover over it to open submenu (don't click!)
853 inviteMenuItem.dispatchEvent(new MouseEvent('mouseenter', { bubbles: true }));
854 inviteMenuItem.dispatchEvent(new MouseEvent('mouseover', { bubbles: true }));
855 console.log('Hovered over invite option, waiting for submenu...');
856 setTimeout(resolve, 800);
857 } else {
858 const menuItems = document.querySelectorAll('[role="menuitem"]');
859 console.error('Invite option not found in context menu. Available options:',
860 Array.from(menuItems).map(item => item.textContent.trim()).join(', '));
861 resolve();
862 }
863 }, 500);
864 });
865 }
866
867 // Select server from invite modal
868 async function selectServerFromModal() {
869 return new Promise((resolve) => {
870 setTimeout(() => {
871 console.log('Looking for server in submenu...');
872
873 // The server appears in submenu with ID format: user-context-invite-to-server--[SERVER_ID]
874 const serverMenuItem = document.getElementById('user-context-invite-to-server--' + targetServerId);
875
876 if (serverMenuItem) {
877 serverMenuItem.click();
878 console.log('Clicked target server:', targetServerId);
879 setTimeout(resolve, 500);
880 return;
881 }
882
883 // If not found by exact ID, try to find by partial match or name
884 const allServerItems = document.querySelectorAll('[id^="user-context-invite-to-server--"]');
885 console.log('Found', allServerItems.length, 'server options in submenu');
886
887 for (const serverItem of allServerItems) {
888 const itemId = serverItem.id.replace('user-context-invite-to-server--', '');
889 const serverName = serverItem.textContent.trim();
890
891 console.log('Checking server:', serverName, 'ID:', itemId);
892
893 // Try matching by ID or name
894 if (itemId === targetServerId || serverName.toLowerCase().includes(targetServerId.toLowerCase())) {
895 serverItem.click();
896 console.log('Selected target server:', serverName);
897 setTimeout(resolve, 500);
898 return;
899 }
900 }
901
902 console.error('Target server not found in submenu. Looking for:', targetServerId);
903 console.error('Available servers:',
904 Array.from(allServerItems).map(el => el.textContent.trim() + ' (ID: ' + el.id.replace('user-context-invite-to-server--', '') + ')').join(', '));
905 resolve();
906 }, 300);
907 });
908 }
909
910 // Close any open modals
911 function closeModals() {
912 const closeButtons = document.querySelectorAll('[aria-label="Close"], [class*="closeButton"]');
913 closeButtons.forEach(button => button.click());
914
915 // Press Escape key
916 document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', keyCode: 27, bubbles: true }));
917 }
918
919 // Invite single user
920 async function inviteUser(userId) {
921 const memberElement = document.querySelector(`.mass-inviter-checkbox[data-user-id="${userId}"]`)?.closest('.member__5d473');
922
923 if (!memberElement) {
924 console.error('Member element not found for user:', userId);
925 return false;
926 }
927
928 if (!targetServerId) {
929 alert('Please set a target server ID first!');
930 return false;
931 }
932
933 try {
934 console.log('Inviting user:', userId);
935
936 // Right-click the member
937 await rightClickMember(memberElement);
938
939 // Click invite option
940 await clickInviteOption();
941
942 // Select server from modal
943 await selectServerFromModal();
944
945 // Mark as invited
946 invitedUsers.add(userId);
947 selectedUsers.delete(userId);
948 await saveState();
949
950 // Update UI
951 const checkbox = memberElement.querySelector('.mass-inviter-checkbox');
952 if (checkbox) {
953 checkbox.checked = false;
954 checkbox.disabled = true;
955 }
956 memberElement.classList.add('mass-inviter-invited');
957
958 // Close any modals
959 closeModals();
960
961 console.log('User invited successfully:', userId);
962 return true;
963 } catch (error) {
964 console.error('Error inviting user:', error);
965 closeModals();
966 return false;
967 }
968 }
969
970 // Start inviting process
971 async function startInviting() {
972 if (isInviting) return;
973
974 if (!targetServerId) {
975 alert('Please set a target server ID first!');
976 return;
977 }
978
979 if (selectedUsers.size === 0) {
980 alert('Please select at least one user to invite!');
981 return;
982 }
983
984 isInviting = true;
985 document.getElementById('mass-inviter-start').style.display = 'none';
986 document.getElementById('mass-inviter-pause').style.display = 'block';
987
988 const usersToInvite = [...selectedUsers];
989 console.log('Starting invitation process for', usersToInvite.length, 'users');
990
991 // Get members container for scrolling
992 const membersContainer = document.querySelector('.members_c8ffbb');
993 const scrollableContainer = membersContainer;
994 let currentScrollPosition = 0;
995
996 for (let i = 0; i < usersToInvite.length; i++) {
997 const userId = usersToInvite[i];
998
999 if (!isInviting) {
1000 console.log('Invitation process paused');
1001 break;
1002 }
1003
1004 // Scroll down periodically to keep loading members
1005 if (i > 0 && i % 5 === 0 && scrollableContainer) {
1006 const scrollHeight = scrollableContainer.scrollHeight;
1007 const clientHeight = scrollableContainer.clientHeight;
1008 const maxScroll = scrollHeight - clientHeight;
1009
1010 // Scroll down by 500px
1011 currentScrollPosition = Math.min(currentScrollPosition + 500, maxScroll);
1012 scrollableContainer.scrollTop = currentScrollPosition;
1013 console.log('Scrolled to position:', currentScrollPosition);
1014
1015 // Wait for members to load
1016 await new Promise(resolve => setTimeout(resolve, 500));
1017
1018 // Re-add checkboxes to newly loaded members
1019 addCheckboxesToAllMembers();
1020 }
1021
1022 await inviteUser(userId);
1023 updateControlPanel();
1024
1025 // Wait between invitations to avoid rate limiting
1026 await new Promise(resolve => setTimeout(resolve, 30000));
1027 }
1028
1029 // Scroll back to top when done
1030 if (scrollableContainer) {
1031 scrollableContainer.scrollTop = 0;
1032 }
1033
1034 isInviting = false;
1035 document.getElementById('mass-inviter-start').style.display = 'block';
1036 document.getElementById('mass-inviter-pause').style.display = 'none';
1037
1038 console.log('Invitation process completed');
1039 }
1040
1041 // Pause inviting process
1042 function pauseInviting() {
1043 isInviting = false;
1044 document.getElementById('mass-inviter-start').style.display = 'block';
1045 document.getElementById('mass-inviter-pause').style.display = 'none';
1046 console.log('Invitation process paused by user');
1047 }
1048
1049 // Initialize extension
1050 async function init() {
1051 console.log('Initializing Discord Mass User Inviter');
1052
1053 await loadState();
1054
1055 // Function to setup observers
1056 function setupObservers() {
1057 const membersContainer = document.querySelector('.members_c8ffbb');
1058 if (membersContainer) {
1059 console.log('Members list found, adding checkboxes');
1060
1061 addCheckboxesToAllMembers();
1062
1063 // Create control panel if it doesn't exist
1064 if (!document.getElementById('mass-inviter-panel')) {
1065 createControlPanel();
1066 }
1067
1068 // Observe for new members being added
1069 const observer = new MutationObserver(debounce(() => {
1070 console.log('Members list changed, updating checkboxes');
1071 addCheckboxesToAllMembers();
1072 }, 200));
1073
1074 observer.observe(membersContainer, {
1075 childList: true,
1076 subtree: true
1077 });
1078
1079 console.log('Observer attached to members container');
1080 return true;
1081 }
1082 return false;
1083 }
1084
1085 // Initial setup
1086 const initialWait = setInterval(() => {
1087 if (setupObservers()) {
1088 clearInterval(initialWait);
1089 console.log('Extension initialized successfully');
1090 }
1091 }, 1000);
1092
1093 // Watch for the entire members container being replaced (when switching servers)
1094 const bodyObserver = new MutationObserver(debounce(() => {
1095 const membersContainer = document.querySelector('.members_c8ffbb');
1096 if (membersContainer) {
1097 // Check if we need to re-add checkboxes
1098 const hasCheckboxes = membersContainer.querySelector('.mass-inviter-checkbox');
1099 if (!hasCheckboxes) {
1100 console.log('Members container replaced, re-initializing...');
1101 setupObservers();
1102 }
1103 }
1104 }, 100));
1105
1106 bodyObserver.observe(document.body, {
1107 childList: true,
1108 subtree: true
1109 });
1110
1111 console.log('Body observer attached for server switches');
1112 }
1113
1114 // Start when DOM is ready
1115 if (document.readyState === 'loading') {
1116 document.addEventListener('DOMContentLoaded', init);
1117 } else {
1118 init();
1119 }
1120})();