├── .gitignore ├── extension ├── icons │ ├── icon_128.png │ ├── icon_16.png │ ├── icon_32.png │ └── icon_48.png ├── background.js ├── popup.js ├── manifest.json ├── popup.html ├── style.css ├── content.js ├── jquery-ui.min.css └── jquery-3.7.1.min.js ├── LICENSE ├── CONTRIBUTING.md └── README.md /.gitignore: -------------------------------------------------------------------------------- 1 | extension/p.py 2 | .cursor/rules 3 | -------------------------------------------------------------------------------- /extension/icons/icon_128.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Saarthakkj/chat-navigator/HEAD/extension/icons/icon_128.png -------------------------------------------------------------------------------- /extension/icons/icon_16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Saarthakkj/chat-navigator/HEAD/extension/icons/icon_16.png -------------------------------------------------------------------------------- /extension/icons/icon_32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Saarthakkj/chat-navigator/HEAD/extension/icons/icon_32.png -------------------------------------------------------------------------------- /extension/icons/icon_48.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Saarthakkj/chat-navigator/HEAD/extension/icons/icon_48.png -------------------------------------------------------------------------------- /extension/background.js: -------------------------------------------------------------------------------- 1 | // This file is intentionally left blank or for future background tasks not related to URL detection. 2 | console.log('[Background] Script loaded. Currently minimal, URL detection moved to content.js'); 3 | -------------------------------------------------------------------------------- /extension/popup.js: -------------------------------------------------------------------------------- 1 | document.addEventListener('DOMContentLoaded', function() { 2 | const themeSelector = document.getElementById('themeSelector'); 3 | const sidebar = document.getElementById('chat'); 4 | 5 | // Load saved theme and set initial state 6 | chrome.storage.sync.get('theme', function(data) { 7 | const savedTheme = data.theme || 'red'; // Default to red 8 | themeSelector.value = savedTheme; 9 | if (sidebar) { 10 | sidebar.className = 'sidebar theme-' + savedTheme; 11 | } 12 | }); 13 | 14 | // Theme switching 15 | themeSelector.addEventListener('change', function(e) { 16 | const selectedTheme = e.target.value; 17 | // Save the theme to storage 18 | chrome.storage.sync.set({theme: selectedTheme}, function() { 19 | console.log('Theme saved:', selectedTheme); 20 | }); 21 | // Update popup sidebar 22 | if (sidebar) { 23 | sidebar.className = 'sidebar theme-' + selectedTheme; 24 | } 25 | }); 26 | }); -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2024 Chat Navigator 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. -------------------------------------------------------------------------------- /extension/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "manifest_version": 3, 3 | "name": "Chat Navigtator", 4 | "version": "1.0", 5 | "description": "Helps you navigate through your chat history with ease.", 6 | 7 | "permissions": [ 8 | "tabs", 9 | "storage" 10 | ], 11 | 12 | "host_permissions": [ 13 | "https://*.google.com/*", 14 | "https://gemini.google.com/*", 15 | "https://grok.com/*" 16 | ], 17 | "content_scripts": [ 18 | { 19 | "matches": [ 20 | "https://chat.openai.com/*", 21 | "https://chatgpt.com/*", 22 | "https://claude.ai/*", 23 | "https://*.perplexity.ai/*", 24 | "https://t3.chat/*", 25 | "https://gemini.google.com/*", 26 | "https://grok.com/*" 27 | ], 28 | "css": ["style.css" , "jquery-ui.min.css"], 29 | "js": [ 30 | "jquery-3.7.1.min.js", 31 | "jquery-ui.min.js", 32 | "content.js" 33 | ], 34 | "run_at": "document_idle" 35 | } 36 | ], 37 | "action": { 38 | "default_popup": "popup.html", 39 | "default_icon": { 40 | "16": "icons/icon_16.png", 41 | "48": "icons/icon_32.png", 42 | "128": "icons/icon_128.png" 43 | }, 44 | "default_title": "My Page Modifier" 45 | }, 46 | 47 | "background": { 48 | "service_worker": "background.js" 49 | }, 50 | "icons": { 51 | "16": "icons/icon_16.png", 52 | "48": "icons/icon_32.png", 53 | "128": "icons/icon_128.png" 54 | } 55 | } 56 | -------------------------------------------------------------------------------- /extension/popup.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | Chat Navigator 7 | 8 | 9 | 21 |
34 |

35 | Chat Navigator 36 |

