Slack NPS Chat Summarizer

Summarizes all NPS chats from the NPS channel using AI

Size

23.5 KB

Version

1.0.1

Created

Oct 30, 2025

Updated

15 days ago

1// ==UserScript==
2// @name		Slack NPS Chat Summarizer
3// @description		Summarizes all NPS chats from the NPS channel using AI
4// @version		1.0.1
5// @match		https://*.app.slack.com/*
6// @icon		
7// ==/UserScript==
8(function() {
9    'use strict';
10
11    console.log('NPS Chat Summarizer extension loaded');
12
13    // Debounce function to prevent excessive calls
14    function debounce(func, wait) {
15        let timeout;
16        return function executedFunction(...args) {
17            const later = () => {
18                clearTimeout(timeout);
19                func(...args);
20            };
21            clearTimeout(timeout);
22            timeout = setTimeout(later, wait);
23        };
24    }
25
26    // Extract all messages from the current Slack channel
27    function extractMessages() {
28        console.log('Extracting messages from Slack channel...');
29        const messages = [];
30        
31        // Find all message elements in Slack
32        const messageElements = document.querySelectorAll('[data-qa="virtual-list-item"]');
33        
34        console.log(`Found ${messageElements.length} message elements`);
35        
36        messageElements.forEach((element, index) => {
37            try {
38                // Extract sender name
39                const senderElement = element.querySelector('[data-qa="message_sender_name"]') ||
40                                     element.querySelector('.c-message__sender_button') ||
41                                     element.querySelector('.c-message__sender');
42                const sender = senderElement ? senderElement.textContent.trim() : 'Unknown';
43                
44                // Extract message text
45                const messageTextElement = element.querySelector('[data-qa="message-text"]') ||
46                                          element.querySelector('.c-message_kit__text') ||
47                                          element.querySelector('.p-rich_text_section');
48                const text = messageTextElement ? messageTextElement.textContent.trim() : '';
49                
50                // Extract timestamp
51                const timestampElement = element.querySelector('[data-qa="message_timestamp"]') ||
52                                        element.querySelector('.c-timestamp');
53                const timestamp = timestampElement ? timestampElement.textContent.trim() : '';
54                
55                if (text && text.length > 0) {
56                    messages.push({
57                        sender: sender,
58                        text: text,
59                        timestamp: timestamp
60                    });
61                }
62            } catch (error) {
63                console.error(`Error extracting message ${index}:`, error);
64            }
65        });
66        
67        console.log(`Successfully extracted ${messages.length} messages`);
68        return messages;
69    }
70
71    // Create and show loading indicator
72    function showLoadingIndicator() {
73        const existingLoader = document.getElementById('nps-summarizer-loader');
74        if (existingLoader) return;
75        
76        const loader = document.createElement('div');
77        loader.id = 'nps-summarizer-loader';
78        loader.innerHTML = `
79            <div style="display: flex; align-items: center; gap: 10px;">
80                <div class="spinner"></div>
81                <span>AI is analyzing NPS chats...</span>
82            </div>
83        `;
84        loader.style.cssText = `
85            position: fixed;
86            top: 20px;
87            right: 20px;
88            background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
89            color: white;
90            padding: 15px 20px;
91            border-radius: 8px;
92            box-shadow: 0 4px 12px rgba(0,0,0,0.3);
93            z-index: 10000;
94            font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
95            font-size: 14px;
96        `;
97        
98        document.body.appendChild(loader);
99        
100        // Add spinner animation
101        const style = document.createElement('style');
102        style.textContent = `
103            #nps-summarizer-loader .spinner {
104                width: 20px;
105                height: 20px;
106                border: 3px solid rgba(255,255,255,0.3);
107                border-top-color: white;
108                border-radius: 50%;
109                animation: spin 1s linear infinite;
110            }
111            @keyframes spin {
112                to { transform: rotate(360deg); }
113            }
114        `;
115        document.head.appendChild(style);
116    }
117
118    // Hide loading indicator
119    function hideLoadingIndicator() {
120        const loader = document.getElementById('nps-summarizer-loader');
121        if (loader) loader.remove();
122    }
123
124    // Display summary in a modal
125    function displaySummary(summary) {
126        console.log('Displaying summary modal');
127        
128        // Remove existing modal if any
129        const existingModal = document.getElementById('nps-summary-modal');
130        if (existingModal) existingModal.remove();
131        
132        const modal = document.createElement('div');
133        modal.id = 'nps-summary-modal';
134        modal.innerHTML = `
135            <div class="modal-overlay">
136                <div class="modal-content">
137                    <div class="modal-header">
138                        <h2>📊 NPS Chat Summary</h2>
139                        <button class="close-btn" id="close-summary-modal"></button>
140                    </div>
141                    <div class="modal-body">
142                        <div class="summary-section">
143                            <h3>🎯 Key Themes</h3>
144                            <ul>
145                                ${summary.keyThemes.map(theme => `<li>${theme}</li>`).join('')}
146                            </ul>
147                        </div>
148                        
149                        <div class="summary-section">
150                            <h3>😊 Positive Feedback</h3>
151                            <ul>
152                                ${summary.positiveFeedback.map(item => `<li>${item}</li>`).join('')}
153                            </ul>
154                        </div>
155                        
156                        <div class="summary-section">
157                            <h3>⚠️ Issues & Concerns</h3>
158                            <ul>
159                                ${summary.issues.map(item => `<li>${item}</li>`).join('')}
160                            </ul>
161                        </div>
162                        
163                        <div class="summary-section">
164                            <h3>💡 Recommendations</h3>
165                            <ul>
166                                ${summary.recommendations.map(item => `<li>${item}</li>`).join('')}
167                            </ul>
168                        </div>
169                        
170                        <div class="summary-section">
171                            <h3>📈 Overall Sentiment</h3>
172                            <p class="sentiment-text">${summary.overallSentiment}</p>
173                        </div>
174                    </div>
175                </div>
176            </div>
177        `;
178        
179        // Add styles
180        const style = document.createElement('style');
181        style.textContent = `
182            #nps-summary-modal .modal-overlay {
183                position: fixed;
184                top: 0;
185                left: 0;
186                right: 0;
187                bottom: 0;
188                background: rgba(0, 0, 0, 0.7);
189                display: flex;
190                align-items: center;
191                justify-content: center;
192                z-index: 10001;
193                animation: fadeIn 0.3s ease;
194            }
195            
196            #nps-summary-modal .modal-content {
197                background: white;
198                border-radius: 12px;
199                max-width: 800px;
200                max-height: 90vh;
201                width: 90%;
202                overflow: hidden;
203                box-shadow: 0 20px 60px rgba(0,0,0,0.3);
204                animation: slideUp 0.3s ease;
205            }
206            
207            #nps-summary-modal .modal-header {
208                background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
209                color: white;
210                padding: 20px 30px;
211                display: flex;
212                justify-content: space-between;
213                align-items: center;
214            }
215            
216            #nps-summary-modal .modal-header h2 {
217                margin: 0;
218                font-size: 24px;
219                font-weight: 600;
220            }
221            
222            #nps-summary-modal .close-btn {
223                background: rgba(255,255,255,0.2);
224                border: none;
225                color: white;
226                font-size: 24px;
227                width: 36px;
228                height: 36px;
229                border-radius: 50%;
230                cursor: pointer;
231                display: flex;
232                align-items: center;
233                justify-content: center;
234                transition: background 0.2s;
235            }
236            
237            #nps-summary-modal .close-btn:hover {
238                background: rgba(255,255,255,0.3);
239            }
240            
241            #nps-summary-modal .modal-body {
242                padding: 30px;
243                overflow-y: auto;
244                max-height: calc(90vh - 80px);
245                font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
246            }
247            
248            #nps-summary-modal .summary-section {
249                margin-bottom: 30px;
250            }
251            
252            #nps-summary-modal .summary-section h3 {
253                color: #333;
254                font-size: 18px;
255                margin-bottom: 12px;
256                font-weight: 600;
257            }
258            
259            #nps-summary-modal .summary-section ul {
260                list-style: none;
261                padding: 0;
262                margin: 0;
263            }
264            
265            #nps-summary-modal .summary-section li {
266                background: #f8f9fa;
267                padding: 12px 16px;
268                margin-bottom: 8px;
269                border-radius: 6px;
270                border-left: 3px solid #667eea;
271                color: #333;
272                line-height: 1.5;
273            }
274            
275            #nps-summary-modal .sentiment-text {
276                background: #f8f9fa;
277                padding: 16px;
278                border-radius: 6px;
279                color: #333;
280                line-height: 1.6;
281                margin: 0;
282            }
283            
284            @keyframes fadeIn {
285                from { opacity: 0; }
286                to { opacity: 1; }
287            }
288            
289            @keyframes slideUp {
290                from { transform: translateY(30px); opacity: 0; }
291                to { transform: translateY(0); opacity: 1; }
292            }
293        `;
294        document.head.appendChild(style);
295        
296        document.body.appendChild(modal);
297        
298        // Close modal handlers
299        document.getElementById('close-summary-modal').addEventListener('click', () => {
300            modal.remove();
301        });
302        
303        modal.querySelector('.modal-overlay').addEventListener('click', (e) => {
304            if (e.target.classList.contains('modal-overlay')) {
305                modal.remove();
306            }
307        });
308    }
309
310    // Main function to summarize NPS chats
311    async function summarizeNPSChats() {
312        console.log('Starting NPS chat summarization...');
313        
314        try {
315            showLoadingIndicator();
316            
317            // Extract messages
318            const messages = extractMessages();
319            
320            if (messages.length === 0) {
321                alert('No messages found in the current channel. Please make sure you are in the NPS channel.');
322                hideLoadingIndicator();
323                return;
324            }
325            
326            console.log(`Analyzing ${messages.length} messages with AI...`);
327            
328            // Prepare messages for AI analysis
329            const messagesText = messages.map((msg, idx) => 
330                `[${idx + 1}] ${msg.sender} (${msg.timestamp}): ${msg.text}`
331            ).join('\n\n');
332            
333            // Call AI to analyze and summarize
334            const prompt = `You are analyzing NPS (Net Promoter Score) feedback from a Slack channel. Below are all the messages from the NPS channel. Please analyze them and provide a comprehensive summary.
335
336Messages:
337${messagesText}
338
339Please analyze these NPS chats and provide:
3401. Key themes that emerge from the feedback
3412. Positive feedback and what customers appreciate
3423. Issues and concerns raised by customers
3434. Actionable recommendations for improvement
3445. Overall sentiment analysis
345
346Focus on extracting actionable insights that can help improve the product/service.`;
347
348            const summary = await RM.aiCall(prompt, {
349                type: "json_schema",
350                json_schema: {
351                    name: "nps_summary",
352                    schema: {
353                        type: "object",
354                        properties: {
355                            keyThemes: {
356                                type: "array",
357                                items: { type: "string" },
358                                description: "Main themes identified in the NPS feedback"
359                            },
360                            positiveFeedback: {
361                                type: "array",
362                                items: { type: "string" },
363                                description: "Positive aspects mentioned by customers"
364                            },
365                            issues: {
366                                type: "array",
367                                items: { type: "string" },
368                                description: "Problems and concerns raised"
369                            },
370                            recommendations: {
371                                type: "array",
372                                items: { type: "string" },
373                                description: "Actionable recommendations for improvement"
374                            },
375                            overallSentiment: {
376                                type: "string",
377                                description: "Overall sentiment analysis of the feedback"
378                            }
379                        },
380                        required: ["keyThemes", "positiveFeedback", "issues", "recommendations", "overallSentiment"]
381                    }
382                }
383            });
384            
385            console.log('AI analysis complete:', summary);
386            
387            hideLoadingIndicator();
388            
389            // Display the summary
390            displaySummary(summary);
391            
392            // Store the summary for future reference
393            await GM.setValue('last_nps_summary', JSON.stringify({
394                summary: summary,
395                timestamp: new Date().toISOString(),
396                messageCount: messages.length
397            }));
398            
399        } catch (error) {
400            console.error('Error summarizing NPS chats:', error);
401            hideLoadingIndicator();
402            alert('Failed to summarize NPS chats. Error: ' + error.message);
403        }
404    }
405
406    // Create the summarize button
407    function createSummarizeButton() {
408        console.log('Creating summarize button...');
409        
410        // Remove existing button if any
411        const existingButton = document.getElementById('nps-summarize-btn');
412        if (existingButton) {
413            existingButton.remove();
414        }
415        
416        // Find the channel header to place the button
417        const channelHeader = document.querySelector('[data-qa="channel_header"]') ||
418                             document.querySelector('.p-ia__view_header') ||
419                             document.querySelector('.p-view_header');
420        
421        if (!channelHeader) {
422            console.log('Channel header not found, will retry...');
423            return false;
424        }
425        
426        const button = document.createElement('button');
427        button.id = 'nps-summarize-btn';
428        button.innerHTML = '📊 Summarize NPS Chats';
429        button.style.cssText = `
430            background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
431            color: white;
432            border: none;
433            padding: 8px 16px;
434            border-radius: 6px;
435            font-size: 14px;
436            font-weight: 600;
437            cursor: pointer;
438            margin-left: 12px;
439            transition: transform 0.2s, box-shadow 0.2s;
440            font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
441            box-shadow: 0 2px 8px rgba(102, 126, 234, 0.3);
442        `;
443        
444        button.addEventListener('mouseenter', () => {
445            button.style.transform = 'translateY(-2px)';
446            button.style.boxShadow = '0 4px 12px rgba(102, 126, 234, 0.4)';
447        });
448        
449        button.addEventListener('mouseleave', () => {
450            button.style.transform = 'translateY(0)';
451            button.style.boxShadow = '0 2px 8px rgba(102, 126, 234, 0.3)';
452        });
453        
454        button.addEventListener('click', summarizeNPSChats);
455        
456        // Insert the button into the header
457        const headerActions = channelHeader.querySelector('.p-ia__view_header__actions') ||
458                             channelHeader.querySelector('[data-qa="channel_header_actions"]') ||
459                             channelHeader;
460        
461        headerActions.appendChild(button);
462        console.log('Summarize button created successfully');
463        return true;
464    }
465
466    // Initialize the extension
467    function init() {
468        console.log('Initializing NPS Chat Summarizer...');
469        
470        // Wait for the page to be fully loaded
471        if (document.readyState === 'loading') {
472            document.addEventListener('DOMContentLoaded', init);
473            return;
474        }
475        
476        // Try to create the button
477        const buttonCreated = createSummarizeButton();
478        
479        if (!buttonCreated) {
480            // If button creation failed, retry after a delay
481            setTimeout(init, 2000);
482            return;
483        }
484        
485        // Watch for navigation changes in Slack (SPA)
486        const observer = new MutationObserver(debounce(() => {
487            const existingButton = document.getElementById('nps-summarize-btn');
488            if (!existingButton) {
489                console.log('Button disappeared, recreating...');
490                createSummarizeButton();
491            }
492        }, 1000));
493        
494        observer.observe(document.body, {
495            childList: true,
496            subtree: true
497        });
498        
499        console.log('NPS Chat Summarizer initialized successfully');
500    }
501
502    // Start the extension
503    init();
504})();
Slack NPS Chat Summarizer | Robomonkey