Upload folder using huggingface_hub
Browse files- README.md +26 -0
- standalone/front/index.html +63 -7
- standalone/front/scripts/events.js +9 -0
- standalone/front/scripts/main.js +2 -0
- standalone/front/scripts/presets.js +47 -0
- standalone/front/scripts/sessions.js +27 -1
- standalone/front/styles/style.css +59 -0
README.md
CHANGED
@@ -351,3 +351,29 @@ $ python main.py # or gradio main.py
|
|
351 |
|
352 |
# Acknowledgments
|
353 |
This is a project built during the Vertex sprints held by Google's ML Developer Programs team. We are thankful to be granted good amount of GCP credits to do this project.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
351 |
|
352 |
# Acknowledgments
|
353 |
This is a project built during the Vertex sprints held by Google's ML Developer Programs team. We are thankful to be granted good amount of GCP credits to do this project.
|
354 |
+
# AdaptSum
|
355 |
+
|
356 |
+
[](https://github.com/deep-diver/AdaptSum/actions/workflows/sync_to_spaces.yml)
|
357 |
+
|
358 |
+
AdaptSum stands for Adaptive Summarization. This project focuses on developing an LLM-powered system for dynamic summarization. Instead of generating entirely new summaries with each update, the system intelligently identifies and modifies only the necessary parts of the existing summary. This approach aims to create a more efficient and fluid summarization process within a continuous chat interaction with an LLM.
|
359 |
+
|
360 |
+
# Instructions
|
361 |
+
|
362 |
+
1. Install dependencies
|
363 |
+
```shell
|
364 |
+
$ pip install requirements.txt
|
365 |
+
```
|
366 |
+
|
367 |
+
2. Setup Gemini API Key
|
368 |
+
```shell
|
369 |
+
$ export GEMINI_API_KEY=xxxxx
|
370 |
+
```
|
371 |
+
> note that GEMINI API KEY should be obtained from Google AI Studio. Vertex AI is not supported at the moment (this is because Gemini SDK does not provide file uploading functionality for Vertex AI usage now).
|
372 |
+
|
373 |
+
3. Run Gradio app
|
374 |
+
```shell
|
375 |
+
$ python main.py # or gradio main.py
|
376 |
+
```
|
377 |
+
|
378 |
+
# Acknowledgments
|
379 |
+
This is a project built during the Vertex sprints held by Google's ML Developer Programs team. We are thankful to be granted good amount of GCP credits to do this project.
|
standalone/front/index.html
CHANGED
@@ -3,7 +3,7 @@
|
|
3 |
<head>
|
4 |
<meta charset="UTF-8" />
|
5 |
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
6 |
-
<title>Chat UI
|
7 |
<link href="https://fonts.googleapis.com/css2?family=Poppins:wght@400;500&display=swap" rel="stylesheet">
|
8 |
<!-- Include marked.js for Markdown rendering -->
|
9 |
<script src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
|
@@ -26,6 +26,61 @@
|
|
26 |
</div>
|
27 |
<!-- <h3>Chat History</h3> -->
|
28 |
<ul id="sessionList"></ul>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
29 |
</div>
|
30 |
|
31 |
<!-- Main Chat Wrapper -->
|
@@ -64,7 +119,7 @@
|
|
64 |
<div class="input-container">
|
65 |
<div class="button-row">
|
66 |
<button id="customBtn1">TLDR</button>
|
67 |
-
<button id="customBtn3" class="toggle-btn-summarize"><img src="assets/summarize.svg" alt="Icon" class="svg-icon"></button>
|
68 |
<button id="customBtn2"><img src="assets/settings.svg" alt="Icon" class="svg-icon"></button>
|
69 |
</div>
|
70 |
<div id="fileAttachments" class="file-attachments"></div>
|
@@ -114,12 +169,12 @@
|
|
114 |
<div class="settings-group">
|
115 |
<select id="modelSelect" name="modelSelect">
|
116 |
<optgroup label="OpenAI">
|
117 |
-
<option value="gpt-4o">
|
118 |
-
<option value="gpt-4o-mini">
|
119 |
</optgroup>
|
120 |
<optgroup label="Anthropic">
|
121 |
-
<option value="claude-3.5-sonnet">
|
122 |
-
<option value="claude-3.7-sonnet">
|
123 |
</optgroup>
|
124 |
<optgroup label="Google">
|
125 |
<option value="gemini-2.0-flash">Gemini 2.0 Flash</option>
|
@@ -133,7 +188,8 @@
|
|
133 |
<optgroup label="Mistral">
|
134 |
<option value="mistral-large-latest">Mistral Large</option>
|
135 |
<option value="mistral-codestral-latest">CodeStral</option>
|
136 |
-
<option value="mistral-ministral-8b-latest">
|
|
|
137 |
</optgroup>
|
138 |
</select>
|
139 |
</div>
|
|
|
3 |
<head>
|
4 |
<meta charset="UTF-8" />
|
5 |
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
6 |
+
<title>Unimaginable Chat UI</title>
|
7 |
<link href="https://fonts.googleapis.com/css2?family=Poppins:wght@400;500&display=swap" rel="stylesheet">
|
8 |
<!-- Include marked.js for Markdown rendering -->
|
9 |
<script src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
|
|
|
26 |
</div>
|
27 |
<!-- <h3>Chat History</h3> -->
|
28 |
<ul id="sessionList"></ul>
|
29 |
+
<div id="presetContainer">
|
30 |
+
<div class="preset-header">Model Selection Presets</div>
|
31 |
+
<select id="preset1" class="preset-select" title="Command + Shift + 1">
|
32 |
+
<optgroup label="OpenAI">
|
33 |
+
<option value="gpt-4o">GPT-4o</option>
|
34 |
+
<option value="gpt-4o-mini">GPT-4o Mini</option>
|
35 |
+
</optgroup>
|
36 |
+
<optgroup label="Anthropic">
|
37 |
+
<option value="claude-3.5-sonnet">Claude 3.5 Sonnet</option>
|
38 |
+
<option value="claude-3.7-sonnet">Claude 3.7 Sonnet</option>
|
39 |
+
</optgroup>
|
40 |
+
<optgroup label="Google">
|
41 |
+
<option value="gemini-2.0-flash">Gemini 2.0 Flash</option>
|
42 |
+
<option value="gemini-2.0-flash-lite">Gemini 2.0 Flash Lite</option>
|
43 |
+
</optgroup>
|
44 |
+
<optgroup label="Hugging Face">
|
45 |
+
<option value="huggingface/meta-llama/Llama-3.3-70B-Instruct">Llama 3.3 70B Instruct</option>
|
46 |
+
<option value="huggingface/deepseek-ai/DeepSeek-R1-Distill-Qwen-32B">DeepSeek R1 Distill Qwen 32B</option>
|
47 |
+
<option value="huggingface/Qwen/Qwen2.5-72B-Instruct">Qwen 2.5 72B Instruct</option>
|
48 |
+
</optgroup>
|
49 |
+
<optgroup label="Mistral">
|
50 |
+
<option value="mistral-large-latest">Mistral Large</option>
|
51 |
+
<option value="mistral-codestral-latest">CodeStral</option>
|
52 |
+
<option value="mistral-ministral-8b-latest">Ministral 8B</option>
|
53 |
+
<option value="mistral-ministral-3b-latest">Ministral 3B</option>
|
54 |
+
</optgroup>
|
55 |
+
</select>
|
56 |
+
|
57 |
+
<select id="preset2" class="preset-select" title="Command + Shift + 2">
|
58 |
+
<!-- Use the same options as above -->
|
59 |
+
<optgroup label="OpenAI">
|
60 |
+
<option value="gpt-4o">GPT-4o</option>
|
61 |
+
<option value="gpt-4o-mini">GPT-4o Mini</option>
|
62 |
+
</optgroup>
|
63 |
+
<optgroup label="Anthropic">
|
64 |
+
<option value="claude-3.5-sonnet">Claude 3.5 Sonnet</option>
|
65 |
+
<option value="claude-3.7-sonnet">Claude 3.7 Sonnet</option>
|
66 |
+
</optgroup>
|
67 |
+
<optgroup label="Google">
|
68 |
+
<option value="gemini-2.0-flash">Gemini 2.0 Flash</option>
|
69 |
+
<option value="gemini-2.0-flash-lite">Gemini 2.0 Flash Lite</option>
|
70 |
+
</optgroup>
|
71 |
+
<optgroup label="Hugging Face">
|
72 |
+
<option value="huggingface/meta-llama/Llama-3.3-70B-Instruct">Llama 3.3 70B Instruct</option>
|
73 |
+
<option value="huggingface/deepseek-ai/DeepSeek-R1-Distill-Qwen-32B">DeepSeek R1 Distill Qwen 32B</option>
|
74 |
+
<option value="huggingface/Qwen/Qwen2.5-72B-Instruct">Qwen 2.5 72B Instruct</option>
|
75 |
+
</optgroup>
|
76 |
+
<optgroup label="Mistral">
|
77 |
+
<option value="mistral-large-latest">Mistral Large</option>
|
78 |
+
<option value="mistral-codestral-latest">CodeStral</option>
|
79 |
+
<option value="mistral-ministral-8b-latest">Ministral 8B</option>
|
80 |
+
<option value="mistral-ministral-3b-latest">Ministral 3B</option>
|
81 |
+
</optgroup>
|
82 |
+
</select>
|
83 |
+
</div>
|
84 |
</div>
|
85 |
|
86 |
<!-- Main Chat Wrapper -->
|
|
|
119 |
<div class="input-container">
|
120 |
<div class="button-row">
|
121 |
<button id="customBtn1">TLDR</button>
|
122 |
+
<button id="customBtn3" class="toggle-btn-summarize" title="Toggle summarization"><img src="assets/summarize.svg" alt="Icon" class="svg-icon"></button>
|
123 |
<button id="customBtn2"><img src="assets/settings.svg" alt="Icon" class="svg-icon"></button>
|
124 |
</div>
|
125 |
<div id="fileAttachments" class="file-attachments"></div>
|
|
|
169 |
<div class="settings-group">
|
170 |
<select id="modelSelect" name="modelSelect">
|
171 |
<optgroup label="OpenAI">
|
172 |
+
<option value="gpt-4o">GPT-4o</option>
|
173 |
+
<option value="gpt-4o-mini">GPT-4o Mini</option>
|
174 |
</optgroup>
|
175 |
<optgroup label="Anthropic">
|
176 |
+
<option value="claude-3.5-sonnet">Claude 3.5 Sonnet</option>
|
177 |
+
<option value="claude-3.7-sonnet">Claude 3.7 Sonnet</option>
|
178 |
</optgroup>
|
179 |
<optgroup label="Google">
|
180 |
<option value="gemini-2.0-flash">Gemini 2.0 Flash</option>
|
|
|
188 |
<optgroup label="Mistral">
|
189 |
<option value="mistral-large-latest">Mistral Large</option>
|
190 |
<option value="mistral-codestral-latest">CodeStral</option>
|
191 |
+
<option value="mistral-ministral-8b-latest">Ministral 8B</option>
|
192 |
+
<option value="mistral-ministral-3b-latest">Ministral 3B</option>
|
193 |
</optgroup>
|
194 |
</select>
|
195 |
</div>
|
standalone/front/scripts/events.js
CHANGED
@@ -120,6 +120,15 @@ saveSettingsBtn.addEventListener('click', () => {
|
|
120 |
sessionSettings.model = modelSelect.value;
|
121 |
console.log('Session settings saved:', sessions[currentSessionIndex].settings);
|
122 |
settingsOverlay.classList.remove('active');
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
123 |
});
|
124 |
const editTitleBtn = document.getElementById('editTitleBtn');
|
125 |
editTitleBtn.addEventListener('click', () => {
|
|
|
120 |
sessionSettings.model = modelSelect.value;
|
121 |
console.log('Session settings saved:', sessions[currentSessionIndex].settings);
|
122 |
settingsOverlay.classList.remove('active');
|
123 |
+
|
124 |
+
const preset1 = document.getElementById('preset1');
|
125 |
+
const preset2 = document.getElementById('preset2');
|
126 |
+
|
127 |
+
preset1.value = "gpt-4o-mini";
|
128 |
+
preset2.value = "gpt-4o-mini";
|
129 |
+
|
130 |
+
preset2.classList.remove('active');
|
131 |
+
preset1.classList.remove('active');
|
132 |
});
|
133 |
const editTitleBtn = document.getElementById('editTitleBtn');
|
134 |
editTitleBtn.addEventListener('click', () => {
|
standalone/front/scripts/main.js
CHANGED
@@ -1,6 +1,7 @@
|
|
1 |
// main.js
|
2 |
import { initSessions, sessions, currentSessionIndex, renderCurrentSession } from './sessions.js';
|
3 |
import { updateHamburgerPosition } from './navigation.js';
|
|
|
4 |
|
5 |
// Import events so they register their listeners.
|
6 |
import './events.js';
|
@@ -10,6 +11,7 @@ import './utils.js';
|
|
10 |
// Initialize sessions
|
11 |
initSessions();
|
12 |
updateHamburgerPosition();
|
|
|
13 |
|
14 |
// Attach sessions-related variables to the window for global access
|
15 |
window.sessions = sessions;
|
|
|
1 |
// main.js
|
2 |
import { initSessions, sessions, currentSessionIndex, renderCurrentSession } from './sessions.js';
|
3 |
import { updateHamburgerPosition } from './navigation.js';
|
4 |
+
import { initPresets } from './presets.js';
|
5 |
|
6 |
// Import events so they register their listeners.
|
7 |
import './events.js';
|
|
|
11 |
// Initialize sessions
|
12 |
initSessions();
|
13 |
updateHamburgerPosition();
|
14 |
+
initPresets();
|
15 |
|
16 |
// Attach sessions-related variables to the window for global access
|
17 |
window.sessions = sessions;
|
standalone/front/scripts/presets.js
ADDED
@@ -0,0 +1,47 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import { renderCurrentSession, sessions, currentSessionIndex } from './sessions.js';
|
2 |
+
|
3 |
+
// presets.js
|
4 |
+
export function initPresets() {
|
5 |
+
const presetContainer = document.getElementById('presetContainer');
|
6 |
+
if (!presetContainer) return;
|
7 |
+
|
8 |
+
const preset1 = document.getElementById('preset1');
|
9 |
+
const preset2 = document.getElementById('preset2');
|
10 |
+
const modelSelect = document.getElementById('modelSelect');
|
11 |
+
|
12 |
+
preset1.value = "gpt-4o-mini";
|
13 |
+
preset2.value = "gpt-4o-mini";
|
14 |
+
|
15 |
+
// Keyboard shortcuts: Ctrl/Cmd + Shift + 1, 2, 3
|
16 |
+
document.addEventListener('keydown', (e) => {
|
17 |
+
if ((e.ctrlKey || e.metaKey) && e.shiftKey && !e.altKey) {
|
18 |
+
switch (e.key) {
|
19 |
+
case '1':
|
20 |
+
preset2.classList.remove('active');
|
21 |
+
preset1.classList.add('active');
|
22 |
+
modelSelect.value = preset1.value;
|
23 |
+
window.sessions[currentSessionIndex].settings.model = preset1.value;
|
24 |
+
break;
|
25 |
+
case '2':
|
26 |
+
preset1.classList.remove('active');
|
27 |
+
preset2.classList.add('active');
|
28 |
+
modelSelect.value = preset2.value;
|
29 |
+
window.sessions[currentSessionIndex].settings.model = preset2.value;
|
30 |
+
break;
|
31 |
+
}
|
32 |
+
}
|
33 |
+
});
|
34 |
+
|
35 |
+
preset1.addEventListener('change', () => {
|
36 |
+
console.log(currentSessionIndex);
|
37 |
+
window.sessions[currentSessionIndex].settings.model_preset1 = preset1.value;
|
38 |
+
console.log(window.sessions[currentSessionIndex].settings.model_preset1);
|
39 |
+
});
|
40 |
+
|
41 |
+
preset2.addEventListener('change', () => {
|
42 |
+
console.log(currentSessionIndex);
|
43 |
+
window.sessions[currentSessionIndex].settings.model_preset2 = preset2.value;
|
44 |
+
console.log(window.sessions[currentSessionIndex].settings.model_preset2);
|
45 |
+
});
|
46 |
+
}
|
47 |
+
|
standalone/front/scripts/sessions.js
CHANGED
@@ -20,6 +20,8 @@ export function initSessions() {
|
|
20 |
maxTokens: 8096,
|
21 |
persona: "professional",
|
22 |
model: "gpt-4o-mini",
|
|
|
|
|
23 |
enableSummarization: false,
|
24 |
},
|
25 |
});
|
@@ -52,6 +54,19 @@ export function renderSessionList() {
|
|
52 |
currentCardIndex = 0;
|
53 |
renderSessionList();
|
54 |
renderCurrentSession();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
55 |
});
|
56 |
if (index === currentSessionIndex) li.classList.add('active');
|
57 |
sessionList.appendChild(li);
|
@@ -212,8 +227,10 @@ document.getElementById('newSessionBtn').addEventListener('click', () => {
|
|
212 |
maxTokens: 8096,
|
213 |
persona: "professional",
|
214 |
model: "gpt-4o-mini",
|
|
|
|
|
215 |
enableSummarization: false
|
216 |
-
}
|
217 |
};
|
218 |
|
219 |
sessions.push(newSession);
|
@@ -224,4 +241,13 @@ document.getElementById('newSessionBtn').addEventListener('click', () => {
|
|
224 |
|
225 |
renderSessionList();
|
226 |
renderCurrentSession();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
227 |
});
|
|
|
20 |
maxTokens: 8096,
|
21 |
persona: "professional",
|
22 |
model: "gpt-4o-mini",
|
23 |
+
model_preset1: "gpt-4o-mini",
|
24 |
+
model_preset2: "gpt-4o-mini",
|
25 |
enableSummarization: false,
|
26 |
},
|
27 |
});
|
|
|
54 |
currentCardIndex = 0;
|
55 |
renderSessionList();
|
56 |
renderCurrentSession();
|
57 |
+
|
58 |
+
console.log(currentSessionIndex);
|
59 |
+
|
60 |
+
const preset1 = document.getElementById('preset1');
|
61 |
+
const preset2 = document.getElementById('preset2');
|
62 |
+
|
63 |
+
preset1.value = sessions[currentSessionIndex].settings.model_preset1;
|
64 |
+
preset2.value = sessions[currentSessionIndex].settings.model_preset2;
|
65 |
+
console.log(preset1.value);
|
66 |
+
console.log(preset2.value);
|
67 |
+
|
68 |
+
preset2.classList.remove('active');
|
69 |
+
preset1.classList.remove('active');
|
70 |
});
|
71 |
if (index === currentSessionIndex) li.classList.add('active');
|
72 |
sessionList.appendChild(li);
|
|
|
227 |
maxTokens: 8096,
|
228 |
persona: "professional",
|
229 |
model: "gpt-4o-mini",
|
230 |
+
model_preset1: "gpt-4o-mini",
|
231 |
+
model_preset2: "gpt-4o-mini",
|
232 |
enableSummarization: false
|
233 |
+
}
|
234 |
};
|
235 |
|
236 |
sessions.push(newSession);
|
|
|
241 |
|
242 |
renderSessionList();
|
243 |
renderCurrentSession();
|
244 |
+
|
245 |
+
const preset1 = document.getElementById('preset1');
|
246 |
+
const preset2 = document.getElementById('preset2');
|
247 |
+
|
248 |
+
preset1.value = "gpt-4o-mini";
|
249 |
+
preset2.value = "gpt-4o-mini";
|
250 |
+
|
251 |
+
preset2.classList.remove('active');
|
252 |
+
preset1.classList.remove('active');
|
253 |
});
|
standalone/front/styles/style.css
CHANGED
@@ -65,6 +65,7 @@
|
|
65 |
display: flex;
|
66 |
flex-direction: column;
|
67 |
}
|
|
|
68 |
.nav-bar.hidden {
|
69 |
transform: translateX(-100%);
|
70 |
width: 0;
|
@@ -993,3 +994,61 @@
|
|
993 |
.toggle-btn-summarize.active {
|
994 |
background-color: #4a90e2 !important; /* Vivid color when active */
|
995 |
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
65 |
display: flex;
|
66 |
flex-direction: column;
|
67 |
}
|
68 |
+
|
69 |
.nav-bar.hidden {
|
70 |
transform: translateX(-100%);
|
71 |
width: 0;
|
|
|
994 |
.toggle-btn-summarize.active {
|
995 |
background-color: #4a90e2 !important; /* Vivid color when active */
|
996 |
}
|
997 |
+
|
998 |
+
/* Model Presets */
|
999 |
+
/* Container for presets */
|
1000 |
+
#presetContainer {
|
1001 |
+
flex-shrink: 0;
|
1002 |
+
padding: 10px 24px;
|
1003 |
+
border-top: 1px solid #ddd;
|
1004 |
+
background-color: #fff;
|
1005 |
+
}
|
1006 |
+
|
1007 |
+
/* Improved preset select styling */
|
1008 |
+
#presetContainer {
|
1009 |
+
position: absolute;
|
1010 |
+
bottom: 20px; /* same as nav-bar padding */
|
1011 |
+
left: 20px;
|
1012 |
+
right: 20px;
|
1013 |
+
background: #fff;
|
1014 |
+
padding: 10px;
|
1015 |
+
border-top: 1px solid #ddd;
|
1016 |
+
}
|
1017 |
+
|
1018 |
+
/* Enhanced preset select styling */
|
1019 |
+
.preset-header {
|
1020 |
+
text-align: center;
|
1021 |
+
margin-bottom: 10px;
|
1022 |
+
}
|
1023 |
+
|
1024 |
+
|
1025 |
+
.preset-select {
|
1026 |
+
appearance: none;
|
1027 |
+
-webkit-appearance: none;
|
1028 |
+
-moz-appearance: none;
|
1029 |
+
width: 100%;
|
1030 |
+
padding: 12px 16px;
|
1031 |
+
margin-bottom: 10px;
|
1032 |
+
font-size: 1em;
|
1033 |
+
border: 1px solid #ccc;
|
1034 |
+
border-radius: 8px;
|
1035 |
+
background-color: #fafafa;
|
1036 |
+
background-image: url('data:image/svg+xml;utf8,<svg fill="gray" height="24" viewBox="0 0 24 24" width="24" xmlns="http://www.w3.org/2000/svg"><path d="M7 10l5 5 5-5z"/></svg>');
|
1037 |
+
background-repeat: no-repeat;
|
1038 |
+
background-position: right 12px center;
|
1039 |
+
transition: background-color 0.3s ease, border-color 0.3s ease, box-shadow 0.3s ease;
|
1040 |
+
cursor: pointer;
|
1041 |
+
}
|
1042 |
+
|
1043 |
+
.preset-select.active {
|
1044 |
+
background-color: #4a90e2;
|
1045 |
+
color: #fff;
|
1046 |
+
border-color: #4a90e2;
|
1047 |
+
box-shadow: 0 0 8px rgba(74, 144, 226, 0.3);
|
1048 |
+
outline: none;
|
1049 |
+
}
|
1050 |
+
|
1051 |
+
#sessionList {
|
1052 |
+
overflow-y: auto;
|
1053 |
+
padding-bottom: 150px; /* Adjust this value as needed */
|
1054 |
+
}
|