37 | 38 | 50 | 79 |
80 | 81 | 82 | 83 | 84 | -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | # Contributing to Chat Navigator 2 | 3 | Thank you for your interest in contributing to Chat Navigator! This document provides guidelines and instructions for contributing to this project. 4 | 5 | ## Code of Conduct 6 | 7 | By participating in this project, you agree to maintain a respectful and inclusive environment for everyone. 8 | 9 | ## How to Contribute 10 | 11 | ### Reporting Bugs 12 | 13 | 1. Check if the bug has already been reported in the Issues section 14 | 2. If not, create a new issue with a clear and descriptive title 15 | 3. Include as much relevant information as possible: 16 | - Steps to reproduce the bug 17 | - Expected behavior 18 | - Actual behavior 19 | - Screenshots if applicable 20 | - Browser version and OS information 21 | 22 | ### Suggesting Features 23 | 24 | 1. Check if the feature has already been suggested in the Issues section 25 | 2. If not, create a new issue with a clear and descriptive title 26 | 3. Provide a detailed description of the feature 27 | 4. Explain why this feature would be useful 28 | 5. Include any relevant examples or mockups 29 | 30 | ### Pull Requests 31 | 32 | 1. Fork the repository 33 | 2. Create a new branch for your feature/fix: 34 | ```bash 35 | git checkout -b feature/amazing-feature 36 | ``` 37 | 3. Make your changes 38 | 4. Commit your changes with clear, descriptive commit messages: 39 | ```bash 40 | git commit -m 'Add some amazing feature' 41 | ``` 42 | 5. Push to your fork: 43 | ```bash 44 | git push origin feature/amazing-feature 45 | ``` 46 | 6. Open a Pull Request 47 | 48 | ### Development Setup 49 | 50 | 1. Clone the repository 51 | 2. Install dependencies (if any) 52 | 3. Make your changes 53 | 4. Test your changes thoroughly 54 | 5. Ensure your code follows the project's style guidelines 55 | 56 | ### Code Style 57 | 58 | - Follow the existing code style in the project 59 | - Use meaningful variable and function names 60 | - Add comments for complex logic 61 | - Keep functions small and focused 62 | - Write clear commit messages 63 | 64 | ### Testing 65 | 66 | - Test your changes across different browsers 67 | - Ensure the extension works on all supported platforms 68 | - Test edge cases and error scenarios 69 | - Update tests if you're modifying existing functionality 70 | 71 | ## Review Process 72 | 73 | 1. All pull requests will be reviewed by maintainers 74 | 2. We may request changes or improvements 75 | 3. Once approved, your changes will be merged into the main branch 76 | 77 | ## Questions? 78 | 79 | If you have any questions about contributing, feel free to open an issue or contact the maintainers. 80 | 81 | Thank you for contributing to Chat Navigator! -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Chat Navigator 2 | 3 | ## Overview 4 | Chat Navigator is a browser extension designed to enhance the user experience of LLM chat applications by providing an interactive navigation sidebar. The extension automatically indexes conversation messages and enables quick navigation through long chat histories. 5 | 6 | ## Features 7 | - Real-time message indexing and navigation 8 | - Movable sidebar interface 9 | - Theme customization with multiple color schemes 10 | - Automatic detection of new messages 11 | - Direct message linking and navigation 12 | - Cross-platform compatibility with major LLM chat applications 13 | 14 | ## Technical Implementation 15 | - Built as a Chrome extension using Manifest V3 16 | - Utilizes MutationObserver for real-time message detection 17 | - Implements drag-and-drop functionality for sidebar positioning 18 | - Features a modular theme system with CSS variables 19 | - Persists user preferences using Chrome Storage API 20 | 21 | ## Supported Platforms 22 | - ChatGPT (chat.openai.com) 23 | - Perplexity AI (perplexity.ai) 24 | 25 | ## Installation 26 | 1. Clone the repository 27 | 2. Open Chrome and navigate to `chrome://extensions/` 28 | 3. Enable Developer mode 29 | 4. Click "Load unpacked" and select the extension directory 30 | 5. The extension will automatically activate on supported chat applications 31 | 32 | ## Usage 33 | 1. Visit any supported LLM chat application 34 | 2. The navigation sidebar will appear on the right side of the screen 35 | 3. Click any indexed message to navigate directly to that point in the conversation 36 | 4. Drag the sidebar to reposition it anywhere on the screen 37 | 5. Use the extension popup to customize the theme 38 | 39 | ## Development 40 | The extension consists of several key components: 41 | - `content.js`: Main extension logic and DOM manipulation 42 | - `popup.html/js`: Theme selection interface 43 | - `style.css`: Theme definitions and UI styling 44 | - `manifest.json`: Extension configuration and permissions 45 | 46 | ## Future Enhancements 47 | - Message search functionality 48 | - Custom message annotations 49 | - Support for additional chat platforms 50 | 51 | ## License 52 | This project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details. 53 | 54 | ## Contributing 55 | Contributions are welcome! Please feel free to submit a Pull Request. For major changes, please open an issue first to discuss what you would like to change. 56 | 57 | 1. Fork the repository 58 | 2. Create your feature branch (`git checkout -b feature/amazing-feature`) 59 | 3. Commit your changes (`git commit -m 'Add some amazing feature'`) 60 | 4. Push to the branch (`git push origin feature/amazing-feature`) 61 | 5. Open a Pull Request 62 | 63 | Please make sure to update tests as appropriate and follow the existing code style. 64 | -------------------------------------------------------------------------------- /extension/style.css: -------------------------------------------------------------------------------- 1 | /* Base styles and CSS variables for theming */ 2 | :root { 3 | /* Default theme variables */ 4 | --primary-color: #52ff5e; 5 | --secondary-color: #ffcdd2; 6 | --text-color: #ffffff; 7 | --hover-color: rgba(255, 255, 255, 0.1); 8 | --shadow-color: rgba(0, 0, 0, 0.1); 9 | --primary-color-rgb: 82, 255, 94; 10 | } 11 | 12 | /* Theme classes with v0.dev-inspired adjustments */ 13 | .theme-red { 14 | --primary-color: #ff1744; /* Brighter, more vibrant red */ 15 | --secondary-color: #ff5252; 16 | --text-color: #ffffff; 17 | --primary-color-rgb: 255, 23, 68; 18 | } 19 | 20 | .theme-blue { 21 | --primary-color: #2979ff; /* Vibrant techno blue */ 22 | --secondary-color: #448aff; 23 | --text-color: #ffffff; 24 | --primary-color-rgb: 41, 121, 255; 25 | } 26 | 27 | .theme-pink { 28 | --primary-color: #f50057; /* Bold, neon-like pink */ 29 | --secondary-color: #ec407a; 30 | --text-color: #ffffff; 31 | --primary-color-rgb: 245, 0, 87; 32 | } 33 | 34 | .theme-white { 35 | --primary-color: #fafafa; /* Clean off-white */ 36 | --secondary-color: #eeeeee; 37 | --text-color: #212121; /* Dark text for contrast */ 38 | --primary-color-rgb: 250, 250, 250; 39 | } 40 | 41 | .theme-gray { 42 | --primary-color: #757575; /* Modern muted gray */ 43 | --secondary-color: #bdbdbd; 44 | --text-color: #ffffff; 45 | --primary-color-rgb: 117, 117, 117; 46 | } 47 | 48 | .theme-dark { 49 | --primary-color: #000000; /* Pure black for v0.dev dark vibe */ 50 | --secondary-color: #212121; 51 | --text-color: #e0e0e0; 52 | --primary-color-rgb: 0, 0, 0; 53 | } 54 | 55 | /* Navigator Bar Styles */ 56 | .navigator-bar { 57 | position: fixed; 58 | right: 10px; 59 | top: 20%; 60 | width: 8px; 61 | max-height: 60vh; 62 | background-color: var(--primary-color); 63 | border-radius: 4px; 64 | z-index: 9999; 65 | display: flex; 66 | flex-direction: column; 67 | gap: 4px; 68 | padding: 4px; 69 | box-shadow: 0 2px 10px var(--shadow-color); 70 | transition: all 0.3s ease-in-out; 71 | overflow: hidden; 72 | } 73 | 74 | /* Expanded state for navigator bar */ 75 | .navigator-bar.expanded { 76 | width: 220px; 77 | background-color: var(--primary-color); 78 | border-radius: 6px; 79 | padding: 8px 4px; 80 | } 81 | 82 | /* Header for the navigator bar */ 83 | .nav-header { 84 | display: none; 85 | padding: 4px 8px; 86 | margin-bottom: 8px; 87 | border-bottom: 1px solid var(--secondary-color); 88 | } 89 | 90 | .navigator-bar.expanded .nav-header { 91 | display: block; 92 | } 93 | 94 | .nav-title { 95 | color: var(--text-color); 96 | font-size: 14px; 97 | font-weight: bold; 98 | white-space: nowrap; 99 | } 100 | 101 | /* Nav items */ 102 | .nav-item { 103 | height: 8px; 104 | width: 100%; 105 | background-color: var(--secondary-color); 106 | border-radius: 2px; 107 | cursor: pointer; 108 | transition: all 0.2s ease; 109 | position: relative; 110 | margin: 2px 0; 111 | } 112 | 113 | .navigator-bar.expanded .nav-item { 114 | height: auto; 115 | background-color: transparent; 116 | display: flex; 117 | align-items: center; 118 | border-radius: 4px; 119 | padding: 6px 4px; 120 | } 121 | 122 | .navigator-bar.expanded .nav-item:hover, 123 | .navigator-bar.expanded .nav-item.active { 124 | background-color: var(--secondary-color); 125 | } 126 | 127 | .nav-item-content { 128 | display: flex; 129 | align-items: center; 130 | width: 100%; 131 | } 132 | 133 | .nav-item:hover { 134 | transform: scaleX(1.2); 135 | background-color: var(--text-color); 136 | } 137 | 138 | .navigator-bar.expanded .nav-item:hover { 139 | transform: none; 140 | background-color: var(--secondary-color); 141 | } 142 | 143 | /* Nav item text only shows on expanded */ 144 | .nav-item-text { 145 | display: none; 146 | color: var(--text-color); 147 | font-size: 12px; 148 | white-space: nowrap; 149 | overflow: hidden; 150 | text-overflow: ellipsis; 151 | margin-left: 4px; 152 | } 153 | 154 | .navigator-bar.expanded .nav-item-text { 155 | display: block; 156 | max-width: 200px; 157 | } 158 | 159 | /* Active state for nav items */ 160 | .nav-item.active { 161 | background-color: var(--text-color); 162 | } 163 | 164 | .navigator-bar.expanded .nav-item.active { 165 | background-color: var(--secondary-color); 166 | font-weight: bold; 167 | } 168 | 169 | .navigator-bar.expanded .nav-item.active .nav-item-text { 170 | font-weight: bold; 171 | } 172 | 173 | /* Expanded Sidebar Styles */ 174 | #chat.sidebar { 175 | width: max-content; 176 | min-width: 200px; 177 | min-height: max-content; 178 | max-height: 60vh; 179 | background-color: var(--primary-color); 180 | color: var(--text-color); 181 | position: fixed; 182 | z-index: 9999; 183 | box-shadow: 0 4px 12px var(--shadow-color); 184 | border-radius: 8px; 185 | overflow: auto; 186 | display: table; 187 | resize: both; /* Enables resizing in both directions */ 188 | border-collapse: separate; /* Changed from collapse to separate */ 189 | border-spacing: 0; 190 | font-size: 14px; 191 | transition: opacity 0.3s ease; 192 | } 193 | 194 | /* Table-specific styling with v0.dev touch */ 195 | #chat.sidebar table { 196 | width: 100%; 197 | border-collapse: collapse; 198 | background: rgba(255, 255, 255, 0.1); 199 | } 200 | 201 | #chat.sidebar tr:nth-child(even) td { 202 | background-color: rgba(255, 255, 255, 0.05); /* Zebra striping for readability */ 203 | } 204 | 205 | #chat.sidebar td { 206 | padding: 0.5em; 207 | border-bottom: 1px solid var(--secondary-color); 208 | } 209 | 210 | #chat.sidebar td a { 211 | color: var(--text-color); 212 | text-decoration: none; 213 | display: block; 214 | padding: 4px; 215 | border-radius: 4px; 216 | transition: background-color 0.2s ease; 217 | } 218 | 219 | #chat.sidebar td a:hover { 220 | background-color: var(--secondary-color); 221 | } 222 | 223 | /* Message tooltip */ 224 | .message-tooltip { 225 | position: fixed; 226 | max-width: 300px; 227 | padding: 10px; 228 | background-color: var(--primary-color); 229 | color: var(--text-color); 230 | border-radius: 6px; 231 | z-index: 10000; 232 | display: none; 233 | box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2); 234 | font-size: 14px; 235 | max-height: 200px; 236 | overflow-y: auto; 237 | } 238 | 239 | /* Message highlight effect - improved */ 240 | .highlight-message { 241 | animation: highlight-pulse 2s ease-in-out; 242 | position: relative; 243 | outline: 2px solid var(--primary-color); 244 | box-shadow: 0 0 10px var(--primary-color); 245 | } 246 | 247 | @keyframes highlight-pulse { 248 | 0%, 100% { 249 | background-color: transparent; 250 | outline-color: transparent; 251 | } 252 | 50% { 253 | background-color: rgba(var(--primary-color-rgb, 255, 23, 68), 0.2); 254 | outline-color: var(--primary-color); 255 | } 256 | } 257 | 258 | .macos-controls { 259 | display: flex; 260 | gap: 0.5rem; 261 | padding: 0.5rem; 262 | } 263 | .macos-controls button { 264 | width: 0.75rem; 265 | height: 0.75rem; 266 | border-radius: 50%; 267 | display: inline-block; 268 | } 269 | .close { 270 | background: #ff5f56; 271 | } 272 | .minimize { 273 | background: #ffbd2e; 274 | } 275 | .maximize { 276 | background: #27c93f; 277 | } 278 | 279 | .minimized { 280 | height: 40px !important; /* Only header visible */ 281 | min-height: 0 !important; 282 | max-height: 40px !important; 283 | overflow: hidden; 284 | } 285 | 286 | .sidebar-header { 287 | max-height: 90px; 288 | padding: 8px; 289 | cursor: move; 290 | background-color: var(--secondary-color); 291 | border-radius: 8px 8px 0 0; 292 | user-select: none; /* Prevent text selection while dragging */ 293 | } 294 | 295 | #chat.sidebar.minimized tr:not(:first-child) { 296 | display: none; 297 | } 298 | -------------------------------------------------------------------------------- /extension/content.js: -------------------------------------------------------------------------------- 1 | //! TODO : 2 | //! 1. Fix the initial random animation [done] 3 | //! 2. Make it support all websites 4 | 5 | //? claudes-chat-element : *
/ 6 | 7 | console.log("[Content] content.js loaded"); 8 | 9 | // Global variable for chat sidebar and navigator bar 10 | let chat_sidebar; 11 | let navigator_bar; 12 | let currentChatConfig = null; 13 | let lastUrl = window.location.href; 14 | var index = 1; 15 | let messageIndexMap = new Map(); // Store message references by index 16 | 17 | // Define selectors for different chat applications directly in content.js 18 | const chatSelectors = { 19 | 'chatgpt.com': { 20 | type: 'openai', 21 | selectors: { 22 | chatContainer: 'article', // Parent of messages 23 | messageSelector: '.whitespace-pre-wrap' // Individual message elements 24 | } 25 | }, 26 | 'claude.ai': { 27 | type: 'claude', 28 | selectors: { 29 | chatContainer: 'font-user-message grid grid-cols-1 gap-2 py-0.5 text-[0.9375rem] leading-6', // More specific parent for Claude messages 30 | messageSelector: '.whitespace-pre-wrap break-words', // Messages from Claude 31 | } 32 | }, 33 | 'perplexity.ai': { 34 | type: 'perplexity', 35 | selectors: { 36 | chatContainer: 'main', 37 | messageSelector: 'div.prose' // General prose content for Perplexity 38 | } 39 | }, 40 | 'gemini.google.com': { 41 | type: 'gemini', 42 | selectors: { 43 | chatContainer: 'message-list', // Custom element or specific class for Gemini 44 | messageSelector: 'message-content' // Selector for actual message text 45 | } 46 | }, 47 | 'grok.com': { 48 | type: 'grok', 49 | selectors: { 50 | // These are guesses and might need refinement for Grok 51 | chatContainer: 'div[data-testid="conversation"] > div > div[style*="flex-direction: column;"]', 52 | messageSelector: 'div[data-testid^="message-"]' 53 | } 54 | }, 55 | 't3.chat': { 56 | type: 't3chat', 57 | selectors: { 58 | // These are guesses for T3 Chat 59 | chatContainer: '.chat-messages-container', 60 | messageSelector: '.message-bubble' 61 | } 62 | } 63 | }; 64 | 65 | // Function to determine which chat application is being used based on URL 66 | function determineChatAppConfig(url) { 67 | console.log('[Content] Determining chat app for URL:', url); 68 | if (!url) { 69 | console.warn('[Content] No URL provided to determineChatAppConfig'); 70 | return null; 71 | } 72 | for (const [domain, config] of Object.entries(chatSelectors)) { 73 | if (url.includes(domain)) { 74 | console.log('[Content] Found matching chat app:', config.type); 75 | return config; 76 | } 77 | } 78 | console.log('[Content] No matching chat app found for URL:', url); 79 | return null; 80 | } 81 | 82 | // Function to apply theme 83 | function applyTheme(theme) { 84 | if (navigator_bar) { 85 | navigator_bar.className = `navigator-bar theme-${theme}`; 86 | console.log("[Content] Theme applied to navigator bar:", theme); 87 | } 88 | 89 | if (chat_sidebar) { 90 | chat_sidebar.className = `sidebar theme-${theme}`; 91 | console.log("[Content] Theme applied to sidebar:", theme); 92 | } 93 | } 94 | 95 | async function waitforelement(selector) { 96 | console.log("[Content] Waiting for element with selector:", selector); 97 | try { 98 | return new Promise((resolve) => { 99 | const checkElements = () => { 100 | let ele = document.querySelector(selector); 101 | if (ele) { 102 | ele = ele.parentNode; 103 | console.log("[Content] Element found:", selector); 104 | resolve(ele); 105 | } else { 106 | console.log("[Content] {else} Element not found:", selector); 107 | setTimeout(checkElements, 500); 108 | } 109 | }; 110 | checkElements(); 111 | }); 112 | } catch (e) { 113 | console.error("[Content] Error executing waitforelements:", e); 114 | } 115 | } 116 | 117 | // Function to initialize chat navigation system 118 | async function initializeChatNavigation() { 119 | console.log("[Content] Initializing chat navigation for URL:", window.location.href); 120 | currentChatConfig = determineChatAppConfig(window.location.href); 121 | 122 | if (currentChatConfig) { 123 | console.log("[Content] Determined chat config:", currentChatConfig); 124 | await setupNavigator(); 125 | } else { 126 | console.warn("[Content] No chat configuration found for this URL. Navigation not initialized."); 127 | // Optionally, remove any existing elements 128 | if (navigator_bar) { 129 | navigator_bar.remove(); 130 | navigator_bar = null; 131 | } 132 | if (chat_sidebar) { 133 | chat_sidebar.remove(); 134 | chat_sidebar = null; 135 | } 136 | console.log("[Content] Removed existing navigation elements as no config for current URL."); 137 | } 138 | } 139 | 140 | // Function to setup the navigator bar and sidebar 141 | async function setupNavigator() { 142 | console.log("[Content] Setting up chat navigator with config:", currentChatConfig); 143 | if (!currentChatConfig || !currentChatConfig.selectors || !currentChatConfig.selectors.chatContainer) { 144 | console.warn("[Content] Invalid or incomplete chat config, skipping setup. Config:", currentChatConfig); 145 | return; 146 | } 147 | 148 | try { 149 | // Remove any existing elements 150 | if (document.querySelector("#chat-navigator-bar")) { 151 | document.querySelector("#chat-navigator-bar").remove(); 152 | } 153 | if (document.querySelector("#chat.sidebar")) { 154 | document.querySelector("#chat.sidebar").remove(); 155 | } 156 | 157 | console.log("[Content] {try} currentChatConfig.selectors.chatContainer:", currentChatConfig.selectors.chatContainer); 158 | const targetElementParent = await waitforelement(currentChatConfig.selectors.chatContainer); 159 | console.log("[Content] Found target element:", targetElementParent); 160 | 161 | // Clear index map 162 | messageIndexMap.clear(); 163 | index = 1; 164 | 165 | // Create navigator bar (collapsed state) 166 | navigator_bar = document.createElement("div"); 167 | navigator_bar.id = "chat-navigator-bar"; 168 | navigator_bar.className = 'navigator-bar'; 169 | 170 | // Add header to navigator bar 171 | const navHeader = document.createElement("div"); 172 | navHeader.className = "nav-header"; 173 | navHeader.innerHTML = `Chat Messages`; 174 | navigator_bar.appendChild(navHeader); 175 | 176 | // Create sidebar (expanded state - initially hidden) 177 | chat_sidebar = document.createElement("table"); 178 | chat_sidebar.id = "chat"; 179 | chat_sidebar.className = 'sidebar'; 180 | chat_sidebar.style.display = 'none'; // Hidden initially 181 | chat_sidebar.style.position = 'absolute'; 182 | 183 | // Create header row for dragging 184 | const headerRow = document.createElement("tr"); 185 | const headerCell = document.createElement("td"); 186 | headerCell.className = "sidebar-header"; 187 | 188 | // Add macOS controls inside header 189 | headerCell.innerHTML = ` 190 |
191 | 192 | 193 | 194 |
195 | `; 196 | 197 | headerCell.colSpan = 1; 198 | headerRow.appendChild(headerCell); 199 | chat_sidebar.appendChild(headerRow); 200 | 201 | // Make the sidebar draggable by its header 202 | dragElement(chat_sidebar); 203 | 204 | document.body.appendChild(navigator_bar); 205 | document.body.appendChild(chat_sidebar); 206 | 207 | // Hook up sidebar controls 208 | const closeBtn = chat_sidebar.querySelector(".close"); 209 | const minimizeBtn = chat_sidebar.querySelector(".minimize"); 210 | const maximizeBtn = chat_sidebar.querySelector(".maximize"); 211 | 212 | closeBtn.addEventListener("click", () => { 213 | chat_sidebar.style.display = "none"; 214 | console.log("[Content] Chat sidebar closed"); 215 | }); 216 | minimizeBtn.addEventListener("click", () => { 217 | chat_sidebar.classList.add("minimized"); 218 | console.log("[Content] Chat sidebar minimized"); 219 | }); 220 | maximizeBtn.addEventListener("click", () => { 221 | chat_sidebar.classList.remove("minimized"); 222 | console.log("[Content] Chat sidebar maximized"); 223 | }); 224 | 225 | // Setup jQuery resizable for sidebar 226 | $(function() { 227 | $("#chat").resizable({ 228 | handles: "n, e, s, w, ne, se, sw, nw", 229 | start: function() { 230 | resizeObserver.observe(chat_sidebar); 231 | }, 232 | stop: function() { 233 | resizeObserver.disconnect(); 234 | } 235 | }); 236 | }); 237 | 238 | // Create but don't connect the observer initially 239 | const resizeObserver = new ResizeObserver(() => { 240 | if (!chat_sidebar) return; 241 | const width = chat_sidebar.offsetWidth; 242 | const height = chat_sidebar.offsetHeight; 243 | const fontSize = Math.min(width, height) / 10; 244 | chat_sidebar.style.fontSize = `${fontSize}px`; 245 | }); 246 | 247 | // Apply theme 248 | chrome.storage.sync.get('theme', function(data) { 249 | const theme = data.theme || 'red'; 250 | applyTheme(theme); 251 | }); 252 | 253 | // Setup the hover behavior for navigator bar - show expanded text labels 254 | navigator_bar.addEventListener('mouseenter', () => { 255 | navigator_bar.classList.add('expanded'); 256 | 257 | // If user hovers the bar but doesn't have sidebar open yet, still position it 258 | const rect = navigator_bar.getBoundingClientRect(); 259 | chat_sidebar.style.top = `${rect.top}px`; 260 | chat_sidebar.style.left = `${rect.right + 10}px`; // Position to the right of the bar 261 | chat_sidebar.style.display = "table"; 262 | }); 263 | 264 | navigator_bar.addEventListener('mouseleave', (e) => { 265 | if (e.relatedTarget !== chat_sidebar && !chat_sidebar.contains(e.relatedTarget)) { 266 | navigator_bar.classList.remove('expanded'); 267 | setTimeout(() => { 268 | if (!chat_sidebar.matches(':hover')) { 269 | chat_sidebar.style.display = "none"; 270 | } 271 | }, 100); 272 | } 273 | }); 274 | 275 | // Allow sidebar to close when mouse leaves it 276 | chat_sidebar.addEventListener('mouseleave', (e) => { 277 | if (e.relatedTarget !== navigator_bar) { 278 | setTimeout(() => { 279 | if (!navigator_bar.matches(':hover')) { 280 | chat_sidebar.style.display = "none"; 281 | navigator_bar.classList.remove('expanded'); 282 | } 283 | }, 100); 284 | } 285 | }); 286 | 287 | // Handle existing chats 288 | const existingMessages = targetElementParent.querySelectorAll(currentChatConfig.selectors.messageSelector); 289 | console.log(`[Content] Processing ${existingMessages.length} existing chat messages using selector: ${currentChatConfig.selectors.messageSelector}`); 290 | 291 | existingMessages.forEach((messageNode) => { 292 | try { 293 | messageNode.id = `index-${index}`; 294 | messageIndexMap.set(index, messageNode); 295 | const shortText = messageNode.textContent.substring(0, 12).trim() + "..."; 296 | addMessageToSidebar(index, shortText, messageNode.id); 297 | addMessageToBar(index); 298 | index++; 299 | } catch (e) { 300 | console.error("[Content] Error processing existing message:", e, "Message content:", messageNode.innerHTML); 301 | } 302 | }); 303 | 304 | // Observe new chats within the chatContainer (targetElementParent) 305 | const observerConfig = { childList: true, subtree: true }; 306 | const obs = new MutationObserver(mutationCallback); 307 | try { 308 | obs.observe(targetElementParent, observerConfig); 309 | console.log("[Content] MutationObserver initialized for target element:", targetElementParent); 310 | } catch (e) { 311 | console.error("[Content] Observer initialization failed, error:", e); 312 | } 313 | } catch (error) { 314 | console.error("[Content] Error in setupNavigator:", error); 315 | } 316 | } 317 | 318 | // Function to add a message indicator to the navigator bar 319 | function addMessageToBar(messageIndex) { 320 | if (!navigator_bar) return; 321 | 322 | const barItem = document.createElement("div"); 323 | barItem.className = "nav-item"; 324 | barItem.dataset.index = messageIndex; 325 | 326 | // Create wrapper for better hover effects 327 | const barItemContent = document.createElement("div"); 328 | barItemContent.className = "nav-item-content"; 329 | 330 | // Get the message text for this item 331 | let messageText = "Message " + messageIndex; 332 | if (messageIndexMap.has(messageIndex)) { 333 | const messageNode = messageIndexMap.get(messageIndex); 334 | messageText = messageNode.textContent.substring(0, 30).trim(); 335 | if (messageNode.textContent.length > 30) { 336 | messageText += "..."; 337 | } 338 | } 339 | 340 | // Add the anchor text that will show on hover 341 | const anchorText = document.createElement("span"); 342 | anchorText.className = "nav-item-text"; 343 | anchorText.textContent = messageText; 344 | barItemContent.appendChild(anchorText); 345 | 346 | barItem.appendChild(barItemContent); 347 | 348 | // Add click event to scroll to the message 349 | barItem.addEventListener("click", () => { 350 | // Ensure sidebar is visible 351 | navigator_bar.classList.add('expanded'); 352 | chat_sidebar.style.display = "table"; 353 | 354 | const messageId = `index-${messageIndex}`; 355 | const messageElement = document.getElementById(messageId); 356 | if (messageElement) { 357 | messageElement.scrollIntoView({ behavior: "smooth", block: "center" }); 358 | 359 | // Add a brief highlight effect 360 | messageElement.classList.add("highlight-message"); 361 | setTimeout(() => { 362 | messageElement.classList.remove("highlight-message"); 363 | }, 2000); 364 | 365 | // Mark this nav item as active 366 | document.querySelectorAll('.nav-item').forEach(item => { 367 | item.classList.remove('active'); 368 | }); 369 | barItem.classList.add('active'); 370 | } 371 | }); 372 | 373 | navigator_bar.appendChild(barItem); 374 | } 375 | 376 | // Function to add a message to the sidebar table 377 | function addMessageToSidebar(messageIndex, rowData, id) { 378 | if (!chat_sidebar) return; 379 | 380 | const newRow = chat_sidebar.insertRow(); 381 | const cell = newRow.insertCell(); 382 | cell.innerHTML = `${messageIndex}. ${rowData}`; 383 | 384 | // Add hover behavior to show full text 385 | cell.addEventListener("mouseenter", () => { 386 | if (messageIndexMap.has(messageIndex)) { 387 | const fullMessage = messageIndexMap.get(messageIndex); 388 | const fullText = fullMessage.textContent.trim(); 389 | 390 | // Create or update tooltip 391 | let tooltip = document.getElementById("message-tooltip"); 392 | if (!tooltip) { 393 | tooltip = document.createElement("div"); 394 | tooltip.id = "message-tooltip"; 395 | tooltip.className = "message-tooltip"; 396 | document.body.appendChild(tooltip); 397 | } 398 | 399 | tooltip.textContent = fullText; 400 | 401 | // Position the tooltip 402 | const rect = cell.getBoundingClientRect(); 403 | tooltip.style.top = `${rect.top}px`; 404 | tooltip.style.left = `${rect.right + 10}px`; 405 | tooltip.style.display = "block"; 406 | } 407 | }); 408 | 409 | cell.addEventListener("mouseleave", () => { 410 | const tooltip = document.getElementById("message-tooltip"); 411 | if (tooltip) { 412 | tooltip.style.display = "none"; 413 | } 414 | }); 415 | 416 | console.log("[Content] Added new sidebar row:", rowData); 417 | } 418 | 419 | // Function to handle URL changes 420 | function checkUrlChange() { 421 | const currentUrl = window.location.href; 422 | if (currentUrl !== lastUrl) { 423 | console.log('[Content] URL changed from', lastUrl, 'to', currentUrl); 424 | lastUrl = currentUrl; 425 | // Re-initialize the navigation for the new URL 426 | initializeChatNavigation(); 427 | } 428 | } 429 | 430 | // Start checking for URL changes 431 | setInterval(checkUrlChange, 1000); 432 | 433 | // Listen for theme changes from storage 434 | chrome.storage.onChanged.addListener(function(changes, namespace) { 435 | if (changes.theme) { 436 | console.log("[Content] Theme change detected:", changes.theme.newValue); 437 | applyTheme(changes.theme.newValue); 438 | } 439 | }); 440 | 441 | // Callback for MutationObserver 442 | function mutationCallback(mutationList, observer) { 443 | if (!currentChatConfig || !currentChatConfig.selectors || !currentChatConfig.selectors.messageSelector) { 444 | return; 445 | } 446 | 447 | for (const mutation of mutationList) { 448 | if (mutation.type === 'childList') { 449 | mutation.addedNodes.forEach(node => { 450 | // Check if the added node itself is a message or contains messages 451 | if (node.nodeType === Node.ELEMENT_NODE) { 452 | const messages = []; 453 | if (node.matches(currentChatConfig.selectors.messageSelector)) { 454 | messages.push(node); 455 | } 456 | // Also check children if the added node is a container 457 | messages.push(...node.querySelectorAll(currentChatConfig.selectors.messageSelector)); 458 | 459 | messages.forEach(messageNode => { 460 | if (!messageNode.id) { // Process only if not already processed 461 | try { 462 | messageNode.id = `index-${index}`; 463 | messageIndexMap.set(index, messageNode); 464 | const shortText = messageNode.textContent.substring(0, 12).trim() + "..."; 465 | addMessageToSidebar(index, shortText, messageNode.id); 466 | addMessageToBar(index); 467 | index++; 468 | console.log("[Content] Added new chat message via MutationObserver:", messageNode.id); 469 | } catch (e) { 470 | console.error("[Content] Error processing new message via MutationObserver:", e, "Message content:", messageNode.innerHTML); 471 | } 472 | } 473 | }); 474 | } 475 | }); 476 | } 477 | } 478 | } 479 | 480 | function dragElement(elmnt) { 481 | var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0; 482 | const header = elmnt.querySelector('.sidebar-header'); 483 | 484 | if (header) { 485 | header.onmousedown = dragMouseDown; 486 | } else { 487 | elmnt.onmousedown = dragMouseDown; 488 | } 489 | 490 | function dragMouseDown(e) { 491 | e = e || window.event; 492 | e.preventDefault(); 493 | // get the mouse cursor position at startup: 494 | pos3 = e.clientX; 495 | pos4 = e.clientY; 496 | document.onmouseup = closeDragElement; 497 | // call a function whenever the cursor moves: 498 | document.onmousemove = elementDrag; 499 | } 500 | 501 | function elementDrag(e) { 502 | e = e || window.event; 503 | e.preventDefault(); 504 | // calculate the new cursor position: 505 | pos1 = pos3 - e.clientX; 506 | pos2 = pos4 - e.clientY; 507 | pos3 = e.clientX; 508 | pos4 = e.clientY; 509 | elmnt.style.top = (elmnt.offsetTop - pos2) + "px"; 510 | elmnt.style.left = (elmnt.offsetLeft - pos1) + "px"; 511 | } 512 | 513 | function closeDragElement() { 514 | document.onmouseup = null; 515 | document.onmousemove = null; 516 | } 517 | } 518 | 519 | // Initial load: determine config and setup navigation 520 | (async () => { 521 | await initializeChatNavigation(); 522 | })(); 523 | 524 | 525 | -------------------------------------------------------------------------------- /extension/jquery-ui.min.css: -------------------------------------------------------------------------------- 1 | /*! jQuery UI - v1.14.1 - 2024-10-30 2 | * https://jqueryui.com 3 | * Includes: core.css, accordion.css, autocomplete.css, menu.css, button.css, controlgroup.css, checkboxradio.css, datepicker.css, dialog.css, draggable.css, resizable.css, progressbar.css, selectable.css, selectmenu.css, slider.css, sortable.css, spinner.css, tabs.css, tooltip.css, theme.css 4 | * To view and modify this theme, visit https://jqueryui.com/themeroller/?ffDefault=Arial%2CHelvetica%2Csans-serif&fsDefault=1em&fwDefault=normal&cornerRadius=3px&bgColorHeader=e9e9e9&bgTextureHeader=flat&borderColorHeader=dddddd&fcHeader=333333&iconColorHeader=444444&bgColorContent=ffffff&bgTextureContent=flat&borderColorContent=dddddd&fcContent=333333&iconColorContent=444444&bgColorDefault=f6f6f6&bgTextureDefault=flat&borderColorDefault=c5c5c5&fcDefault=454545&iconColorDefault=777777&bgColorHover=ededed&bgTextureHover=flat&borderColorHover=cccccc&fcHover=2b2b2b&iconColorHover=555555&bgColorActive=007fff&bgTextureActive=flat&borderColorActive=003eff&fcActive=ffffff&iconColorActive=ffffff&bgColorHighlight=fffa90&bgTextureHighlight=flat&borderColorHighlight=dad55e&fcHighlight=777620&iconColorHighlight=777620&bgColorError=fddfdf&bgTextureError=flat&borderColorError=f1a899&fcError=5f3f3f&iconColorError=cc0000&bgColorOverlay=aaaaaa&bgTextureOverlay=flat&bgImgOpacityOverlay=0&opacityOverlay=30&bgColorShadow=666666&bgTextureShadow=flat&bgImgOpacityShadow=0&opacityShadow=30&thicknessShadow=5px&offsetTopShadow=0px&offsetLeftShadow=0px&cornerRadiusShadow=8px 5 | * Copyright OpenJS Foundation and other contributors; Licensed MIT */ 6 | 7 | /* Layout helpers 8 | ----------------------------------*/ 9 | .ui-helper-hidden { 10 | display: none; 11 | } 12 | .ui-helper-hidden-accessible { 13 | border: 0; 14 | clip: rect(0 0 0 0); 15 | height: 1px; 16 | margin: -1px; 17 | overflow: hidden; 18 | padding: 0; 19 | position: absolute; 20 | width: 1px; 21 | } 22 | .ui-helper-reset { 23 | margin: 0; 24 | padding: 0; 25 | border: 0; 26 | outline: 0; 27 | line-height: 1.3; 28 | text-decoration: none; 29 | font-size: 100%; 30 | list-style: none; 31 | } 32 | .ui-helper-clearfix:before, 33 | .ui-helper-clearfix:after { 34 | content: ""; 35 | display: table; 36 | border-collapse: collapse; 37 | } 38 | .ui-helper-clearfix:after { 39 | clear: both; 40 | } 41 | .ui-helper-zfix { 42 | width: 100%; 43 | height: 100%; 44 | top: 0; 45 | left: 0; 46 | position: absolute; 47 | opacity: 0; 48 | } 49 | 50 | .ui-front { 51 | z-index: 100; 52 | } 53 | 54 | 55 | /* Interaction Cues 56 | ----------------------------------*/ 57 | .ui-state-disabled { 58 | cursor: default !important; 59 | pointer-events: none; 60 | } 61 | 62 | 63 | /* Icons 64 | ----------------------------------*/ 65 | .ui-icon { 66 | display: inline-block; 67 | vertical-align: middle; 68 | margin-top: -.25em; 69 | position: relative; 70 | text-indent: -99999px; 71 | overflow: hidden; 72 | background-repeat: no-repeat; 73 | } 74 | 75 | .ui-widget-icon-block { 76 | left: 50%; 77 | margin-left: -8px; 78 | display: block; 79 | } 80 | 81 | /* Misc visuals 82 | ----------------------------------*/ 83 | 84 | /* Overlays */ 85 | .ui-widget-overlay { 86 | position: fixed; 87 | top: 0; 88 | left: 0; 89 | width: 100%; 90 | height: 100%; 91 | } 92 | .ui-accordion .ui-accordion-header { 93 | display: block; 94 | cursor: pointer; 95 | position: relative; 96 | margin: 2px 0 0 0; 97 | padding: .5em .5em .5em .7em; 98 | font-size: 100%; 99 | } 100 | .ui-accordion .ui-accordion-content { 101 | padding: 1em 2.2em; 102 | border-top: 0; 103 | overflow: auto; 104 | } 105 | .ui-autocomplete { 106 | position: absolute; 107 | top: 0; 108 | left: 0; 109 | cursor: default; 110 | } 111 | .ui-menu { 112 | list-style: none; 113 | padding: 0; 114 | margin: 0; 115 | display: block; 116 | outline: 0; 117 | } 118 | .ui-menu .ui-menu { 119 | position: absolute; 120 | } 121 | .ui-menu .ui-menu-item { 122 | margin: 0; 123 | cursor: pointer; 124 | } 125 | .ui-menu .ui-menu-item-wrapper { 126 | position: relative; 127 | padding: 3px 1em 3px .4em; 128 | } 129 | .ui-menu .ui-menu-divider { 130 | margin: 5px 0; 131 | height: 0; 132 | font-size: 0; 133 | line-height: 0; 134 | border-width: 1px 0 0 0; 135 | } 136 | .ui-menu .ui-state-focus, 137 | .ui-menu .ui-state-active { 138 | margin: -1px; 139 | } 140 | 141 | /* icon support */ 142 | .ui-menu-icons { 143 | position: relative; 144 | } 145 | .ui-menu-icons .ui-menu-item-wrapper { 146 | padding-left: 2em; 147 | } 148 | 149 | /* left-aligned */ 150 | .ui-menu .ui-icon { 151 | position: absolute; 152 | top: 0; 153 | bottom: 0; 154 | left: .2em; 155 | margin: auto 0; 156 | } 157 | 158 | /* right-aligned */ 159 | .ui-menu .ui-menu-icon { 160 | left: auto; 161 | right: 0; 162 | } 163 | .ui-button { 164 | padding: .4em 1em; 165 | display: inline-block; 166 | position: relative; 167 | line-height: normal; 168 | margin-right: .1em; 169 | cursor: pointer; 170 | vertical-align: middle; 171 | text-align: center; 172 | -webkit-user-select: none; 173 | user-select: none; 174 | } 175 | 176 | .ui-button, 177 | .ui-button:link, 178 | .ui-button:visited, 179 | .ui-button:hover, 180 | .ui-button:active { 181 | text-decoration: none; 182 | } 183 | 184 | /* to make room for the icon, a width needs to be set here */ 185 | .ui-button-icon-only { 186 | width: 2em; 187 | box-sizing: border-box; 188 | text-indent: -9999px; 189 | white-space: nowrap; 190 | } 191 | 192 | /* no icon support for input elements */ 193 | input.ui-button.ui-button-icon-only { 194 | text-indent: 0; 195 | } 196 | 197 | /* button icon element(s) */ 198 | .ui-button-icon-only .ui-icon { 199 | position: absolute; 200 | top: 50%; 201 | left: 50%; 202 | margin-top: -8px; 203 | margin-left: -8px; 204 | } 205 | 206 | .ui-button.ui-icon-notext .ui-icon { 207 | padding: 0; 208 | width: 2.1em; 209 | height: 2.1em; 210 | text-indent: -9999px; 211 | white-space: nowrap; 212 | 213 | } 214 | 215 | input.ui-button.ui-icon-notext .ui-icon { 216 | width: auto; 217 | height: auto; 218 | text-indent: 0; 219 | white-space: normal; 220 | padding: .4em 1em; 221 | } 222 | 223 | /* workarounds */ 224 | /* Support: Firefox 5 - 125+ */ 225 | input.ui-button::-moz-focus-inner, 226 | button.ui-button::-moz-focus-inner { 227 | border: 0; 228 | padding: 0; 229 | } 230 | .ui-controlgroup { 231 | vertical-align: middle; 232 | display: inline-block; 233 | } 234 | .ui-controlgroup > .ui-controlgroup-item { 235 | float: left; 236 | margin-left: 0; 237 | margin-right: 0; 238 | } 239 | .ui-controlgroup > .ui-controlgroup-item:focus, 240 | .ui-controlgroup > .ui-controlgroup-item.ui-visual-focus { 241 | z-index: 9999; 242 | } 243 | .ui-controlgroup-vertical > .ui-controlgroup-item { 244 | display: block; 245 | float: none; 246 | width: 100%; 247 | margin-top: 0; 248 | margin-bottom: 0; 249 | text-align: left; 250 | } 251 | .ui-controlgroup-vertical .ui-controlgroup-item { 252 | box-sizing: border-box; 253 | } 254 | .ui-controlgroup .ui-controlgroup-label { 255 | padding: .4em 1em; 256 | } 257 | .ui-controlgroup .ui-controlgroup-label span { 258 | font-size: 80%; 259 | } 260 | .ui-controlgroup-horizontal .ui-controlgroup-label + .ui-controlgroup-item { 261 | border-left: none; 262 | } 263 | .ui-controlgroup-vertical .ui-controlgroup-label + .ui-controlgroup-item { 264 | border-top: none; 265 | } 266 | .ui-controlgroup-horizontal .ui-controlgroup-label.ui-widget-content { 267 | border-right: none; 268 | } 269 | .ui-controlgroup-vertical .ui-controlgroup-label.ui-widget-content { 270 | border-bottom: none; 271 | } 272 | 273 | /* Spinner specific style fixes */ 274 | .ui-controlgroup-vertical .ui-spinner-input { 275 | width: calc( 100% - 2.4em ); 276 | } 277 | .ui-controlgroup-vertical .ui-spinner .ui-spinner-up { 278 | border-top-style: solid; 279 | } 280 | 281 | .ui-checkboxradio-label .ui-icon-background { 282 | box-shadow: inset 1px 1px 1px #ccc; 283 | border-radius: .12em; 284 | border: none; 285 | } 286 | .ui-checkboxradio-radio-label .ui-icon-background { 287 | width: 16px; 288 | height: 16px; 289 | border-radius: 1em; 290 | overflow: visible; 291 | border: none; 292 | } 293 | .ui-checkboxradio-radio-label.ui-checkboxradio-checked .ui-icon, 294 | .ui-checkboxradio-radio-label.ui-checkboxradio-checked:hover .ui-icon { 295 | background-image: none; 296 | width: 8px; 297 | height: 8px; 298 | border-width: 4px; 299 | border-style: solid; 300 | } 301 | .ui-checkboxradio-disabled { 302 | pointer-events: none; 303 | } 304 | .ui-datepicker { 305 | width: 17em; 306 | padding: .2em .2em 0; 307 | display: none; 308 | } 309 | .ui-datepicker .ui-datepicker-header { 310 | position: relative; 311 | padding: .2em 0; 312 | } 313 | .ui-datepicker .ui-datepicker-prev, 314 | .ui-datepicker .ui-datepicker-next { 315 | position: absolute; 316 | top: 2px; 317 | width: 1.8em; 318 | height: 1.8em; 319 | } 320 | .ui-datepicker .ui-datepicker-prev-hover, 321 | .ui-datepicker .ui-datepicker-next-hover { 322 | top: 1px; 323 | } 324 | .ui-datepicker .ui-datepicker-prev { 325 | left: 2px; 326 | } 327 | .ui-datepicker .ui-datepicker-next { 328 | right: 2px; 329 | } 330 | .ui-datepicker .ui-datepicker-prev-hover { 331 | left: 1px; 332 | } 333 | .ui-datepicker .ui-datepicker-next-hover { 334 | right: 1px; 335 | } 336 | .ui-datepicker .ui-datepicker-prev span, 337 | .ui-datepicker .ui-datepicker-next span { 338 | display: block; 339 | position: absolute; 340 | left: 50%; 341 | margin-left: -8px; 342 | top: 50%; 343 | margin-top: -8px; 344 | } 345 | .ui-datepicker .ui-datepicker-title { 346 | margin: 0 2.3em; 347 | line-height: 1.8em; 348 | text-align: center; 349 | } 350 | .ui-datepicker .ui-datepicker-title select { 351 | font-size: 1em; 352 | margin: 1px 0; 353 | } 354 | .ui-datepicker select.ui-datepicker-month, 355 | .ui-datepicker select.ui-datepicker-year { 356 | width: 45%; 357 | } 358 | .ui-datepicker table { 359 | width: 100%; 360 | font-size: .9em; 361 | border-collapse: collapse; 362 | margin: 0 0 .4em; 363 | } 364 | .ui-datepicker th { 365 | padding: .7em .3em; 366 | text-align: center; 367 | font-weight: bold; 368 | border: 0; 369 | } 370 | .ui-datepicker td { 371 | border: 0; 372 | padding: 1px; 373 | } 374 | .ui-datepicker td span, 375 | .ui-datepicker td a { 376 | display: block; 377 | padding: .2em; 378 | text-align: right; 379 | text-decoration: none; 380 | } 381 | .ui-datepicker .ui-datepicker-buttonpane { 382 | background-image: none; 383 | margin: .7em 0 0 0; 384 | padding: 0 .2em; 385 | border-left: 0; 386 | border-right: 0; 387 | border-bottom: 0; 388 | } 389 | .ui-datepicker .ui-datepicker-buttonpane button { 390 | float: right; 391 | margin: .5em .2em .4em; 392 | cursor: pointer; 393 | padding: .2em .6em .3em .6em; 394 | width: auto; 395 | overflow: visible; 396 | } 397 | .ui-datepicker .ui-datepicker-buttonpane button.ui-datepicker-current { 398 | float: left; 399 | } 400 | 401 | /* with multiple calendars */ 402 | .ui-datepicker.ui-datepicker-multi { 403 | width: auto; 404 | } 405 | .ui-datepicker-multi .ui-datepicker-group { 406 | float: left; 407 | } 408 | .ui-datepicker-multi .ui-datepicker-group table { 409 | width: 95%; 410 | margin: 0 auto .4em; 411 | } 412 | .ui-datepicker-multi-2 .ui-datepicker-group { 413 | width: 50%; 414 | } 415 | .ui-datepicker-multi-3 .ui-datepicker-group { 416 | width: 33.3%; 417 | } 418 | .ui-datepicker-multi-4 .ui-datepicker-group { 419 | width: 25%; 420 | } 421 | .ui-datepicker-multi .ui-datepicker-group-last .ui-datepicker-header, 422 | .ui-datepicker-multi .ui-datepicker-group-middle .ui-datepicker-header { 423 | border-left-width: 0; 424 | } 425 | .ui-datepicker-multi .ui-datepicker-buttonpane { 426 | clear: left; 427 | } 428 | .ui-datepicker-row-break { 429 | clear: both; 430 | width: 100%; 431 | font-size: 0; 432 | } 433 | 434 | /* RTL support */ 435 | .ui-datepicker-rtl { 436 | direction: rtl; 437 | } 438 | .ui-datepicker-rtl .ui-datepicker-prev { 439 | right: 2px; 440 | left: auto; 441 | } 442 | .ui-datepicker-rtl .ui-datepicker-next { 443 | left: 2px; 444 | right: auto; 445 | } 446 | .ui-datepicker-rtl .ui-datepicker-prev:hover { 447 | right: 1px; 448 | left: auto; 449 | } 450 | .ui-datepicker-rtl .ui-datepicker-next:hover { 451 | left: 1px; 452 | right: auto; 453 | } 454 | .ui-datepicker-rtl .ui-datepicker-buttonpane { 455 | clear: right; 456 | } 457 | .ui-datepicker-rtl .ui-datepicker-buttonpane button { 458 | float: left; 459 | } 460 | .ui-datepicker-rtl .ui-datepicker-buttonpane button.ui-datepicker-current, 461 | .ui-datepicker-rtl .ui-datepicker-group { 462 | float: right; 463 | } 464 | .ui-datepicker-rtl .ui-datepicker-group-last .ui-datepicker-header, 465 | .ui-datepicker-rtl .ui-datepicker-group-middle .ui-datepicker-header { 466 | border-right-width: 0; 467 | border-left-width: 1px; 468 | } 469 | 470 | /* Icons */ 471 | .ui-datepicker .ui-icon { 472 | display: block; 473 | text-indent: -99999px; 474 | overflow: hidden; 475 | background-repeat: no-repeat; 476 | left: .5em; 477 | top: .3em; 478 | } 479 | .ui-dialog { 480 | position: absolute; 481 | top: 0; 482 | left: 0; 483 | padding: .2em; 484 | outline: 0; 485 | } 486 | .ui-dialog .ui-dialog-titlebar { 487 | padding: .4em 1em; 488 | position: relative; 489 | } 490 | .ui-dialog .ui-dialog-title { 491 | float: left; 492 | margin: .1em 0; 493 | white-space: nowrap; 494 | width: 90%; 495 | overflow: hidden; 496 | text-overflow: ellipsis; 497 | } 498 | .ui-dialog .ui-dialog-titlebar-close { 499 | position: absolute; 500 | right: .3em; 501 | top: 50%; 502 | width: 20px; 503 | margin: -10px 0 0 0; 504 | padding: 1px; 505 | height: 20px; 506 | } 507 | .ui-dialog .ui-dialog-content { 508 | position: relative; 509 | border: 0; 510 | padding: .5em 1em; 511 | background: none; 512 | overflow: auto; 513 | } 514 | .ui-dialog .ui-dialog-buttonpane { 515 | text-align: left; 516 | border-width: 1px 0 0 0; 517 | background-image: none; 518 | margin-top: .5em; 519 | padding: .3em 1em .5em .4em; 520 | } 521 | .ui-dialog .ui-dialog-buttonpane .ui-dialog-buttonset { 522 | float: right; 523 | } 524 | .ui-dialog .ui-dialog-buttonpane button { 525 | margin: .5em .4em .5em 0; 526 | cursor: pointer; 527 | } 528 | .ui-dialog .ui-resizable-n { 529 | height: 2px; 530 | top: 0; 531 | } 532 | .ui-dialog .ui-resizable-e { 533 | width: 2px; 534 | right: 0; 535 | } 536 | .ui-dialog .ui-resizable-s { 537 | height: 2px; 538 | bottom: 0; 539 | } 540 | .ui-dialog .ui-resizable-w { 541 | width: 2px; 542 | left: 0; 543 | } 544 | .ui-dialog .ui-resizable-se, 545 | .ui-dialog .ui-resizable-sw, 546 | .ui-dialog .ui-resizable-ne, 547 | .ui-dialog .ui-resizable-nw { 548 | width: 7px; 549 | height: 7px; 550 | } 551 | .ui-dialog .ui-resizable-se { 552 | right: 0; 553 | bottom: 0; 554 | } 555 | .ui-dialog .ui-resizable-sw { 556 | left: 0; 557 | bottom: 0; 558 | } 559 | .ui-dialog .ui-resizable-ne { 560 | right: 0; 561 | top: 0; 562 | } 563 | .ui-dialog .ui-resizable-nw { 564 | left: 0; 565 | top: 0; 566 | } 567 | .ui-draggable .ui-dialog-titlebar { 568 | cursor: move; 569 | } 570 | .ui-draggable-handle { 571 | touch-action: none; 572 | } 573 | .ui-resizable { 574 | position: relative; 575 | } 576 | .ui-resizable-handle { 577 | position: absolute; 578 | font-size: 0.1px; 579 | display: block; 580 | touch-action: none; 581 | } 582 | .ui-resizable-disabled .ui-resizable-handle, 583 | .ui-resizable-autohide .ui-resizable-handle { 584 | display: none; 585 | } 586 | .ui-resizable-n { 587 | cursor: n-resize; 588 | height: 7px; 589 | width: 100%; 590 | top: -5px; 591 | left: 0; 592 | } 593 | .ui-resizable-s { 594 | cursor: s-resize; 595 | height: 7px; 596 | width: 100%; 597 | bottom: -5px; 598 | left: 0; 599 | } 600 | .ui-resizable-e { 601 | cursor: e-resize; 602 | width: 7px; 603 | right: -5px; 604 | top: 0; 605 | height: 100%; 606 | } 607 | .ui-resizable-w { 608 | cursor: w-resize; 609 | width: 7px; 610 | left: -5px; 611 | top: 0; 612 | height: 100%; 613 | } 614 | .ui-resizable-se { 615 | cursor: se-resize; 616 | width: 12px; 617 | height: 12px; 618 | right: 1px; 619 | bottom: 1px; 620 | } 621 | .ui-resizable-sw { 622 | cursor: sw-resize; 623 | width: 9px; 624 | height: 9px; 625 | left: -5px; 626 | bottom: -5px; 627 | } 628 | .ui-resizable-nw { 629 | cursor: nw-resize; 630 | width: 9px; 631 | height: 9px; 632 | left: -5px; 633 | top: -5px; 634 | } 635 | .ui-resizable-ne { 636 | cursor: ne-resize; 637 | width: 9px; 638 | height: 9px; 639 | right: -5px; 640 | top: -5px; 641 | } 642 | .ui-progressbar { 643 | height: 2em; 644 | text-align: left; 645 | overflow: hidden; 646 | } 647 | .ui-progressbar .ui-progressbar-value { 648 | margin: -1px; 649 | height: 100%; 650 | } 651 | .ui-progressbar .ui-progressbar-overlay { 652 | background: url(""); 653 | height: 100%; 654 | opacity: 0.25; 655 | } 656 | .ui-progressbar-indeterminate .ui-progressbar-value { 657 | background-image: none; 658 | } 659 | .ui-selectable { 660 | touch-action: none; 661 | } 662 | .ui-selectable-helper { 663 | position: absolute; 664 | z-index: 100; 665 | border: 1px dotted black; 666 | } 667 | .ui-selectmenu-menu { 668 | padding: 0; 669 | margin: 0; 670 | position: absolute; 671 | top: 0; 672 | left: 0; 673 | display: none; 674 | } 675 | .ui-selectmenu-menu .ui-menu { 676 | overflow: auto; 677 | overflow-x: hidden; 678 | padding-bottom: 1px; 679 | } 680 | .ui-selectmenu-menu .ui-menu .ui-selectmenu-optgroup { 681 | font-size: 1em; 682 | font-weight: bold; 683 | line-height: 1.5; 684 | padding: 2px 0.4em; 685 | margin: 0.5em 0 0 0; 686 | height: auto; 687 | border: 0; 688 | } 689 | .ui-selectmenu-open { 690 | display: block; 691 | } 692 | .ui-selectmenu-text { 693 | display: block; 694 | margin-right: 20px; 695 | overflow: hidden; 696 | text-overflow: ellipsis; 697 | } 698 | .ui-selectmenu-button.ui-button { 699 | text-align: left; 700 | white-space: nowrap; 701 | width: 14em; 702 | } 703 | .ui-selectmenu-icon.ui-icon { 704 | float: right; 705 | margin-top: 0; 706 | } 707 | .ui-slider { 708 | position: relative; 709 | text-align: left; 710 | } 711 | .ui-slider .ui-slider-handle { 712 | position: absolute; 713 | z-index: 2; 714 | width: 1.2em; 715 | height: 1.2em; 716 | cursor: pointer; 717 | touch-action: none; 718 | } 719 | .ui-slider .ui-slider-range { 720 | position: absolute; 721 | z-index: 1; 722 | font-size: .7em; 723 | display: block; 724 | border: 0; 725 | background-position: 0 0; 726 | } 727 | 728 | .ui-slider-horizontal { 729 | height: .8em; 730 | } 731 | .ui-slider-horizontal .ui-slider-handle { 732 | top: -.3em; 733 | margin-left: -.6em; 734 | } 735 | .ui-slider-horizontal .ui-slider-range { 736 | top: 0; 737 | height: 100%; 738 | } 739 | .ui-slider-horizontal .ui-slider-range-min { 740 | left: 0; 741 | } 742 | .ui-slider-horizontal .ui-slider-range-max { 743 | right: 0; 744 | } 745 | 746 | .ui-slider-vertical { 747 | width: .8em; 748 | height: 100px; 749 | } 750 | .ui-slider-vertical .ui-slider-handle { 751 | left: -.3em; 752 | margin-left: 0; 753 | margin-bottom: -.6em; 754 | } 755 | .ui-slider-vertical .ui-slider-range { 756 | left: 0; 757 | width: 100%; 758 | } 759 | .ui-slider-vertical .ui-slider-range-min { 760 | bottom: 0; 761 | } 762 | .ui-slider-vertical .ui-slider-range-max { 763 | top: 0; 764 | } 765 | .ui-sortable-handle { 766 | touch-action: none; 767 | } 768 | .ui-spinner { 769 | position: relative; 770 | display: inline-block; 771 | overflow: hidden; 772 | padding: 0; 773 | vertical-align: middle; 774 | } 775 | .ui-spinner-input { 776 | border: none; 777 | background: none; 778 | color: inherit; 779 | padding: .222em 0; 780 | margin: .2em 0; 781 | vertical-align: middle; 782 | margin-left: .4em; 783 | margin-right: 2em; 784 | } 785 | .ui-spinner-button { 786 | width: 1.6em; 787 | height: 50%; 788 | font-size: .5em; 789 | padding: 0; 790 | margin: 0; 791 | text-align: center; 792 | position: absolute; 793 | cursor: default; 794 | display: block; 795 | overflow: hidden; 796 | right: 0; 797 | } 798 | /* more specificity required here to override default borders */ 799 | .ui-spinner a.ui-spinner-button { 800 | border-top-style: none; 801 | border-bottom-style: none; 802 | border-right-style: none; 803 | } 804 | .ui-spinner-up { 805 | top: 0; 806 | } 807 | .ui-spinner-down { 808 | bottom: 0; 809 | } 810 | .ui-tabs { 811 | position: relative;/* position: relative prevents IE scroll bug (element with position: relative inside container with overflow: auto appear as "fixed") */ 812 | padding: .2em; 813 | } 814 | .ui-tabs .ui-tabs-nav { 815 | margin: 0; 816 | padding: .2em .2em 0; 817 | } 818 | .ui-tabs .ui-tabs-nav li { 819 | list-style: none; 820 | float: left; 821 | position: relative; 822 | top: 0; 823 | margin: 1px .2em 0 0; 824 | border-bottom-width: 0; 825 | padding: 0; 826 | white-space: nowrap; 827 | } 828 | .ui-tabs .ui-tabs-nav .ui-tabs-anchor { 829 | float: left; 830 | padding: .5em 1em; 831 | text-decoration: none; 832 | } 833 | .ui-tabs .ui-tabs-nav li.ui-tabs-active { 834 | margin-bottom: -1px; 835 | padding-bottom: 1px; 836 | } 837 | .ui-tabs .ui-tabs-nav li.ui-tabs-active .ui-tabs-anchor, 838 | .ui-tabs .ui-tabs-nav li.ui-state-disabled .ui-tabs-anchor, 839 | .ui-tabs .ui-tabs-nav li.ui-tabs-loading .ui-tabs-anchor { 840 | cursor: text; 841 | } 842 | .ui-tabs-collapsible .ui-tabs-nav li.ui-tabs-active .ui-tabs-anchor { 843 | cursor: pointer; 844 | } 845 | .ui-tabs .ui-tabs-panel { 846 | display: block; 847 | border-width: 0; 848 | padding: 1em 1.4em; 849 | background: none; 850 | } 851 | .ui-tooltip { 852 | padding: 8px; 853 | position: absolute; 854 | z-index: 9999; 855 | max-width: 300px; 856 | } 857 | body .ui-tooltip { 858 | border-width: 2px; 859 | } 860 | /* Component containers 861 | ----------------------------------*/ 862 | .ui-widget { 863 | font-family: Arial,Helvetica,sans-serif; 864 | font-size: 1em; 865 | } 866 | .ui-widget .ui-widget { 867 | font-size: 1em; 868 | } 869 | .ui-widget input, 870 | .ui-widget select, 871 | .ui-widget textarea, 872 | .ui-widget button { 873 | font-family: Arial,Helvetica,sans-serif; 874 | font-size: 1em; 875 | } 876 | .ui-widget.ui-widget-content { 877 | border: 1px solid #c5c5c5; 878 | } 879 | .ui-widget-content { 880 | border: 1px solid #dddddd; 881 | background: #ffffff; 882 | color: #333333; 883 | } 884 | .ui-widget-content a { 885 | color: #333333; 886 | } 887 | .ui-widget-header { 888 | border: 1px solid #dddddd; 889 | background: #e9e9e9; 890 | color: #333333; 891 | font-weight: bold; 892 | } 893 | .ui-widget-header a { 894 | color: #333333; 895 | } 896 | 897 | /* Interaction states 898 | ----------------------------------*/ 899 | .ui-state-default, 900 | .ui-widget-content .ui-state-default, 901 | .ui-widget-header .ui-state-default, 902 | .ui-button, 903 | 904 | /* We use html here because we need a greater specificity to make sure disabled 905 | works properly when clicked or hovered */ 906 | html .ui-button.ui-state-disabled:hover, 907 | html .ui-button.ui-state-disabled:active { 908 | border: 1px solid #c5c5c5; 909 | background: #f6f6f6; 910 | font-weight: normal; 911 | color: #454545; 912 | } 913 | .ui-state-default a, 914 | .ui-state-default a:link, 915 | .ui-state-default a:visited, 916 | a.ui-button, 917 | a:link.ui-button, 918 | a:visited.ui-button, 919 | .ui-button { 920 | color: #454545; 921 | text-decoration: none; 922 | } 923 | .ui-state-hover, 924 | .ui-widget-content .ui-state-hover, 925 | .ui-widget-header .ui-state-hover, 926 | .ui-state-focus, 927 | .ui-widget-content .ui-state-focus, 928 | .ui-widget-header .ui-state-focus, 929 | .ui-button:hover, 930 | .ui-button:focus { 931 | border: 1px solid #cccccc; 932 | background: #ededed; 933 | font-weight: normal; 934 | color: #2b2b2b; 935 | } 936 | .ui-state-hover a, 937 | .ui-state-hover a:hover, 938 | .ui-state-hover a:link, 939 | .ui-state-hover a:visited, 940 | .ui-state-focus a, 941 | .ui-state-focus a:hover, 942 | .ui-state-focus a:link, 943 | .ui-state-focus a:visited, 944 | a.ui-button:hover, 945 | a.ui-button:focus { 946 | color: #2b2b2b; 947 | text-decoration: none; 948 | } 949 | 950 | .ui-visual-focus { 951 | box-shadow: 0 0 3px 1px rgb(94, 158, 214); 952 | } 953 | .ui-state-active, 954 | .ui-widget-content .ui-state-active, 955 | .ui-widget-header .ui-state-active, 956 | a.ui-button:active, 957 | .ui-button:active, 958 | .ui-button.ui-state-active:hover { 959 | border: 1px solid #003eff; 960 | background: #007fff; 961 | font-weight: normal; 962 | color: #ffffff; 963 | } 964 | .ui-icon-background, 965 | .ui-state-active .ui-icon-background { 966 | border: #003eff; 967 | background-color: #ffffff; 968 | } 969 | .ui-state-active a, 970 | .ui-state-active a:link, 971 | .ui-state-active a:visited { 972 | color: #ffffff; 973 | text-decoration: none; 974 | } 975 | 976 | /* Interaction Cues 977 | ----------------------------------*/ 978 | .ui-state-highlight, 979 | .ui-widget-content .ui-state-highlight, 980 | .ui-widget-header .ui-state-highlight { 981 | border: 1px solid #dad55e; 982 | background: #fffa90; 983 | color: #777620; 984 | } 985 | .ui-state-checked { 986 | border: 1px solid #dad55e; 987 | background: #fffa90; 988 | } 989 | .ui-state-highlight a, 990 | .ui-widget-content .ui-state-highlight a, 991 | .ui-widget-header .ui-state-highlight a { 992 | color: #777620; 993 | } 994 | .ui-state-error, 995 | .ui-widget-content .ui-state-error, 996 | .ui-widget-header .ui-state-error { 997 | border: 1px solid #f1a899; 998 | background: #fddfdf; 999 | color: #5f3f3f; 1000 | } 1001 | .ui-state-error a, 1002 | .ui-widget-content .ui-state-error a, 1003 | .ui-widget-header .ui-state-error a { 1004 | color: #5f3f3f; 1005 | } 1006 | .ui-state-error-text, 1007 | .ui-widget-content .ui-state-error-text, 1008 | .ui-widget-header .ui-state-error-text { 1009 | color: #5f3f3f; 1010 | } 1011 | .ui-priority-primary, 1012 | .ui-widget-content .ui-priority-primary, 1013 | .ui-widget-header .ui-priority-primary { 1014 | font-weight: bold; 1015 | } 1016 | .ui-priority-secondary, 1017 | .ui-widget-content .ui-priority-secondary, 1018 | .ui-widget-header .ui-priority-secondary { 1019 | opacity: .7; 1020 | font-weight: normal; 1021 | } 1022 | .ui-state-disabled, 1023 | .ui-widget-content .ui-state-disabled, 1024 | .ui-widget-header .ui-state-disabled { 1025 | opacity: .35; 1026 | background-image: none; 1027 | } 1028 | 1029 | /* Icons 1030 | ----------------------------------*/ 1031 | 1032 | /* states and images */ 1033 | .ui-icon { 1034 | width: 16px; 1035 | height: 16px; 1036 | } 1037 | .ui-icon, 1038 | .ui-widget-content .ui-icon { 1039 | background-image: url("images/ui-icons_444444_256x240.png"); 1040 | } 1041 | .ui-widget-header .ui-icon { 1042 | background-image: url("images/ui-icons_444444_256x240.png"); 1043 | } 1044 | .ui-state-hover .ui-icon, 1045 | .ui-state-focus .ui-icon, 1046 | .ui-button:hover .ui-icon, 1047 | .ui-button:focus .ui-icon { 1048 | background-image: url("images/ui-icons_555555_256x240.png"); 1049 | } 1050 | .ui-state-active .ui-icon, 1051 | .ui-button:active .ui-icon { 1052 | background-image: url("images/ui-icons_ffffff_256x240.png"); 1053 | } 1054 | .ui-state-highlight .ui-icon, 1055 | .ui-button .ui-state-highlight.ui-icon { 1056 | background-image: url("images/ui-icons_777620_256x240.png"); 1057 | } 1058 | .ui-state-error .ui-icon, 1059 | .ui-state-error-text .ui-icon { 1060 | background-image: url("images/ui-icons_cc0000_256x240.png"); 1061 | } 1062 | .ui-button .ui-icon { 1063 | background-image: url("images/ui-icons_777777_256x240.png"); 1064 | } 1065 | 1066 | /* positioning */ 1067 | /* Three classes needed to override `.ui-button:hover .ui-icon` */ 1068 | .ui-icon-blank.ui-icon-blank.ui-icon-blank { 1069 | background-image: none; 1070 | } 1071 | .ui-icon-caret-1-n { background-position: 0 0; } 1072 | .ui-icon-caret-1-ne { background-position: -16px 0; } 1073 | .ui-icon-caret-1-e { background-position: -32px 0; } 1074 | .ui-icon-caret-1-se { background-position: -48px 0; } 1075 | .ui-icon-caret-1-s { background-position: -65px 0; } 1076 | .ui-icon-caret-1-sw { background-position: -80px 0; } 1077 | .ui-icon-caret-1-w { background-position: -96px 0; } 1078 | .ui-icon-caret-1-nw { background-position: -112px 0; } 1079 | .ui-icon-caret-2-n-s { background-position: -128px 0; } 1080 | .ui-icon-caret-2-e-w { background-position: -144px 0; } 1081 | .ui-icon-triangle-1-n { background-position: 0 -16px; } 1082 | .ui-icon-triangle-1-ne { background-position: -16px -16px; } 1083 | .ui-icon-triangle-1-e { background-position: -32px -16px; } 1084 | .ui-icon-triangle-1-se { background-position: -48px -16px; } 1085 | .ui-icon-triangle-1-s { background-position: -65px -16px; } 1086 | .ui-icon-triangle-1-sw { background-position: -80px -16px; } 1087 | .ui-icon-triangle-1-w { background-position: -96px -16px; } 1088 | .ui-icon-triangle-1-nw { background-position: -112px -16px; } 1089 | .ui-icon-triangle-2-n-s { background-position: -128px -16px; } 1090 | .ui-icon-triangle-2-e-w { background-position: -144px -16px; } 1091 | .ui-icon-arrow-1-n { background-position: 0 -32px; } 1092 | .ui-icon-arrow-1-ne { background-position: -16px -32px; } 1093 | .ui-icon-arrow-1-e { background-position: -32px -32px; } 1094 | .ui-icon-arrow-1-se { background-position: -48px -32px; } 1095 | .ui-icon-arrow-1-s { background-position: -65px -32px; } 1096 | .ui-icon-arrow-1-sw { background-position: -80px -32px; } 1097 | .ui-icon-arrow-1-w { background-position: -96px -32px; } 1098 | .ui-icon-arrow-1-nw { background-position: -112px -32px; } 1099 | .ui-icon-arrow-2-n-s { background-position: -128px -32px; } 1100 | .ui-icon-arrow-2-ne-sw { background-position: -144px -32px; } 1101 | .ui-icon-arrow-2-e-w { background-position: -160px -32px; } 1102 | .ui-icon-arrow-2-se-nw { background-position: -176px -32px; } 1103 | .ui-icon-arrowstop-1-n { background-position: -192px -32px; } 1104 | .ui-icon-arrowstop-1-e { background-position: -208px -32px; } 1105 | .ui-icon-arrowstop-1-s { background-position: -224px -32px; } 1106 | .ui-icon-arrowstop-1-w { background-position: -240px -32px; } 1107 | .ui-icon-arrowthick-1-n { background-position: 1px -48px; } 1108 | .ui-icon-arrowthick-1-ne { background-position: -16px -48px; } 1109 | .ui-icon-arrowthick-1-e { background-position: -32px -48px; } 1110 | .ui-icon-arrowthick-1-se { background-position: -48px -48px; } 1111 | .ui-icon-arrowthick-1-s { background-position: -64px -48px; } 1112 | .ui-icon-arrowthick-1-sw { background-position: -80px -48px; } 1113 | .ui-icon-arrowthick-1-w { background-position: -96px -48px; } 1114 | .ui-icon-arrowthick-1-nw { background-position: -112px -48px; } 1115 | .ui-icon-arrowthick-2-n-s { background-position: -128px -48px; } 1116 | .ui-icon-arrowthick-2-ne-sw { background-position: -144px -48px; } 1117 | .ui-icon-arrowthick-2-e-w { background-position: -160px -48px; } 1118 | .ui-icon-arrowthick-2-se-nw { background-position: -176px -48px; } 1119 | .ui-icon-arrowthickstop-1-n { background-position: -192px -48px; } 1120 | .ui-icon-arrowthickstop-1-e { background-position: -208px -48px; } 1121 | .ui-icon-arrowthickstop-1-s { background-position: -224px -48px; } 1122 | .ui-icon-arrowthickstop-1-w { background-position: -240px -48px; } 1123 | .ui-icon-arrowreturnthick-1-w { background-position: 0 -64px; } 1124 | .ui-icon-arrowreturnthick-1-n { background-position: -16px -64px; } 1125 | .ui-icon-arrowreturnthick-1-e { background-position: -32px -64px; } 1126 | .ui-icon-arrowreturnthick-1-s { background-position: -48px -64px; } 1127 | .ui-icon-arrowreturn-1-w { background-position: -64px -64px; } 1128 | .ui-icon-arrowreturn-1-n { background-position: -80px -64px; } 1129 | .ui-icon-arrowreturn-1-e { background-position: -96px -64px; } 1130 | .ui-icon-arrowreturn-1-s { background-position: -112px -64px; } 1131 | .ui-icon-arrowrefresh-1-w { background-position: -128px -64px; } 1132 | .ui-icon-arrowrefresh-1-n { background-position: -144px -64px; } 1133 | .ui-icon-arrowrefresh-1-e { background-position: -160px -64px; } 1134 | .ui-icon-arrowrefresh-1-s { background-position: -176px -64px; } 1135 | .ui-icon-arrow-4 { background-position: 0 -80px; } 1136 | .ui-icon-arrow-4-diag { background-position: -16px -80px; } 1137 | .ui-icon-extlink { background-position: -32px -80px; } 1138 | .ui-icon-newwin { background-position: -48px -80px; } 1139 | .ui-icon-refresh { background-position: -64px -80px; } 1140 | .ui-icon-shuffle { background-position: -80px -80px; } 1141 | .ui-icon-transfer-e-w { background-position: -96px -80px; } 1142 | .ui-icon-transferthick-e-w { background-position: -112px -80px; } 1143 | .ui-icon-folder-collapsed { background-position: 0 -96px; } 1144 | .ui-icon-folder-open { background-position: -16px -96px; } 1145 | .ui-icon-document { background-position: -32px -96px; } 1146 | .ui-icon-document-b { background-position: -48px -96px; } 1147 | .ui-icon-note { background-position: -64px -96px; } 1148 | .ui-icon-mail-closed { background-position: -80px -96px; } 1149 | .ui-icon-mail-open { background-position: -96px -96px; } 1150 | .ui-icon-suitcase { background-position: -112px -96px; } 1151 | .ui-icon-comment { background-position: -128px -96px; } 1152 | .ui-icon-person { background-position: -144px -96px; } 1153 | .ui-icon-print { background-position: -160px -96px; } 1154 | .ui-icon-trash { background-position: -176px -96px; } 1155 | .ui-icon-locked { background-position: -192px -96px; } 1156 | .ui-icon-unlocked { background-position: -208px -96px; } 1157 | .ui-icon-bookmark { background-position: -224px -96px; } 1158 | .ui-icon-tag { background-position: -240px -96px; } 1159 | .ui-icon-home { background-position: 0 -112px; } 1160 | .ui-icon-flag { background-position: -16px -112px; } 1161 | .ui-icon-calendar { background-position: -32px -112px; } 1162 | .ui-icon-cart { background-position: -48px -112px; } 1163 | .ui-icon-pencil { background-position: -64px -112px; } 1164 | .ui-icon-clock { background-position: -80px -112px; } 1165 | .ui-icon-disk { background-position: -96px -112px; } 1166 | .ui-icon-calculator { background-position: -112px -112px; } 1167 | .ui-icon-zoomin { background-position: -128px -112px; } 1168 | .ui-icon-zoomout { background-position: -144px -112px; } 1169 | .ui-icon-search { background-position: -160px -112px; } 1170 | .ui-icon-wrench { background-position: -176px -112px; } 1171 | .ui-icon-gear { background-position: -192px -112px; } 1172 | .ui-icon-heart { background-position: -208px -112px; } 1173 | .ui-icon-star { background-position: -224px -112px; } 1174 | .ui-icon-link { background-position: -240px -112px; } 1175 | .ui-icon-cancel { background-position: 0 -128px; } 1176 | .ui-icon-plus { background-position: -16px -128px; } 1177 | .ui-icon-plusthick { background-position: -32px -128px; } 1178 | .ui-icon-minus { background-position: -48px -128px; } 1179 | .ui-icon-minusthick { background-position: -64px -128px; } 1180 | .ui-icon-close { background-position: -80px -128px; } 1181 | .ui-icon-closethick { background-position: -96px -128px; } 1182 | .ui-icon-key { background-position: -112px -128px; } 1183 | .ui-icon-lightbulb { background-position: -128px -128px; } 1184 | .ui-icon-scissors { background-position: -144px -128px; } 1185 | .ui-icon-clipboard { background-position: -160px -128px; } 1186 | .ui-icon-copy { background-position: -176px -128px; } 1187 | .ui-icon-contact { background-position: -192px -128px; } 1188 | .ui-icon-image { background-position: -208px -128px; } 1189 | .ui-icon-video { background-position: -224px -128px; } 1190 | .ui-icon-script { background-position: -240px -128px; } 1191 | .ui-icon-alert { background-position: 0 -144px; } 1192 | .ui-icon-info { background-position: -16px -144px; } 1193 | .ui-icon-notice { background-position: -32px -144px; } 1194 | .ui-icon-help { background-position: -48px -144px; } 1195 | .ui-icon-check { background-position: -64px -144px; } 1196 | .ui-icon-bullet { background-position: -80px -144px; } 1197 | .ui-icon-radio-on { background-position: -96px -144px; } 1198 | .ui-icon-radio-off { background-position: -112px -144px; } 1199 | .ui-icon-pin-w { background-position: -128px -144px; } 1200 | .ui-icon-pin-s { background-position: -144px -144px; } 1201 | .ui-icon-play { background-position: 0 -160px; } 1202 | .ui-icon-pause { background-position: -16px -160px; } 1203 | .ui-icon-seek-next { background-position: -32px -160px; } 1204 | .ui-icon-seek-prev { background-position: -48px -160px; } 1205 | .ui-icon-seek-end { background-position: -64px -160px; } 1206 | .ui-icon-seek-start { background-position: -80px -160px; } 1207 | /* ui-icon-seek-first is deprecated, use ui-icon-seek-start instead */ 1208 | .ui-icon-seek-first { background-position: -80px -160px; } 1209 | .ui-icon-stop { background-position: -96px -160px; } 1210 | .ui-icon-eject { background-position: -112px -160px; } 1211 | .ui-icon-volume-off { background-position: -128px -160px; } 1212 | .ui-icon-volume-on { background-position: -144px -160px; } 1213 | .ui-icon-power { background-position: 0 -176px; } 1214 | .ui-icon-signal-diag { background-position: -16px -176px; } 1215 | .ui-icon-signal { background-position: -32px -176px; } 1216 | .ui-icon-battery-0 { background-position: -48px -176px; } 1217 | .ui-icon-battery-1 { background-position: -64px -176px; } 1218 | .ui-icon-battery-2 { background-position: -80px -176px; } 1219 | .ui-icon-battery-3 { background-position: -96px -176px; } 1220 | .ui-icon-circle-plus { background-position: 0 -192px; } 1221 | .ui-icon-circle-minus { background-position: -16px -192px; } 1222 | .ui-icon-circle-close { background-position: -32px -192px; } 1223 | .ui-icon-circle-triangle-e { background-position: -48px -192px; } 1224 | .ui-icon-circle-triangle-s { background-position: -64px -192px; } 1225 | .ui-icon-circle-triangle-w { background-position: -80px -192px; } 1226 | .ui-icon-circle-triangle-n { background-position: -96px -192px; } 1227 | .ui-icon-circle-arrow-e { background-position: -112px -192px; } 1228 | .ui-icon-circle-arrow-s { background-position: -128px -192px; } 1229 | .ui-icon-circle-arrow-w { background-position: -144px -192px; } 1230 | .ui-icon-circle-arrow-n { background-position: -160px -192px; } 1231 | .ui-icon-circle-zoomin { background-position: -176px -192px; } 1232 | .ui-icon-circle-zoomout { background-position: -192px -192px; } 1233 | .ui-icon-circle-check { background-position: -208px -192px; } 1234 | .ui-icon-circlesmall-plus { background-position: 0 -208px; } 1235 | .ui-icon-circlesmall-minus { background-position: -16px -208px; } 1236 | .ui-icon-circlesmall-close { background-position: -32px -208px; } 1237 | .ui-icon-squaresmall-plus { background-position: -48px -208px; } 1238 | .ui-icon-squaresmall-minus { background-position: -64px -208px; } 1239 | .ui-icon-squaresmall-close { background-position: -80px -208px; } 1240 | .ui-icon-grip-dotted-vertical { background-position: 0 -224px; } 1241 | .ui-icon-grip-dotted-horizontal { background-position: -16px -224px; } 1242 | .ui-icon-grip-solid-vertical { background-position: -32px -224px; } 1243 | .ui-icon-grip-solid-horizontal { background-position: -48px -224px; } 1244 | .ui-icon-gripsmall-diagonal-se { background-position: -64px -224px; } 1245 | .ui-icon-grip-diagonal-se { background-position: -80px -224px; } 1246 | 1247 | 1248 | /* Misc visuals 1249 | ----------------------------------*/ 1250 | 1251 | /* Corner radius */ 1252 | .ui-corner-all, 1253 | .ui-corner-top, 1254 | .ui-corner-left, 1255 | .ui-corner-tl { 1256 | border-top-left-radius: 3px; 1257 | } 1258 | .ui-corner-all, 1259 | .ui-corner-top, 1260 | .ui-corner-right, 1261 | .ui-corner-tr { 1262 | border-top-right-radius: 3px; 1263 | } 1264 | .ui-corner-all, 1265 | .ui-corner-bottom, 1266 | .ui-corner-left, 1267 | .ui-corner-bl { 1268 | border-bottom-left-radius: 3px; 1269 | } 1270 | .ui-corner-all, 1271 | .ui-corner-bottom, 1272 | .ui-corner-right, 1273 | .ui-corner-br { 1274 | border-bottom-right-radius: 3px; 1275 | } 1276 | 1277 | /* Overlays */ 1278 | .ui-widget-overlay { 1279 | background: #aaaaaa; 1280 | opacity: .3; 1281 | } 1282 | .ui-widget-shadow { 1283 | box-shadow: 0px 0px 5px #666666; 1284 | } -------------------------------------------------------------------------------- /extension/jquery-3.7.1.min.js: -------------------------------------------------------------------------------- 1 | /*! jQuery v3.7.1 | (c) OpenJS Foundation and other contributors | jquery.org/license */ 2 | !function(e,t){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=e.document?t(e,!0):function(e){if(!e.document)throw new Error("jQuery requires a window with a document");return t(e)}:t(e)}("undefined"!=typeof window?window:this,function(ie,e){"use strict";var oe=[],r=Object.getPrototypeOf,ae=oe.slice,g=oe.flat?function(e){return oe.flat.call(e)}:function(e){return oe.concat.apply([],e)},s=oe.push,se=oe.indexOf,n={},i=n.toString,ue=n.hasOwnProperty,o=ue.toString,a=o.call(Object),le={},v=function(e){return"function"==typeof e&&"number"!=typeof e.nodeType&&"function"!=typeof e.item},y=function(e){return null!=e&&e===e.window},C=ie.document,u={type:!0,src:!0,nonce:!0,noModule:!0};function m(e,t,n){var r,i,o=(n=n||C).createElement("script");if(o.text=e,t)for(r in u)(i=t[r]||t.getAttribute&&t.getAttribute(r))&&o.setAttribute(r,i);n.head.appendChild(o).parentNode.removeChild(o)}function x(e){return null==e?e+"":"object"==typeof e||"function"==typeof e?n[i.call(e)]||"object":typeof e}var t="3.7.1",l=/HTML$/i,ce=function(e,t){return new ce.fn.init(e,t)};function c(e){var t=!!e&&"length"in e&&e.length,n=x(e);return!v(e)&&!y(e)&&("array"===n||0===t||"number"==typeof t&&0+~]|"+ge+")"+ge+"*"),x=new RegExp(ge+"|>"),j=new RegExp(g),A=new RegExp("^"+t+"$"),D={ID:new RegExp("^#("+t+")"),CLASS:new RegExp("^\\.("+t+")"),TAG:new RegExp("^("+t+"|[*])"),ATTR:new RegExp("^"+p),PSEUDO:new RegExp("^"+g),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+ge+"*(even|odd|(([+-]|)(\\d*)n|)"+ge+"*(?:([+-]|)"+ge+"*(\\d+)|))"+ge+"*\\)|)","i"),bool:new RegExp("^(?:"+f+")$","i"),needsContext:new RegExp("^"+ge+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+ge+"*((?:-\\d)?\\d*)"+ge+"*\\)|)(?=[^-]|$)","i")},N=/^(?:input|select|textarea|button)$/i,q=/^h\d$/i,L=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,H=/[+~]/,O=new RegExp("\\\\[\\da-fA-F]{1,6}"+ge+"?|\\\\([^\\r\\n\\f])","g"),P=function(e,t){var n="0x"+e.slice(1)-65536;return t||(n<0?String.fromCharCode(n+65536):String.fromCharCode(n>>10|55296,1023&n|56320))},M=function(){V()},R=J(function(e){return!0===e.disabled&&fe(e,"fieldset")},{dir:"parentNode",next:"legend"});try{k.apply(oe=ae.call(ye.childNodes),ye.childNodes),oe[ye.childNodes.length].nodeType}catch(e){k={apply:function(e,t){me.apply(e,ae.call(t))},call:function(e){me.apply(e,ae.call(arguments,1))}}}function I(t,e,n,r){var i,o,a,s,u,l,c,f=e&&e.ownerDocument,p=e?e.nodeType:9;if(n=n||[],"string"!=typeof t||!t||1!==p&&9!==p&&11!==p)return n;if(!r&&(V(e),e=e||T,C)){if(11!==p&&(u=L.exec(t)))if(i=u[1]){if(9===p){if(!(a=e.getElementById(i)))return n;if(a.id===i)return k.call(n,a),n}else if(f&&(a=f.getElementById(i))&&I.contains(e,a)&&a.id===i)return k.call(n,a),n}else{if(u[2])return k.apply(n,e.getElementsByTagName(t)),n;if((i=u[3])&&e.getElementsByClassName)return k.apply(n,e.getElementsByClassName(i)),n}if(!(h[t+" "]||d&&d.test(t))){if(c=t,f=e,1===p&&(x.test(t)||m.test(t))){(f=H.test(t)&&U(e.parentNode)||e)==e&&le.scope||((s=e.getAttribute("id"))?s=ce.escapeSelector(s):e.setAttribute("id",s=S)),o=(l=Y(t)).length;while(o--)l[o]=(s?"#"+s:":scope")+" "+Q(l[o]);c=l.join(",")}try{return k.apply(n,f.querySelectorAll(c)),n}catch(e){h(t,!0)}finally{s===S&&e.removeAttribute("id")}}}return re(t.replace(ve,"$1"),e,n,r)}function W(){var r=[];return function e(t,n){return r.push(t+" ")>b.cacheLength&&delete e[r.shift()],e[t+" "]=n}}function F(e){return e[S]=!0,e}function $(e){var t=T.createElement("fieldset");try{return!!e(t)}catch(e){return!1}finally{t.parentNode&&t.parentNode.removeChild(t),t=null}}function B(t){return function(e){return fe(e,"input")&&e.type===t}}function _(t){return function(e){return(fe(e,"input")||fe(e,"button"))&&e.type===t}}function z(t){return function(e){return"form"in e?e.parentNode&&!1===e.disabled?"label"in e?"label"in e.parentNode?e.parentNode.disabled===t:e.disabled===t:e.isDisabled===t||e.isDisabled!==!t&&R(e)===t:e.disabled===t:"label"in e&&e.disabled===t}}function X(a){return F(function(o){return o=+o,F(function(e,t){var n,r=a([],e.length,o),i=r.length;while(i--)e[n=r[i]]&&(e[n]=!(t[n]=e[n]))})})}function U(e){return e&&"undefined"!=typeof e.getElementsByTagName&&e}function V(e){var t,n=e?e.ownerDocument||e:ye;return n!=T&&9===n.nodeType&&n.documentElement&&(r=(T=n).documentElement,C=!ce.isXMLDoc(T),i=r.matches||r.webkitMatchesSelector||r.msMatchesSelector,r.msMatchesSelector&&ye!=T&&(t=T.defaultView)&&t.top!==t&&t.addEventListener("unload",M),le.getById=$(function(e){return r.appendChild(e).id=ce.expando,!T.getElementsByName||!T.getElementsByName(ce.expando).length}),le.disconnectedMatch=$(function(e){return i.call(e,"*")}),le.scope=$(function(){return T.querySelectorAll(":scope")}),le.cssHas=$(function(){try{return T.querySelector(":has(*,:jqfake)"),!1}catch(e){return!0}}),le.getById?(b.filter.ID=function(e){var t=e.replace(O,P);return function(e){return e.getAttribute("id")===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&C){var n=t.getElementById(e);return n?[n]:[]}}):(b.filter.ID=function(e){var n=e.replace(O,P);return function(e){var t="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return t&&t.value===n}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&C){var n,r,i,o=t.getElementById(e);if(o){if((n=o.getAttributeNode("id"))&&n.value===e)return[o];i=t.getElementsByName(e),r=0;while(o=i[r++])if((n=o.getAttributeNode("id"))&&n.value===e)return[o]}return[]}}),b.find.TAG=function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):t.querySelectorAll(e)},b.find.CLASS=function(e,t){if("undefined"!=typeof t.getElementsByClassName&&C)return t.getElementsByClassName(e)},d=[],$(function(e){var t;r.appendChild(e).innerHTML="",e.querySelectorAll("[selected]").length||d.push("\\["+ge+"*(?:value|"+f+")"),e.querySelectorAll("[id~="+S+"-]").length||d.push("~="),e.querySelectorAll("a#"+S+"+*").length||d.push(".#.+[+~]"),e.querySelectorAll(":checked").length||d.push(":checked"),(t=T.createElement("input")).setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),r.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&d.push(":enabled",":disabled"),(t=T.createElement("input")).setAttribute("name",""),e.appendChild(t),e.querySelectorAll("[name='']").length||d.push("\\["+ge+"*name"+ge+"*="+ge+"*(?:''|\"\")")}),le.cssHas||d.push(":has"),d=d.length&&new RegExp(d.join("|")),l=function(e,t){if(e===t)return a=!0,0;var n=!e.compareDocumentPosition-!t.compareDocumentPosition;return n||(1&(n=(e.ownerDocument||e)==(t.ownerDocument||t)?e.compareDocumentPosition(t):1)||!le.sortDetached&&t.compareDocumentPosition(e)===n?e===T||e.ownerDocument==ye&&I.contains(ye,e)?-1:t===T||t.ownerDocument==ye&&I.contains(ye,t)?1:o?se.call(o,e)-se.call(o,t):0:4&n?-1:1)}),T}for(e in I.matches=function(e,t){return I(e,null,null,t)},I.matchesSelector=function(e,t){if(V(e),C&&!h[t+" "]&&(!d||!d.test(t)))try{var n=i.call(e,t);if(n||le.disconnectedMatch||e.document&&11!==e.document.nodeType)return n}catch(e){h(t,!0)}return 0":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(O,P),e[3]=(e[3]||e[4]||e[5]||"").replace(O,P),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||I.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&I.error(e[0]),e},PSEUDO:function(e){var t,n=!e[6]&&e[2];return D.CHILD.test(e[0])?null:(e[3]?e[2]=e[4]||e[5]||"":n&&j.test(n)&&(t=Y(n,!0))&&(t=n.indexOf(")",n.length-t)-n.length)&&(e[0]=e[0].slice(0,t),e[2]=n.slice(0,t)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(O,P).toLowerCase();return"*"===e?function(){return!0}:function(e){return fe(e,t)}},CLASS:function(e){var t=s[e+" "];return t||(t=new RegExp("(^|"+ge+")"+e+"("+ge+"|$)"))&&s(e,function(e){return t.test("string"==typeof e.className&&e.className||"undefined"!=typeof e.getAttribute&&e.getAttribute("class")||"")})},ATTR:function(n,r,i){return function(e){var t=I.attr(e,n);return null==t?"!="===r:!r||(t+="","="===r?t===i:"!="===r?t!==i:"^="===r?i&&0===t.indexOf(i):"*="===r?i&&-1:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i;function T(e,n,r){return v(n)?ce.grep(e,function(e,t){return!!n.call(e,t,e)!==r}):n.nodeType?ce.grep(e,function(e){return e===n!==r}):"string"!=typeof n?ce.grep(e,function(e){return-1)[^>]*|#([\w-]+))$/;(ce.fn.init=function(e,t,n){var r,i;if(!e)return this;if(n=n||k,"string"==typeof e){if(!(r="<"===e[0]&&">"===e[e.length-1]&&3<=e.length?[null,e,null]:S.exec(e))||!r[1]&&t)return!t||t.jquery?(t||n).find(e):this.constructor(t).find(e);if(r[1]){if(t=t instanceof ce?t[0]:t,ce.merge(this,ce.parseHTML(r[1],t&&t.nodeType?t.ownerDocument||t:C,!0)),w.test(r[1])&&ce.isPlainObject(t))for(r in t)v(this[r])?this[r](t[r]):this.attr(r,t[r]);return this}return(i=C.getElementById(r[2]))&&(this[0]=i,this.length=1),this}return e.nodeType?(this[0]=e,this.length=1,this):v(e)?void 0!==n.ready?n.ready(e):e(ce):ce.makeArray(e,this)}).prototype=ce.fn,k=ce(C);var E=/^(?:parents|prev(?:Until|All))/,j={children:!0,contents:!0,next:!0,prev:!0};function A(e,t){while((e=e[t])&&1!==e.nodeType);return e}ce.fn.extend({has:function(e){var t=ce(e,this),n=t.length;return this.filter(function(){for(var e=0;e\x20\t\r\n\f]*)/i,Ce=/^$|^module$|\/(?:java|ecma)script/i;xe=C.createDocumentFragment().appendChild(C.createElement("div")),(be=C.createElement("input")).setAttribute("type","radio"),be.setAttribute("checked","checked"),be.setAttribute("name","t"),xe.appendChild(be),le.checkClone=xe.cloneNode(!0).cloneNode(!0).lastChild.checked,xe.innerHTML="",le.noCloneChecked=!!xe.cloneNode(!0).lastChild.defaultValue,xe.innerHTML="",le.option=!!xe.lastChild;var ke={thead:[1,"","
"],col:[2,"","
"],tr:[2,"","
"],td:[3,"","
"],_default:[0,"",""]};function Se(e,t){var n;return n="undefined"!=typeof e.getElementsByTagName?e.getElementsByTagName(t||"*"):"undefined"!=typeof e.querySelectorAll?e.querySelectorAll(t||"*"):[],void 0===t||t&&fe(e,t)?ce.merge([e],n):n}function Ee(e,t){for(var n=0,r=e.length;n",""]);var je=/<|&#?\w+;/;function Ae(e,t,n,r,i){for(var o,a,s,u,l,c,f=t.createDocumentFragment(),p=[],d=0,h=e.length;d\s*$/g;function Re(e,t){return fe(e,"table")&&fe(11!==t.nodeType?t:t.firstChild,"tr")&&ce(e).children("tbody")[0]||e}function Ie(e){return e.type=(null!==e.getAttribute("type"))+"/"+e.type,e}function We(e){return"true/"===(e.type||"").slice(0,5)?e.type=e.type.slice(5):e.removeAttribute("type"),e}function Fe(e,t){var n,r,i,o,a,s;if(1===t.nodeType){if(_.hasData(e)&&(s=_.get(e).events))for(i in _.remove(t,"handle events"),s)for(n=0,r=s[i].length;n").attr(n.scriptAttrs||{}).prop({charset:n.scriptCharset,src:n.url}).on("load error",i=function(e){r.remove(),i=null,e&&t("error"===e.type?404:200,e.type)}),C.head.appendChild(r[0])},abort:function(){i&&i()}}});var Jt,Kt=[],Zt=/(=)\?(?=&|$)|\?\?/;ce.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=Kt.pop()||ce.expando+"_"+jt.guid++;return this[e]=!0,e}}),ce.ajaxPrefilter("json jsonp",function(e,t,n){var r,i,o,a=!1!==e.jsonp&&(Zt.test(e.url)?"url":"string"==typeof e.data&&0===(e.contentType||"").indexOf("application/x-www-form-urlencoded")&&Zt.test(e.data)&&"data");if(a||"jsonp"===e.dataTypes[0])return r=e.jsonpCallback=v(e.jsonpCallback)?e.jsonpCallback():e.jsonpCallback,a?e[a]=e[a].replace(Zt,"$1"+r):!1!==e.jsonp&&(e.url+=(At.test(e.url)?"&":"?")+e.jsonp+"="+r),e.converters["script json"]=function(){return o||ce.error(r+" was not called"),o[0]},e.dataTypes[0]="json",i=ie[r],ie[r]=function(){o=arguments},n.always(function(){void 0===i?ce(ie).removeProp(r):ie[r]=i,e[r]&&(e.jsonpCallback=t.jsonpCallback,Kt.push(r)),o&&v(i)&&i(o[0]),o=i=void 0}),"script"}),le.createHTMLDocument=((Jt=C.implementation.createHTMLDocument("").body).innerHTML="
",2===Jt.childNodes.length),ce.parseHTML=function(e,t,n){return"string"!=typeof e?[]:("boolean"==typeof t&&(n=t,t=!1),t||(le.createHTMLDocument?((r=(t=C.implementation.createHTMLDocument("")).createElement("base")).href=C.location.href,t.head.appendChild(r)):t=C),o=!n&&[],(i=w.exec(e))?[t.createElement(i[1])]:(i=Ae([e],t,o),o&&o.length&&ce(o).remove(),ce.merge([],i.childNodes)));var r,i,o},ce.fn.load=function(e,t,n){var r,i,o,a=this,s=e.indexOf(" ");return-1").append(ce.parseHTML(e)).find(r):e)}).always(n&&function(e,t){a.each(function(){n.apply(this,o||[e.responseText,t,e])})}),this},ce.expr.pseudos.animated=function(t){return ce.grep(ce.timers,function(e){return t===e.elem}).length},ce.offset={setOffset:function(e,t,n){var r,i,o,a,s,u,l=ce.css(e,"position"),c=ce(e),f={};"static"===l&&(e.style.position="relative"),s=c.offset(),o=ce.css(e,"top"),u=ce.css(e,"left"),("absolute"===l||"fixed"===l)&&-1<(o+u).indexOf("auto")?(a=(r=c.position()).top,i=r.left):(a=parseFloat(o)||0,i=parseFloat(u)||0),v(t)&&(t=t.call(e,n,ce.extend({},s))),null!=t.top&&(f.top=t.top-s.top+a),null!=t.left&&(f.left=t.left-s.left+i),"using"in t?t.using.call(e,f):c.css(f)}},ce.fn.extend({offset:function(t){if(arguments.length)return void 0===t?this:this.each(function(e){ce.offset.setOffset(this,t,e)});var e,n,r=this[0];return r?r.getClientRects().length?(e=r.getBoundingClientRect(),n=r.ownerDocument.defaultView,{top:e.top+n.pageYOffset,left:e.left+n.pageXOffset}):{top:0,left:0}:void 0},position:function(){if(this[0]){var e,t,n,r=this[0],i={top:0,left:0};if("fixed"===ce.css(r,"position"))t=r.getBoundingClientRect();else{t=this.offset(),n=r.ownerDocument,e=r.offsetParent||n.documentElement;while(e&&(e===n.body||e===n.documentElement)&&"static"===ce.css(e,"position"))e=e.parentNode;e&&e!==r&&1===e.nodeType&&((i=ce(e).offset()).top+=ce.css(e,"borderTopWidth",!0),i.left+=ce.css(e,"borderLeftWidth",!0))}return{top:t.top-i.top-ce.css(r,"marginTop",!0),left:t.left-i.left-ce.css(r,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var e=this.offsetParent;while(e&&"static"===ce.css(e,"position"))e=e.offsetParent;return e||J})}}),ce.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(t,i){var o="pageYOffset"===i;ce.fn[t]=function(e){return M(this,function(e,t,n){var r;if(y(e)?r=e:9===e.nodeType&&(r=e.defaultView),void 0===n)return r?r[i]:e[t];r?r.scrollTo(o?r.pageXOffset:n,o?n:r.pageYOffset):e[t]=n},t,e,arguments.length)}}),ce.each(["top","left"],function(e,n){ce.cssHooks[n]=Ye(le.pixelPosition,function(e,t){if(t)return t=Ge(e,n),_e.test(t)?ce(e).position()[n]+"px":t})}),ce.each({Height:"height",Width:"width"},function(a,s){ce.each({padding:"inner"+a,content:s,"":"outer"+a},function(r,o){ce.fn[o]=function(e,t){var n=arguments.length&&(r||"boolean"!=typeof e),i=r||(!0===e||!0===t?"margin":"border");return M(this,function(e,t,n){var r;return y(e)?0===o.indexOf("outer")?e["inner"+a]:e.document.documentElement["client"+a]:9===e.nodeType?(r=e.documentElement,Math.max(e.body["scroll"+a],r["scroll"+a],e.body["offset"+a],r["offset"+a],r["client"+a])):void 0===n?ce.css(e,t,i):ce.style(e,t,n,i)},s,n?e:void 0,n)}})}),ce.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(e,t){ce.fn[t]=function(e){return this.on(t,e)}}),ce.fn.extend({bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return 1===arguments.length?this.off(e,"**"):this.off(t,e||"**",n)},hover:function(e,t){return this.on("mouseenter",e).on("mouseleave",t||e)}}),ce.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(e,n){ce.fn[n]=function(e,t){return 0