// ==UserScript== // @name 在线网页聊天室 // @namespace http://tampermonkey.net/ // @version 3.3 // @description Supabase realtime chat 基于Supabase的跨网页聊天室(优化版) // @match https://www.guozaoke.com/* // @match https://juejin.cn/* // @match https://*/* // @grant GM_xmlhttpRequest // @grant GM_getValue // @grant GM_setValue // @run-at document-start // @connect supabase.co // @require https://unpkg.com/@supabase/supabase-js@2.49.3/dist/umd/supabase.js // ==/UserScript== (function() { 'use strict'; // 配置参数 const CONFIG = { SUPABASE_URL: 'https://icaugjyuwenraxxgwvzf.supabase.co', SUPABASE_KEY: 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6ImljYXVnanl1d2VucmF4eGd3dnpmIiwicm9sZSI6ImFub24iLCJpYXQiOjE3NDI4ODcwNjcsImV4cCI6MjA1ODQ2MzA2N30.-IsrU3_NyoqDxFeNH1l2d6SgVv9pPA0uIVEA44FmuSQ', CHAT_UI: { width: 320, height: 420, position: { right: '20px', bottom: '20px' }, theme: { primary: '#007FFF', background: '#1A1A1A', text: '#FFFFFF', inputBg: '#2D2D2D' } } }; // 样式管理模块 const StyleManager = (() => { const cssVariables = ` :root { --chat-bg: ${CONFIG.CHAT_UI.theme.background}; --chat-text: ${CONFIG.CHAT_UI.theme.text}; --primary-color: ${CONFIG.CHAT_UI.theme.primary}; --input-bg: ${CONFIG.CHAT_UI.theme.inputBg}; } `; const scrollbarCSS = ` #chat-messages::-webkit-scrollbar { width: 8px; background: transparent; } #chat-messages::-webkit-scrollbar-thumb { background: #5c5c5c; border-radius: 4px; } #chat-messages { scrollbar-width: thin; scrollbar-color: #5c5c5c #2d2d2d; } `; return { inject: () => { const style = document.createElement('style'); style.textContent = `${cssVariables} ${scrollbarCSS}`; document.head.appendChild(style); } }; })(); // Supabase 客户端管理 const SupabaseClient = (() => { let client; return { initialize: async () => { client = window.supabase.createClient( CONFIG.SUPABASE_URL, CONFIG.SUPABASE_KEY, { realtime: { params: { eventsPerSecond: 10 } } } ); return client; }, getClient: () => client }; })(); // 用户管理模块 const UserManager = { getOrCreateId: () => { const USER_NAME = document.querySelector("div.username")?.innerText; if(USER_NAME){ GM_setValue('chat_user_id', USER_NAME); return USER_NAME; } let userId = GM_getValue('chat_user_id'); if (!userId) { userId = USER_NAME || `user_${crypto.randomUUID().slice(0, 8)}`; GM_setValue('chat_user_id', userId); } return userId; } }; // 聊天室核心功能 class ChatRoom { constructor(supabase) { this.supabase = supabase; this.userId = UserManager.getOrCreateId(); this.lastSendTime = 0; this.initUI(); this.setupRealtime(); this.loadHistory(); this.loadUserInfo(); } initUI() { this.container = document.createElement('div'); this.container.id = 'chat-container'; Object.assign(this.container.style, { position: 'fixed', right: CONFIG.CHAT_UI.position.right, bottom: CONFIG.CHAT_UI.position.bottom, width: `${CONFIG.CHAT_UI.width}px`, height: `${CONFIG.CHAT_UI.height}px`, backgroundColor: 'var(--chat-bg)', borderRadius: '12px', boxShadow: '0 4px 16px rgba(0,0,0,0.2)', zIndex: 9999 }); this.messageArea = this.createMessageArea(); this.input = this.createInput(); this.sendButton = this.createButton(); this.container.append(this.messageArea, this.input, this.sendButton); document.body.appendChild(this.container); } createMessageArea() { const div = document.createElement('div'); Object.assign(div.style, { height: '320px', padding: '16px', overflowY: 'auto', color: 'var(--chat-text)' }); div.id = 'chat-messages'; return div; } createInput() { const input = document.createElement('textarea'); Object.assign(input.style, { width: 'calc(100% - 32px)', height: '40px', margin: '8px 16px', padding: '8px 12px', backgroundColor: 'var(--input-bg)', border: '1px solid #3A3A3A', borderRadius: '8px', color: 'var(--chat-text)', resize: 'none' }); input.placeholder = '输入消息(Ctrl + Enter 发送)'; return input; } createButton() { const button = document.createElement('button'); Object.assign(button.style, { position: 'absolute', right: '16px', bottom: '16px', padding: '8px 20px', backgroundColor: 'var(--primary-color)', color: '#FFF', border: 'none', borderRadius: '8px', cursor: 'pointer' }); button.textContent = '发送'; return button; } async setupRealtime() { this.channel = this.supabase.channel(location.host) //'realtime-chat .on('postgres_changes', { event: 'INSERT', schema: 'public', table: 'messages' }, payload => this.addMessage(payload.new)) .subscribe(); } addMessage(message) { //if (message.domain !== location.host) return; // 过滤非法消息 const isOwn = message.user_id === this.userId; const msgElement = document.createElement('div'); msgElement.innerHTML = `