Upload folder using huggingface_hub
Browse files- README.md +24 -0
- ui-temp.html +819 -0
README.md
CHANGED
@@ -203,3 +203,27 @@ $ python main.py # or gradio main.py
|
|
203 |
|
204 |
# Acknowledgments
|
205 |
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.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
203 |
|
204 |
# Acknowledgments
|
205 |
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.
|
206 |
+
# AdaptSum
|
207 |
+
|
208 |
+
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.
|
209 |
+
|
210 |
+
# Instructions
|
211 |
+
|
212 |
+
1. Install dependencies
|
213 |
+
```shell
|
214 |
+
$ pip install requirements.txt
|
215 |
+
```
|
216 |
+
|
217 |
+
2. Setup Gemini API Key
|
218 |
+
```shell
|
219 |
+
$ export GEMINI_API_KEY=xxxxx
|
220 |
+
```
|
221 |
+
> 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).
|
222 |
+
|
223 |
+
3. Run Gradio app
|
224 |
+
```shell
|
225 |
+
$ python main.py # or gradio main.py
|
226 |
+
```
|
227 |
+
|
228 |
+
# Acknowledgments
|
229 |
+
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.
|
ui-temp.html
ADDED
@@ -0,0 +1,819 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="UTF-8" />
|
5 |
+
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
6 |
+
<title>Chat UI with Per‑Session Summary, Settings & History</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>
|
10 |
+
<style>
|
11 |
+
/* Global Styles */
|
12 |
+
* { box-sizing: border-box; }
|
13 |
+
body {
|
14 |
+
margin: 0;
|
15 |
+
padding: 0;
|
16 |
+
font-family: 'Poppins', sans-serif;
|
17 |
+
background: #e0e0e0;
|
18 |
+
overflow: hidden;
|
19 |
+
}
|
20 |
+
/* Overall App Container: Left Nav and Chat Area */
|
21 |
+
.app-container {
|
22 |
+
display: flex;
|
23 |
+
width: 100vw;
|
24 |
+
height: 100vh;
|
25 |
+
}
|
26 |
+
/* Left Navigation Bar for Chat History */
|
27 |
+
.nav-bar {
|
28 |
+
width: 250px;
|
29 |
+
background: #ffffff;
|
30 |
+
padding: 20px;
|
31 |
+
box-shadow: 2px 0 12px rgba(0,0,0,0.1);
|
32 |
+
overflow-y: auto;
|
33 |
+
}
|
34 |
+
.nav-bar h3 {
|
35 |
+
margin-top: 0;
|
36 |
+
font-size: 1.3em;
|
37 |
+
}
|
38 |
+
.nav-bar ul {
|
39 |
+
list-style: none;
|
40 |
+
padding: 0;
|
41 |
+
margin: 0;
|
42 |
+
}
|
43 |
+
.nav-bar li {
|
44 |
+
padding: 10px;
|
45 |
+
margin-bottom: 10px;
|
46 |
+
background: #fff;
|
47 |
+
border-radius: 8px;
|
48 |
+
cursor: pointer;
|
49 |
+
display: flex;
|
50 |
+
justify-content: space-between;
|
51 |
+
align-items: center;
|
52 |
+
transition: background 0.3s;
|
53 |
+
font-size: 1.1em;
|
54 |
+
}
|
55 |
+
.nav-bar li.active,
|
56 |
+
.nav-bar li:hover {
|
57 |
+
background: #667eea;
|
58 |
+
color: #fff;
|
59 |
+
}
|
60 |
+
.nav-bar li button.remove-session {
|
61 |
+
background: transparent;
|
62 |
+
border: none;
|
63 |
+
color: inherit;
|
64 |
+
font-size: 1em;
|
65 |
+
cursor: pointer;
|
66 |
+
}
|
67 |
+
.new-session-btn {
|
68 |
+
width: 100%;
|
69 |
+
padding: 10px;
|
70 |
+
background: #667eea;
|
71 |
+
color: #fff;
|
72 |
+
border: none;
|
73 |
+
border-radius: 8px;
|
74 |
+
cursor: pointer;
|
75 |
+
font-size: 1.1em;
|
76 |
+
margin-top: 10px;
|
77 |
+
}
|
78 |
+
/* Chat Wrapper */
|
79 |
+
.chat-wrapper {
|
80 |
+
flex: 1;
|
81 |
+
background: #fff;
|
82 |
+
margin: 20px;
|
83 |
+
border-radius: 20px;
|
84 |
+
box-shadow: 0 12px 40px rgba(0,0,0,0.15);
|
85 |
+
display: flex;
|
86 |
+
flex-direction: column;
|
87 |
+
overflow: hidden;
|
88 |
+
position: relative;
|
89 |
+
}
|
90 |
+
/* Top Buttons for Overlays */
|
91 |
+
.top-buttons {
|
92 |
+
position: absolute;
|
93 |
+
top: 20px;
|
94 |
+
left: 50%;
|
95 |
+
transform: translateX(-50%);
|
96 |
+
display: flex;
|
97 |
+
gap: 10px;
|
98 |
+
z-index: 15;
|
99 |
+
}
|
100 |
+
.top-button {
|
101 |
+
height: 40px;
|
102 |
+
background: #667eea;
|
103 |
+
color: #fff;
|
104 |
+
border: none;
|
105 |
+
border-radius: 20px;
|
106 |
+
cursor: pointer;
|
107 |
+
box-shadow: 0 2px 6px rgba(0,0,0,0.2);
|
108 |
+
display: flex;
|
109 |
+
align-items: center;
|
110 |
+
justify-content: center;
|
111 |
+
font-size: 1.1em;
|
112 |
+
text-transform: uppercase;
|
113 |
+
padding: 0 20px;
|
114 |
+
}
|
115 |
+
.top-button:hover { background: #556cd6; }
|
116 |
+
/* Turn Label */
|
117 |
+
#turnLabel {
|
118 |
+
position: absolute;
|
119 |
+
top: 20px;
|
120 |
+
right: 20px;
|
121 |
+
background: #ff9800;
|
122 |
+
color: #fff;
|
123 |
+
padding: 5px 10px;
|
124 |
+
border-radius: 10px;
|
125 |
+
font-size: 1em;
|
126 |
+
z-index: 15;
|
127 |
+
}
|
128 |
+
/* Carousel Area */
|
129 |
+
.carousel-wrapper {
|
130 |
+
position: relative;
|
131 |
+
flex: 1;
|
132 |
+
background: #F9FBFD;
|
133 |
+
overflow: hidden;
|
134 |
+
margin-top: 80px;
|
135 |
+
}
|
136 |
+
.carousel {
|
137 |
+
display: flex;
|
138 |
+
height: 100%;
|
139 |
+
transition: transform 0.5s ease;
|
140 |
+
}
|
141 |
+
.card {
|
142 |
+
min-width: 100%;
|
143 |
+
height: 100%;
|
144 |
+
padding: 90px 30px 30px 30px;
|
145 |
+
display: flex;
|
146 |
+
flex-direction: column;
|
147 |
+
overflow-y: auto;
|
148 |
+
}
|
149 |
+
.conversation {
|
150 |
+
display: flex;
|
151 |
+
flex-direction: column;
|
152 |
+
gap: 15px;
|
153 |
+
}
|
154 |
+
/* Message Bubbles */
|
155 |
+
.message {
|
156 |
+
padding: 10px 16px;
|
157 |
+
border-radius: 16px;
|
158 |
+
font-size: 1em;
|
159 |
+
line-height: 1.5;
|
160 |
+
max-width: 95%;
|
161 |
+
position: relative;
|
162 |
+
box-shadow: 0 2px 8px rgba(0,0,0,0.07);
|
163 |
+
transition: background 0.3s, transform 0.3s;
|
164 |
+
}
|
165 |
+
.user { background: #F0F4FF; border: 1px solid #D9E2FF; align-self: flex-start; }
|
166 |
+
.ai { background: #FFF4E6; border: 1px solid #FFE0B2; align-self: flex-end; }
|
167 |
+
.message-text {
|
168 |
+
display: block;
|
169 |
+
max-height: 80px;
|
170 |
+
overflow: hidden;
|
171 |
+
transition: max-height 0.3s ease;
|
172 |
+
}
|
173 |
+
.message.expanded .message-text { max-height: none; }
|
174 |
+
.toggle-btn {
|
175 |
+
background: none;
|
176 |
+
border: none;
|
177 |
+
color: #667eea;
|
178 |
+
cursor: pointer;
|
179 |
+
font-size: 0.85em;
|
180 |
+
margin-top: 4px;
|
181 |
+
padding: 0;
|
182 |
+
}
|
183 |
+
.vertical-file-list {
|
184 |
+
margin-bottom: 10px;
|
185 |
+
display: flex;
|
186 |
+
flex-direction: column;
|
187 |
+
gap: 5px;
|
188 |
+
}
|
189 |
+
.file-item-vertical {
|
190 |
+
background: #eaeaea;
|
191 |
+
padding: 5px 8px;
|
192 |
+
border-radius: 4px;
|
193 |
+
font-size: 0.9em;
|
194 |
+
}
|
195 |
+
/* Navigation Buttons */
|
196 |
+
.nav {
|
197 |
+
position: absolute;
|
198 |
+
top: 50%;
|
199 |
+
transform: translateY(-50%);
|
200 |
+
width: 50px;
|
201 |
+
height: 50px;
|
202 |
+
background: rgba(255,255,255,0.5);
|
203 |
+
backdrop-filter: blur(4px);
|
204 |
+
border: none;
|
205 |
+
border-radius: 50%;
|
206 |
+
display: flex;
|
207 |
+
align-items: center;
|
208 |
+
justify-content: center;
|
209 |
+
cursor: pointer;
|
210 |
+
z-index: 10;
|
211 |
+
transition: transform 0.3s, background 0.3s;
|
212 |
+
}
|
213 |
+
.nav:hover { transform: translateY(-50%) scale(1.1); background: rgba(255,255,255,0.8); }
|
214 |
+
.nav:disabled { opacity: 0.5; cursor: default; }
|
215 |
+
#prevBtn { left: 15px; }
|
216 |
+
#nextBtn { right: 15px; }
|
217 |
+
/* Input Section */
|
218 |
+
.input-container {
|
219 |
+
padding: 20px;
|
220 |
+
background: #FAFAFA;
|
221 |
+
border-top: 1px solid #EEE;
|
222 |
+
display: flex;
|
223 |
+
flex-direction: column;
|
224 |
+
}
|
225 |
+
.file-attachments {
|
226 |
+
display: flex;
|
227 |
+
gap: 10px;
|
228 |
+
overflow-x: auto;
|
229 |
+
padding-bottom: 10px;
|
230 |
+
scrollbar-width: thin;
|
231 |
+
}
|
232 |
+
.file-item {
|
233 |
+
flex: 0 0 auto;
|
234 |
+
background: #eee;
|
235 |
+
padding: 5px 10px;
|
236 |
+
border-radius: 5px;
|
237 |
+
display: flex;
|
238 |
+
align-items: center;
|
239 |
+
gap: 5px;
|
240 |
+
font-size: 0.9em;
|
241 |
+
}
|
242 |
+
.file-item button {
|
243 |
+
background: none;
|
244 |
+
border: none;
|
245 |
+
color: #667eea;
|
246 |
+
cursor: pointer;
|
247 |
+
font-size: 1em;
|
248 |
+
}
|
249 |
+
.input-row {
|
250 |
+
display: flex;
|
251 |
+
align-items: center;
|
252 |
+
gap: 10px;
|
253 |
+
}
|
254 |
+
.attach-button {
|
255 |
+
background: #fff;
|
256 |
+
border: 1px solid #DDD;
|
257 |
+
border-radius: 4px;
|
258 |
+
padding: 8px 12px;
|
259 |
+
cursor: pointer;
|
260 |
+
transition: background 0.3s;
|
261 |
+
}
|
262 |
+
.attach-button:hover { background: #f0f0f0; }
|
263 |
+
.input-container textarea {
|
264 |
+
flex: 1;
|
265 |
+
padding: 12px 15px;
|
266 |
+
font-size: 1em;
|
267 |
+
border: 1px solid #DDD;
|
268 |
+
border-radius: 8px;
|
269 |
+
outline: none;
|
270 |
+
resize: none;
|
271 |
+
overflow-y: auto;
|
272 |
+
min-height: 36px;
|
273 |
+
max-height: 150px;
|
274 |
+
line-height: 1.4em;
|
275 |
+
white-space: pre-wrap;
|
276 |
+
transition: border-color 0.3s;
|
277 |
+
}
|
278 |
+
.input-container textarea:focus { border-color: #667eea; }
|
279 |
+
.input-container button {
|
280 |
+
padding: 12px 20px;
|
281 |
+
font-size: 1em;
|
282 |
+
background: #667eea;
|
283 |
+
color: #fff;
|
284 |
+
border: none;
|
285 |
+
border-radius: 8px;
|
286 |
+
cursor: pointer;
|
287 |
+
transition: background 0.3s;
|
288 |
+
}
|
289 |
+
.input-container button:hover { background: #556cd6; }
|
290 |
+
#fileInput { display: none; }
|
291 |
+
/* Summary Overlay Panel (Per‑Session) */
|
292 |
+
#summaryOverlay {
|
293 |
+
position: fixed;
|
294 |
+
top: 0;
|
295 |
+
right: 0;
|
296 |
+
bottom: 0;
|
297 |
+
width: 60%;
|
298 |
+
background: #fff;
|
299 |
+
box-shadow: -4px 0 12px rgba(0,0,0,0.15);
|
300 |
+
transform: translateX(100%);
|
301 |
+
transition: transform 0.3s ease;
|
302 |
+
z-index: 20;
|
303 |
+
display: flex;
|
304 |
+
flex-direction: column;
|
305 |
+
}
|
306 |
+
#summaryOverlay.active { transform: translateX(0); }
|
307 |
+
.summary-header {
|
308 |
+
padding: 16px;
|
309 |
+
background: #667eea;
|
310 |
+
color: #fff;
|
311 |
+
font-size: 1.2em;
|
312 |
+
display: flex;
|
313 |
+
justify-content: space-between;
|
314 |
+
align-items: center;
|
315 |
+
}
|
316 |
+
.summary-header-buttons {
|
317 |
+
display: flex;
|
318 |
+
gap: 10px;
|
319 |
+
}
|
320 |
+
.download-summary {
|
321 |
+
background: #fff;
|
322 |
+
color: #667eea;
|
323 |
+
border: 1px solid #667eea;
|
324 |
+
border-radius: 4px;
|
325 |
+
padding: 4px 8px;
|
326 |
+
cursor: pointer;
|
327 |
+
}
|
328 |
+
.download-summary:hover { background: #667eea; color: #fff; }
|
329 |
+
.close-summary {
|
330 |
+
background: none;
|
331 |
+
border: none;
|
332 |
+
color: #fff;
|
333 |
+
font-size: 1.2em;
|
334 |
+
cursor: pointer;
|
335 |
+
}
|
336 |
+
.summary-content {
|
337 |
+
padding: 16px;
|
338 |
+
overflow-y: auto;
|
339 |
+
flex: 1;
|
340 |
+
}
|
341 |
+
/* Summary content now displays rendered markdown */
|
342 |
+
/* Settings Overlay Panel (Per‑Session) */
|
343 |
+
#settingsOverlay {
|
344 |
+
position: fixed;
|
345 |
+
top: 0;
|
346 |
+
right: 0;
|
347 |
+
bottom: 0;
|
348 |
+
width: 40%;
|
349 |
+
background: #fff;
|
350 |
+
box-shadow: -4px 0 12px rgba(0,0,0,0.15);
|
351 |
+
transform: translateX(100%);
|
352 |
+
transition: transform 0.3s ease;
|
353 |
+
z-index: 20;
|
354 |
+
display: flex;
|
355 |
+
flex-direction: column;
|
356 |
+
}
|
357 |
+
#settingsOverlay.active { transform: translateX(0); }
|
358 |
+
.settings-header {
|
359 |
+
padding: 16px;
|
360 |
+
background: #667eea;
|
361 |
+
color: #fff;
|
362 |
+
font-size: 1.4em;
|
363 |
+
display: flex;
|
364 |
+
justify-content: space-between;
|
365 |
+
align-items: center;
|
366 |
+
}
|
367 |
+
.close-settings {
|
368 |
+
background: none;
|
369 |
+
border: none;
|
370 |
+
color: #fff;
|
371 |
+
font-size: 1.4em;
|
372 |
+
cursor: pointer;
|
373 |
+
}
|
374 |
+
.settings-content {
|
375 |
+
padding: 16px;
|
376 |
+
overflow-y: auto;
|
377 |
+
flex: 1;
|
378 |
+
font-size: 1.1em;
|
379 |
+
line-height: 1.4;
|
380 |
+
}
|
381 |
+
.settings-group {
|
382 |
+
margin-bottom: 16px;
|
383 |
+
display: flex;
|
384 |
+
align-items: center;
|
385 |
+
gap: 10px;
|
386 |
+
}
|
387 |
+
.settings-group label {
|
388 |
+
min-width: 100px;
|
389 |
+
}
|
390 |
+
.save-settings {
|
391 |
+
background: #667eea;
|
392 |
+
color: #fff;
|
393 |
+
border: none;
|
394 |
+
border-radius: 8px;
|
395 |
+
padding: 10px 20px;
|
396 |
+
cursor: pointer;
|
397 |
+
}
|
398 |
+
.save-settings:hover { background: #556cd6; }
|
399 |
+
/* Responsive */
|
400 |
+
@media (max-width: 600px) {
|
401 |
+
.nav-bar { display: none; }
|
402 |
+
.chat-wrapper { margin: 0; }
|
403 |
+
.message { font-size: 0.95em; max-width: 80%; }
|
404 |
+
.nav { width: 40px; height: 40px; font-size: 1.5em; }
|
405 |
+
.card { padding: 90px 20px 20px 20px; }
|
406 |
+
.input-container { padding: 15px; }
|
407 |
+
.input-row { flex-direction: column; gap: 10px; }
|
408 |
+
.input-container button { width: 100%; }
|
409 |
+
.top-button { font-size: 1em; height: 40px; padding: 0 20px; }
|
410 |
+
#summaryOverlay, #settingsOverlay { width: 80%; }
|
411 |
+
}
|
412 |
+
</style>
|
413 |
+
</head>
|
414 |
+
<body>
|
415 |
+
<div class="app-container">
|
416 |
+
<!-- Left Nav Bar for Chat History & Session Management -->
|
417 |
+
<div class="nav-bar">
|
418 |
+
<h3>Chat History</h3>
|
419 |
+
<ul id="sessionList"></ul>
|
420 |
+
<button class="new-session-btn" id="newSessionBtn">New Chat</button>
|
421 |
+
</div>
|
422 |
+
<!-- Main Chat Wrapper -->
|
423 |
+
<div class="chat-wrapper">
|
424 |
+
<div class="top-buttons">
|
425 |
+
<button id="summaryBtn" class="top-button">Summary</button>
|
426 |
+
<button id="settingsBtn" class="top-button">Settings</button>
|
427 |
+
</div>
|
428 |
+
<!-- Turn Label -->
|
429 |
+
<div id="turnLabel">Turn: 0/0</div>
|
430 |
+
<div class="carousel-wrapper">
|
431 |
+
<button id="prevBtn" class="nav">
|
432 |
+
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="#444" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
433 |
+
<polyline points="15 18 9 12 15 6"></polyline>
|
434 |
+
</svg>
|
435 |
+
</button>
|
436 |
+
<div class="carousel" id="carousel">
|
437 |
+
<!-- Conversation cards will be dynamically rendered -->
|
438 |
+
</div>
|
439 |
+
<button id="nextBtn" class="nav">
|
440 |
+
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="#444" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
441 |
+
<polyline points="9 18 15 12 9 6"></polyline>
|
442 |
+
</svg>
|
443 |
+
</button>
|
444 |
+
</div>
|
445 |
+
<!-- Input Section -->
|
446 |
+
<div class="input-container">
|
447 |
+
<div id="fileAttachments" class="file-attachments"></div>
|
448 |
+
<div class="input-row">
|
449 |
+
<button id="attachBtn" class="attach-button">Attach File</button>
|
450 |
+
<input type="file" id="fileInput" multiple accept="image/*,.pdf">
|
451 |
+
<textarea id="chatInput" placeholder="Type your message..."></textarea>
|
452 |
+
<button id="sendBtn">Send</button>
|
453 |
+
</div>
|
454 |
+
</div>
|
455 |
+
</div>
|
456 |
+
</div>
|
457 |
+
|
458 |
+
<!-- Summary Overlay Panel (Per‑Session, rendered markdown) -->
|
459 |
+
<div id="summaryOverlay">
|
460 |
+
<div class="summary-header">
|
461 |
+
<span>Chat Summary</span>
|
462 |
+
<div class="summary-header-buttons">
|
463 |
+
<button id="downloadSummary" class="download-summary">Download</button>
|
464 |
+
<button class="close-summary" id="closeSummaryBtn">×</button>
|
465 |
+
</div>
|
466 |
+
</div>
|
467 |
+
<div class="summary-content" id="summaryContent">
|
468 |
+
<!-- Rendered markdown will appear here -->
|
469 |
+
</div>
|
470 |
+
</div>
|
471 |
+
|
472 |
+
<!-- Settings Overlay Panel (Per‑Session) -->
|
473 |
+
<div id="settingsOverlay">
|
474 |
+
<div class="settings-header">
|
475 |
+
<span>Settings</span>
|
476 |
+
<button class="close-settings" id="closeSettingsBtn">×</button>
|
477 |
+
</div>
|
478 |
+
<div class="settings-content">
|
479 |
+
<form id="settingsForm">
|
480 |
+
<div class="settings-group">
|
481 |
+
<label for="temperature">Temperature:</label>
|
482 |
+
<input type="range" id="temperature" name="temperature" min="0" max="1" step="0.01">
|
483 |
+
<span id="temperatureValue"></span>
|
484 |
+
</div>
|
485 |
+
<div class="settings-group">
|
486 |
+
<label for="maxTokens">Max Tokens:</label>
|
487 |
+
<input type="number" id="maxTokens" name="maxTokens" min="10" max="2048">
|
488 |
+
</div>
|
489 |
+
<div class="settings-group">
|
490 |
+
<label for="persona">Persona:</label>
|
491 |
+
<select id="persona" name="persona">
|
492 |
+
<option value="professional">Professional</option>
|
493 |
+
<option value="friendly">Friendly</option>
|
494 |
+
</select>
|
495 |
+
</div>
|
496 |
+
<button type="button" id="saveSettings" class="save-settings">Save Settings</button>
|
497 |
+
</form>
|
498 |
+
</div>
|
499 |
+
</div>
|
500 |
+
|
501 |
+
<script>
|
502 |
+
// ----------------- Session Management -----------------
|
503 |
+
let sessions = [];
|
504 |
+
let currentSessionIndex = 0;
|
505 |
+
let currentCardIndex = 0;
|
506 |
+
|
507 |
+
// Initialize with a default session
|
508 |
+
function initSessions() {
|
509 |
+
sessions.push({
|
510 |
+
id: Date.now(),
|
511 |
+
name: "Chat Session 1",
|
512 |
+
messages: [],
|
513 |
+
summary: "# Chat Summary\n\nThis is the default summary for Chat Session 1.",
|
514 |
+
settings: { temperature: 0.7, maxTokens: 256, persona: "professional" }
|
515 |
+
});
|
516 |
+
currentSessionIndex = 0;
|
517 |
+
currentCardIndex = 0;
|
518 |
+
renderSessionList();
|
519 |
+
renderCurrentSession();
|
520 |
+
}
|
521 |
+
|
522 |
+
// Render the left nav bar session list
|
523 |
+
function renderSessionList() {
|
524 |
+
const sessionList = document.getElementById('sessionList');
|
525 |
+
sessionList.innerHTML = "";
|
526 |
+
sessions.forEach((session, index) => {
|
527 |
+
const li = document.createElement('li');
|
528 |
+
li.textContent = session.name;
|
529 |
+
li.dataset.index = index;
|
530 |
+
if (index === currentSessionIndex) li.classList.add('active');
|
531 |
+
// Remove button for each session
|
532 |
+
const removeBtn = document.createElement('button');
|
533 |
+
removeBtn.textContent = "×";
|
534 |
+
removeBtn.className = "remove-session";
|
535 |
+
removeBtn.addEventListener('click', (e) => {
|
536 |
+
e.stopPropagation();
|
537 |
+
removeSession(index);
|
538 |
+
});
|
539 |
+
li.appendChild(removeBtn);
|
540 |
+
li.addEventListener('click', () => {
|
541 |
+
currentSessionIndex = index;
|
542 |
+
currentCardIndex = 0;
|
543 |
+
renderSessionList();
|
544 |
+
renderCurrentSession();
|
545 |
+
});
|
546 |
+
sessionList.appendChild(li);
|
547 |
+
});
|
548 |
+
}
|
549 |
+
|
550 |
+
// Create a new session
|
551 |
+
document.getElementById('newSessionBtn').addEventListener('click', () => {
|
552 |
+
const newSession = {
|
553 |
+
id: Date.now(),
|
554 |
+
name: "Chat Session " + (sessions.length + 1),
|
555 |
+
messages: [],
|
556 |
+
summary: "# Chat Summary\n\nThis is the default summary for Chat Session " + (sessions.length + 1) + ".",
|
557 |
+
settings: { temperature: 0.7, maxTokens: 256, persona: "professional" }
|
558 |
+
};
|
559 |
+
sessions.push(newSession);
|
560 |
+
currentSessionIndex = sessions.length - 1;
|
561 |
+
currentCardIndex = 0;
|
562 |
+
renderSessionList();
|
563 |
+
renderCurrentSession();
|
564 |
+
});
|
565 |
+
|
566 |
+
// Remove a session
|
567 |
+
function removeSession(index) {
|
568 |
+
sessions.splice(index, 1);
|
569 |
+
if (sessions.length === 0) {
|
570 |
+
initSessions();
|
571 |
+
} else {
|
572 |
+
if (currentSessionIndex >= sessions.length) {
|
573 |
+
currentSessionIndex = sessions.length - 1;
|
574 |
+
}
|
575 |
+
currentCardIndex = 0;
|
576 |
+
}
|
577 |
+
renderSessionList();
|
578 |
+
renderCurrentSession();
|
579 |
+
}
|
580 |
+
|
581 |
+
// ----------------- Carousel Rendering -----------------
|
582 |
+
const carousel = document.getElementById('carousel');
|
583 |
+
function renderCurrentSession() {
|
584 |
+
const session = sessions[currentSessionIndex];
|
585 |
+
carousel.innerHTML = "";
|
586 |
+
session.messages.forEach(message => {
|
587 |
+
const card = document.createElement('div');
|
588 |
+
card.className = 'card';
|
589 |
+
let attachmentHTML = "";
|
590 |
+
if (message.attachments && message.attachments.length > 0) {
|
591 |
+
attachmentHTML = `<div class="vertical-file-list">` +
|
592 |
+
message.attachments.map(name => `<div class="file-item-vertical">${name}</div>`).join("") +
|
593 |
+
`</div>`;
|
594 |
+
}
|
595 |
+
card.innerHTML = `
|
596 |
+
<div class="conversation">
|
597 |
+
<div class="message user">
|
598 |
+
${attachmentHTML}
|
599 |
+
<span class="message-text">User: ${message.userText}</span>
|
600 |
+
</div>
|
601 |
+
<div class="message ai">
|
602 |
+
<span class="message-text">AI: ${message.aiResponse}</span>
|
603 |
+
</div>
|
604 |
+
</div>
|
605 |
+
`;
|
606 |
+
carousel.appendChild(card);
|
607 |
+
processMessagesInContainer(card);
|
608 |
+
});
|
609 |
+
currentCardIndex = session.messages.length > 0 ? session.messages.length - 1 : 0;
|
610 |
+
updateCarousel();
|
611 |
+
}
|
612 |
+
|
613 |
+
function updateCarousel() {
|
614 |
+
const cards = document.querySelectorAll('.card');
|
615 |
+
carousel.style.transform = `translateX(-${currentCardIndex * 100}%)`;
|
616 |
+
prevBtn.disabled = currentCardIndex === 0;
|
617 |
+
nextBtn.disabled = currentCardIndex === cards.length - 1 || cards.length === 0;
|
618 |
+
updateTurnLabel(cards.length);
|
619 |
+
}
|
620 |
+
|
621 |
+
function updateTurnLabel(totalCards) {
|
622 |
+
const turnLabel = document.getElementById('turnLabel');
|
623 |
+
turnLabel.textContent = `Turn: ${totalCards ? currentCardIndex + 1 + "/" + totalCards : "0/0"}`;
|
624 |
+
}
|
625 |
+
|
626 |
+
// ----------------- Message Processing -----------------
|
627 |
+
function processMessage(messageEl) {
|
628 |
+
if (messageEl.dataset.processed) return;
|
629 |
+
const textEl = messageEl.querySelector('.message-text');
|
630 |
+
if (textEl.scrollHeight > 80) {
|
631 |
+
const toggleBtn = document.createElement('button');
|
632 |
+
toggleBtn.className = 'toggle-btn';
|
633 |
+
toggleBtn.textContent = 'Read more';
|
634 |
+
toggleBtn.addEventListener('click', function() {
|
635 |
+
if (messageEl.classList.contains('expanded')) {
|
636 |
+
messageEl.classList.remove('expanded');
|
637 |
+
toggleBtn.textContent = 'Read more';
|
638 |
+
} else {
|
639 |
+
messageEl.classList.add('expanded');
|
640 |
+
toggleBtn.textContent = 'Read less';
|
641 |
+
}
|
642 |
+
});
|
643 |
+
messageEl.appendChild(toggleBtn);
|
644 |
+
}
|
645 |
+
messageEl.dataset.processed = 'true';
|
646 |
+
}
|
647 |
+
function processMessagesInContainer(container) {
|
648 |
+
container.querySelectorAll('.message').forEach(processMessage);
|
649 |
+
}
|
650 |
+
|
651 |
+
// ----------------- Adding Conversation -----------------
|
652 |
+
const attachedFiles = []; // For demo, store file objects (only names)
|
653 |
+
function addConversation(userText, aiResponse) {
|
654 |
+
const attachmentNames = attachedFiles.map(file => file.name);
|
655 |
+
const message = {
|
656 |
+
userText,
|
657 |
+
aiResponse,
|
658 |
+
attachments: attachmentNames
|
659 |
+
};
|
660 |
+
sessions[currentSessionIndex].messages.push(message);
|
661 |
+
clearFileAttachments();
|
662 |
+
renderCurrentSession();
|
663 |
+
}
|
664 |
+
function clearFileAttachments() {
|
665 |
+
attachedFiles.length = 0;
|
666 |
+
updateFileAttachments();
|
667 |
+
}
|
668 |
+
|
669 |
+
// ----------------- Event Listeners -----------------
|
670 |
+
const sendBtn = document.getElementById('sendBtn');
|
671 |
+
const chatInput = document.getElementById('chatInput');
|
672 |
+
sendBtn.addEventListener('click', () => {
|
673 |
+
const text = chatInput.value;
|
674 |
+
if (text.trim() !== '') {
|
675 |
+
const aiResponse = "I'm here to help!";
|
676 |
+
addConversation(text, aiResponse);
|
677 |
+
chatInput.value = '';
|
678 |
+
chatInput.style.height = "36px";
|
679 |
+
}
|
680 |
+
});
|
681 |
+
chatInput.addEventListener('keydown', function(e) {
|
682 |
+
if (e.key === 'Enter' && (e.ctrlKey || e.metaKey)) {
|
683 |
+
e.preventDefault();
|
684 |
+
sendBtn.click();
|
685 |
+
}
|
686 |
+
});
|
687 |
+
const prevBtn = document.getElementById('prevBtn');
|
688 |
+
const nextBtn = document.getElementById('nextBtn');
|
689 |
+
prevBtn.addEventListener('click', () => {
|
690 |
+
if (currentCardIndex > 0) { currentCardIndex--; updateCarousel(); }
|
691 |
+
});
|
692 |
+
nextBtn.addEventListener('click', () => {
|
693 |
+
const cards = document.querySelectorAll('.card');
|
694 |
+
if (currentCardIndex < cards.length - 1) { currentCardIndex++; updateCarousel(); }
|
695 |
+
});
|
696 |
+
// ----------------- File Attachment Handling -----------------
|
697 |
+
const attachBtn = document.getElementById('attachBtn');
|
698 |
+
const fileInput = document.getElementById('fileInput');
|
699 |
+
const fileAttachments = document.getElementById('fileAttachments');
|
700 |
+
attachBtn.addEventListener('click', () => { fileInput.click(); });
|
701 |
+
fileInput.addEventListener('change', () => {
|
702 |
+
for (const file of fileInput.files) { attachedFiles.push(file); }
|
703 |
+
fileInput.value = "";
|
704 |
+
updateFileAttachments();
|
705 |
+
});
|
706 |
+
function updateFileAttachments() {
|
707 |
+
fileAttachments.innerHTML = "";
|
708 |
+
attachedFiles.forEach((file, index) => {
|
709 |
+
const fileDiv = document.createElement("div");
|
710 |
+
fileDiv.className = "file-item";
|
711 |
+
fileDiv.innerHTML = `<span>${file.name}</span> <button data-index="${index}">×</button>`;
|
712 |
+
fileAttachments.appendChild(fileDiv);
|
713 |
+
});
|
714 |
+
document.querySelectorAll(".file-item button").forEach(btn => {
|
715 |
+
btn.addEventListener("click", (e) => {
|
716 |
+
const idx = e.target.getAttribute("data-index");
|
717 |
+
attachedFiles.splice(idx, 1);
|
718 |
+
updateFileAttachments();
|
719 |
+
});
|
720 |
+
});
|
721 |
+
}
|
722 |
+
|
723 |
+
// ----------------- Summary Overlay (Per‑Session) -----------------
|
724 |
+
const summaryOverlay = document.getElementById('summaryOverlay');
|
725 |
+
const summaryBtn = document.getElementById('summaryBtn');
|
726 |
+
const closeSummaryBtn = document.getElementById('closeSummaryBtn');
|
727 |
+
const summaryContent = document.getElementById('summaryContent');
|
728 |
+
const downloadSummaryBtn = document.getElementById('downloadSummary');
|
729 |
+
|
730 |
+
summaryBtn.addEventListener('click', (e) => {
|
731 |
+
// Close settings if open
|
732 |
+
settingsOverlay.classList.remove('active');
|
733 |
+
// Load current session's summary and render as markdown
|
734 |
+
summaryContent.innerHTML = marked.parse(sessions[currentSessionIndex].summary);
|
735 |
+
summaryOverlay.classList.toggle('active');
|
736 |
+
e.stopPropagation();
|
737 |
+
});
|
738 |
+
closeSummaryBtn.addEventListener('click', () => { summaryOverlay.classList.remove('active'); });
|
739 |
+
downloadSummaryBtn.addEventListener('click', () => {
|
740 |
+
const blob = new Blob([sessions[currentSessionIndex].summary], { type: "text/markdown" });
|
741 |
+
const url = URL.createObjectURL(blob);
|
742 |
+
const a = document.createElement("a");
|
743 |
+
a.href = url;
|
744 |
+
a.download = "summary.md";
|
745 |
+
a.click();
|
746 |
+
URL.revokeObjectURL(url);
|
747 |
+
});
|
748 |
+
|
749 |
+
// ----------------- Settings Overlay (Per‑Session) -----------------
|
750 |
+
const settingsOverlay = document.getElementById('settingsOverlay');
|
751 |
+
const settingsBtn = document.getElementById('settingsBtn');
|
752 |
+
const closeSettingsBtn = document.getElementById('closeSettingsBtn');
|
753 |
+
const temperatureInput = document.getElementById('temperature');
|
754 |
+
const temperatureValue = document.getElementById('temperatureValue');
|
755 |
+
const maxTokensInput = document.getElementById('maxTokens');
|
756 |
+
const personaSelect = document.getElementById('persona');
|
757 |
+
const saveSettingsBtn = document.getElementById('saveSettings');
|
758 |
+
|
759 |
+
settingsBtn.addEventListener('click', (e) => {
|
760 |
+
// Close summary if open
|
761 |
+
summaryOverlay.classList.remove('active');
|
762 |
+
// Load current session's settings
|
763 |
+
const settings = sessions[currentSessionIndex].settings;
|
764 |
+
temperatureInput.value = settings.temperature;
|
765 |
+
temperatureValue.textContent = settings.temperature;
|
766 |
+
maxTokensInput.value = settings.maxTokens;
|
767 |
+
personaSelect.value = settings.persona;
|
768 |
+
settingsOverlay.classList.toggle('active');
|
769 |
+
e.stopPropagation();
|
770 |
+
});
|
771 |
+
closeSettingsBtn.addEventListener('click', () => { settingsOverlay.classList.remove('active'); });
|
772 |
+
temperatureInput.addEventListener('input', () => {
|
773 |
+
temperatureValue.textContent = temperatureInput.value;
|
774 |
+
});
|
775 |
+
saveSettingsBtn.addEventListener('click', () => {
|
776 |
+
sessions[currentSessionIndex].settings = {
|
777 |
+
temperature: parseFloat(temperatureInput.value),
|
778 |
+
maxTokens: parseInt(maxTokensInput.value),
|
779 |
+
persona: personaSelect.value
|
780 |
+
};
|
781 |
+
console.log('Session settings saved:', sessions[currentSessionIndex].settings);
|
782 |
+
settingsOverlay.classList.remove('active');
|
783 |
+
});
|
784 |
+
|
785 |
+
// ----------------- Auto-Dismiss Overlays on Outside Click -----------------
|
786 |
+
document.addEventListener('click', (e) => {
|
787 |
+
if (summaryOverlay.classList.contains('active') &&
|
788 |
+
!summaryOverlay.contains(e.target) &&
|
789 |
+
e.target !== summaryBtn) {
|
790 |
+
summaryOverlay.classList.remove('active');
|
791 |
+
}
|
792 |
+
if (settingsOverlay.classList.contains('active') &&
|
793 |
+
!settingsOverlay.contains(e.target) &&
|
794 |
+
e.target !== settingsBtn) {
|
795 |
+
settingsOverlay.classList.remove('active');
|
796 |
+
}
|
797 |
+
});
|
798 |
+
|
799 |
+
// ----------------- Global Keyboard Navigation -----------------
|
800 |
+
document.addEventListener('keydown', (e) => {
|
801 |
+
if (document.activeElement !== chatInput) {
|
802 |
+
if (e.key === 'ArrowLeft' && currentCardIndex > 0) {
|
803 |
+
currentCardIndex--;
|
804 |
+
updateCarousel();
|
805 |
+
} else if (e.key === 'ArrowRight') {
|
806 |
+
const cards = document.querySelectorAll('.card');
|
807 |
+
if (currentCardIndex < cards.length - 1) {
|
808 |
+
currentCardIndex++;
|
809 |
+
updateCarousel();
|
810 |
+
}
|
811 |
+
}
|
812 |
+
}
|
813 |
+
});
|
814 |
+
|
815 |
+
// ----------------- Initialization -----------------
|
816 |
+
initSessions();
|
817 |
+
</script>
|
818 |
+
</body>
|
819 |
+
</html>
|