Discord Mass User Inviter

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})();
Discord Mass User Inviter | Robomonkey