Haleshot commited on
Commit
f09994a
·
unverified ·
1 Parent(s): e0f3bfe

Enhance course metadata processing and improve HTML/CSS structure for better theming and user experience. Clean up descriptions, implement light/dark mode support, and refine course display logic with improved search functionality.

Browse files
Files changed (1) hide show
  1. scripts/build.py +470 -70
scripts/build.py CHANGED
@@ -6,6 +6,7 @@ import argparse
6
  import json
7
  from typing import List, Dict, Any
8
  from pathlib import Path
 
9
 
10
 
11
  def export_html_wasm(notebook_path: str, output_dir: str, as_app: bool = False) -> bool:
@@ -65,7 +66,17 @@ def get_course_metadata(course_dir: Path) -> Dict[str, Any]:
65
  description_lines.append(line)
66
  elif description_lines: # Stop at the next heading
67
  break
68
- metadata["description"] = " ".join(description_lines).strip()
 
 
 
 
 
 
 
 
 
 
69
 
70
  return metadata
71
 
@@ -99,13 +110,44 @@ def organize_notebooks_by_course(all_notebooks: List[str]) -> Dict[str, Dict[str
99
  display_name = notebook_id
100
  if "_" in notebook_id:
101
  display_name = "_".join(notebook_id.split("_")[1:])
102
- display_name = display_name.replace("_", " ").title()
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
103
 
104
  courses[course_id]["notebooks"].append({
105
  "id": notebook_id,
106
  "path": notebook_path,
107
  "display_name": display_name,
108
- "order": order
 
109
  })
110
 
111
  # Sort notebooks by order
@@ -116,11 +158,12 @@ def organize_notebooks_by_course(all_notebooks: List[str]) -> Dict[str, Dict[str
116
 
117
 
118
  def generate_eva_css() -> str:
119
- """Generate Neon Genesis Evangelion inspired CSS."""
120
  return """
121
  :root {
 
122
  --eva-purple: #9a1eb3;
123
- --eva-green: #00ff00;
124
  --eva-orange: #ff6600;
125
  --eva-blue: #0066ff;
126
  --eva-red: #ff0000;
@@ -132,6 +175,19 @@ def generate_eva_css() -> str:
132
  --eva-transition: all 0.3s ease;
133
  }
134
 
 
 
 
 
 
 
 
 
 
 
 
 
 
135
  body {
136
  background-color: var(--eva-black);
137
  color: var(--eva-text);
@@ -139,6 +195,7 @@ def generate_eva_css() -> str:
139
  margin: 0;
140
  padding: 0;
141
  line-height: 1.6;
 
142
  }
143
 
144
  .eva-container {
@@ -156,10 +213,15 @@ def generate_eva_css() -> str:
156
  align-items: center;
157
  position: sticky;
158
  top: 0;
159
- background-color: rgba(17, 17, 17, 0.95);
160
  z-index: 100;
161
  backdrop-filter: blur(5px);
162
  padding-top: 1rem;
 
 
 
 
 
163
  }
164
 
165
  .eva-logo {
@@ -168,16 +230,21 @@ def generate_eva_css() -> str:
168
  color: var(--eva-green);
169
  text-transform: uppercase;
170
  letter-spacing: 2px;
171
- text-shadow: 0 0 10px rgba(0, 255, 0, 0.5);
 
 
 
 
172
  }
173
 
174
  .eva-nav {
175
  display: flex;
176
  gap: 1.5rem;
 
177
  }
178
 
179
  .eva-nav a {
180
- color: white;
181
  text-decoration: none;
182
  text-transform: uppercase;
183
  font-size: 0.9rem;
@@ -202,6 +269,21 @@ def generate_eva_css() -> str:
202
  animation: scanline 1.5s linear infinite;
203
  }
204
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
205
  .eva-hero {
206
  background-color: var(--eva-terminal-bg);
207
  border: 1px solid var(--eva-green);
@@ -217,6 +299,11 @@ def generate_eva_css() -> str:
217
  background-size: cover;
218
  background-position: center;
219
  background-blend-mode: overlay;
 
 
 
 
 
220
  }
221
 
222
  .eva-hero::before {
@@ -236,7 +323,11 @@ def generate_eva_css() -> str:
236
  color: var(--eva-green);
237
  text-transform: uppercase;
238
  letter-spacing: 2px;
239
- text-shadow: 0 0 10px rgba(0, 255, 0, 0.5);
 
 
 
 
240
  }
241
 
242
  .eva-hero p {
@@ -302,10 +393,11 @@ def generate_eva_css() -> str:
302
  background-color: var(--eva-green);
303
  }
304
 
 
305
  .eva-courses {
306
- display: block;
307
- width: 100%;
308
- max-width: 100%;
309
  }
310
 
311
  .eva-course {
@@ -315,9 +407,9 @@ def generate_eva_css() -> str:
315
  transition: var(--eva-transition);
316
  position: relative;
317
  overflow: hidden;
318
- width: 100%;
319
- display: block;
320
- margin-bottom: 2rem;
321
  }
322
 
323
  .eva-course:hover {
@@ -336,17 +428,67 @@ def generate_eva_css() -> str:
336
  animation: scanline 2s linear infinite;
337
  }
338
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
339
  .eva-course-header {
340
- padding: 1.5rem;
341
  cursor: pointer;
342
  display: flex;
343
  justify-content: space-between;
344
  align-items: center;
345
  border-bottom: 1px solid rgba(154, 30, 179, 0.3);
 
 
 
 
 
 
 
 
346
  }
347
 
348
  .eva-course-title {
349
- font-size: 1.5rem;
350
  color: var(--eva-purple);
351
  text-transform: uppercase;
352
  letter-spacing: 1px;
@@ -363,19 +505,76 @@ def generate_eva_css() -> str:
363
  transform: rotate(180deg);
364
  }
365
 
366
- .eva-course-content {
367
- display: none;
 
 
368
  padding: 1.5rem;
 
 
 
 
 
 
 
 
 
 
369
  }
370
 
371
- .eva-course.active .eva-course-content {
372
- display: block;
 
 
373
  }
374
 
375
  .eva-course-description {
 
376
  margin-bottom: 1.5rem;
377
  font-size: 0.9rem;
378
  line-height: 1.6;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
379
  }
380
 
381
  .eva-notebooks {
@@ -396,6 +595,10 @@ def generate_eva_css() -> str:
396
  border-radius: 0 var(--eva-border-radius) var(--eva-border-radius) 0;
397
  }
398
 
 
 
 
 
399
  .eva-notebook:hover {
400
  background-color: rgba(0, 102, 255, 0.1);
401
  padding-left: 1rem;
@@ -403,7 +606,7 @@ def generate_eva_css() -> str:
403
  }
404
 
405
  .eva-notebook a {
406
- color: white;
407
  text-decoration: none;
408
  display: block;
409
  font-size: 0.9rem;
@@ -460,6 +663,12 @@ def generate_eva_css() -> str:
460
  left: 100%;
461
  }
462
 
 
 
 
 
 
 
463
  .eva-cta {
464
  background-color: var(--eva-terminal-bg);
465
  border: 1px solid var(--eva-orange);
@@ -499,15 +708,20 @@ def generate_eva_css() -> str:
499
  padding-top: 2rem;
500
  border-top: 2px solid var(--eva-green);
501
  display: flex;
502
- justify-content: space-between;
503
  align-items: center;
504
- flex-wrap: wrap;
505
  gap: 2rem;
506
  }
507
 
 
 
 
 
 
508
  .eva-footer-links {
509
  display: flex;
510
  gap: 1.5rem;
 
511
  }
512
 
513
  .eva-footer-links a {
@@ -520,8 +734,26 @@ def generate_eva_css() -> str:
520
  color: var(--eva-green);
521
  }
522
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
523
  .eva-footer-copyright {
524
  font-size: 0.9rem;
 
525
  }
526
 
527
  .eva-search {
@@ -543,7 +775,7 @@ def generate_eva_css() -> str:
543
  }
544
 
545
  .eva-search input:focus {
546
- box-shadow: 0 0 10px rgba(0, 255, 0, 0.3);
547
  }
548
 
549
  .eva-search input::placeholder {
@@ -639,7 +871,7 @@ def generate_index(courses: Dict[str, Dict[str, Any]], output_dir: str) -> None:
639
  with open(index_path, "w", encoding="utf-8") as f:
640
  f.write(
641
  """<!DOCTYPE html>
642
- <html lang="en">
643
  <head>
644
  <meta charset="UTF-8">
645
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
@@ -659,6 +891,9 @@ def generate_index(courses: Dict[str, Dict[str, Any]], output_dir: str) -> None:
659
  <a href="#contribute">Contribute</a>
660
  <a href="https://docs.marimo.io" target="_blank">Documentation</a>
661
  <a href="https://github.com/marimo-team/learn" target="_blank">GitHub</a>
 
 
 
662
  </nav>
663
  </header>
664
 
@@ -698,32 +933,126 @@ def generate_index(courses: Dict[str, Dict[str, Any]], output_dir: str) -> None:
698
  <div class="eva-search">
699
  <input type="text" id="courseSearch" placeholder="Search courses and notebooks...">
700
  <span class="eva-search-icon"><i class="fas fa-search"></i></span>
701
- </div>
702
  <div class="eva-courses">
703
  """
704
  )
705
 
706
- # Sort courses alphabetically
707
- sorted_courses = sorted(courses.values(), key=lambda x: x["title"])
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
708
 
709
- for course in sorted_courses:
710
  # Skip if no notebooks
711
  if not course["notebooks"]:
712
  continue
713
-
 
 
 
 
 
 
714
  f.write(
715
  f'<div class="eva-course" data-course-id="{course["id"]}">\n'
 
 
 
 
 
 
 
716
  f' <div class="eva-course-header">\n'
717
  f' <h2 class="eva-course-title">{course["title"]}</h2>\n'
718
  f' <span class="eva-course-toggle"><i class="fas fa-chevron-down"></i></span>\n'
719
  f' </div>\n'
720
- f' <div class="eva-course-content">\n'
721
  f' <p class="eva-course-description">{course["description"]}</p>\n'
 
 
 
 
 
 
722
  f' <div class="eva-notebooks">\n'
723
  )
724
 
725
  for i, notebook in enumerate(course["notebooks"]):
726
- notebook_number = f"{i+1:02d}"
 
727
  f.write(
728
  f' <div class="eva-notebook">\n'
729
  f' <span class="eva-notebook-number">{notebook_number}</span>\n'
@@ -753,19 +1082,58 @@ def generate_index(courses: Dict[str, Dict[str, Any]], output_dir: str) -> None:
753
  </section>
754
 
755
  <footer class="eva-footer">
756
- <div class="eva-footer-copyright">
757
- © 2024 Marimo Learn. Built with <a href="https://marimo.io" target="_blank" style="color: var(--eva-green);">marimo</a>.
 
 
 
 
 
 
 
 
 
758
  </div>
759
  <div class="eva-footer-links">
760
- <a href="https://marimo.io" target="_blank">Marimo Website</a>
761
  <a href="https://docs.marimo.io" target="_blank">Documentation</a>
762
  <a href="https://github.com/marimo-team/learn" target="_blank">GitHub</a>
763
  </div>
 
 
 
764
  </footer>
765
  </div>
766
 
767
  <script>
768
  document.addEventListener('DOMContentLoaded', function() {
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
769
  // Terminal typing effect for hero text
770
  const heroTitle = document.querySelector('.eva-hero h1');
771
  const heroText = document.querySelector('.eva-hero p');
@@ -801,47 +1169,65 @@ def generate_index(courses: Dict[str, Dict[str, Any]], output_dir: str) -> None:
801
 
802
  typeTitle();
803
 
804
- // Course toggle functionality - fixed to only open one at a time
805
  const courseHeaders = document.querySelectorAll('.eva-course-header');
 
806
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
807
  courseHeaders.forEach(header => {
808
  header.addEventListener('click', function(e) {
809
  e.preventDefault();
810
  e.stopPropagation();
811
 
812
- const currentCourse = this.parentElement;
813
- const isActive = currentCourse.classList.contains('active');
814
-
815
- // First close all courses
816
- document.querySelectorAll('.eva-course').forEach(course => {
817
- course.classList.remove('active');
818
- });
 
 
 
819
 
820
- // Toggle the clicked course
821
- if (!isActive) {
822
- currentCourse.classList.add('active');
823
-
824
- // Check if the course has any notebooks
825
- const notebooks = currentCourse.querySelectorAll('.eva-notebook');
826
- const content = currentCourse.querySelector('.eva-course-content');
827
-
828
- if (notebooks.length === 0 && !content.querySelector('.eva-empty-message')) {
829
- // If no notebooks, show a message
830
- const emptyMessage = document.createElement('p');
831
- emptyMessage.className = 'eva-empty-message';
832
- emptyMessage.textContent = 'No notebooks available in this course yet.';
833
- emptyMessage.style.color = 'var(--eva-text)';
834
- emptyMessage.style.fontStyle = 'italic';
835
- emptyMessage.style.opacity = '0.7';
836
- emptyMessage.style.textAlign = 'center';
837
- emptyMessage.style.padding = '1rem 0';
838
- content.appendChild(emptyMessage);
839
- }
840
- }
841
  });
842
  });
843
 
844
- // Search functionality
845
  const searchInput = document.getElementById('courseSearch');
846
  const courses = document.querySelectorAll('.eva-course');
847
  const notebooks = document.querySelectorAll('.eva-notebook');
@@ -881,6 +1267,10 @@ def generate_index(courses: Dict[str, Dict[str, Any]], output_dir: str) -> None:
881
  // Then show courses and notebooks that match the search
882
  let hasResults = false;
883
 
 
 
 
 
884
  notebooks.forEach(notebook => {
885
  const notebookTitle = notebook.querySelector('a').getAttribute('data-notebook-title').toLowerCase();
886
  const matchesSearch = notebookTitle.includes(searchTerm);
@@ -889,21 +1279,31 @@ def generate_index(courses: Dict[str, Dict[str, Any]], output_dir: str) -> None:
889
 
890
  if (matchesSearch) {
891
  const course = notebook.closest('.eva-course');
892
- course.style.display = 'block';
893
- course.classList.add('active');
894
  hasResults = true;
895
  }
896
  });
897
 
898
- // Also search course titles
899
  courses.forEach(course => {
 
900
  const courseTitle = course.querySelector('.eva-course-title').textContent.toLowerCase();
901
  const courseDescription = course.querySelector('.eva-course-description').textContent.toLowerCase();
902
 
903
- if (courseTitle.includes(searchTerm) || courseDescription.includes(searchTerm)) {
 
 
 
904
  course.style.display = 'block';
905
  course.classList.add('active');
906
  hasResults = true;
 
 
 
 
 
 
 
907
  }
908
  });
909
  });
 
6
  import json
7
  from typing import List, Dict, Any
8
  from pathlib import Path
9
+ import re
10
 
11
 
12
  def export_html_wasm(notebook_path: str, output_dir: str, as_app: bool = False) -> bool:
 
66
  description_lines.append(line)
67
  elif description_lines: # Stop at the next heading
68
  break
69
+ description = " ".join(description_lines).strip()
70
+ # Clean up the description
71
+ description = description.replace("_", "")
72
+ description = description.replace("[work in progress]", "")
73
+ description = description.replace("(https://github.com/marimo-team/learn/issues/51)", "")
74
+ # Remove any other GitHub issue links
75
+ description = re.sub(r'\[.*?\]\(https://github\.com/.*?/issues/\d+\)', '', description)
76
+ description = re.sub(r'https://github\.com/.*?/issues/\d+', '', description)
77
+ # Clean up any double spaces
78
+ description = re.sub(r'\s+', ' ', description).strip()
79
+ metadata["description"] = description
80
 
81
  return metadata
82
 
 
110
  display_name = notebook_id
111
  if "_" in notebook_id:
112
  display_name = "_".join(notebook_id.split("_")[1:])
113
+
114
+ # Convert display name to title case, but handle italics properly
115
+ parts = display_name.split("_")
116
+ formatted_parts = []
117
+
118
+ i = 0
119
+ while i < len(parts):
120
+ if i + 1 < len(parts) and parts[i] == "" and parts[i+1] == "":
121
+ # Skip empty parts that might come from consecutive underscores
122
+ i += 2
123
+ continue
124
+
125
+ if i + 1 < len(parts) and (parts[i] == "" or parts[i+1] == ""):
126
+ # This is an italics marker
127
+ if parts[i] == "":
128
+ # Opening italics
129
+ text_part = parts[i+1].replace("_", " ").title()
130
+ formatted_parts.append(f"<em>{text_part}</em>")
131
+ i += 2
132
+ else:
133
+ # Text followed by italics marker
134
+ text_part = parts[i].replace("_", " ").title()
135
+ formatted_parts.append(text_part)
136
+ i += 1
137
+ else:
138
+ # Regular text
139
+ text_part = parts[i].replace("_", " ").title()
140
+ formatted_parts.append(text_part)
141
+ i += 1
142
+
143
+ display_name = " ".join(formatted_parts)
144
 
145
  courses[course_id]["notebooks"].append({
146
  "id": notebook_id,
147
  "path": notebook_path,
148
  "display_name": display_name,
149
+ "order": order,
150
+ "original_number": notebook_id.split("_")[0] if "_" in notebook_id else ""
151
  })
152
 
153
  # Sort notebooks by order
 
158
 
159
 
160
  def generate_eva_css() -> str:
161
+ """Generate Neon Genesis Evangelion inspired CSS with light/dark mode support."""
162
  return """
163
  :root {
164
+ /* Dark mode colors (default) */
165
  --eva-purple: #9a1eb3;
166
+ --eva-green: #1c7361;
167
  --eva-orange: #ff6600;
168
  --eva-blue: #0066ff;
169
  --eva-red: #ff0000;
 
175
  --eva-transition: all 0.3s ease;
176
  }
177
 
178
+ /* Light mode colors */
179
+ [data-theme="light"] {
180
+ --eva-purple: #7209b7;
181
+ --eva-green: #1c7361;
182
+ --eva-orange: #e65100;
183
+ --eva-blue: #0039cb;
184
+ --eva-red: #c62828;
185
+ --eva-black: #f5f5f5;
186
+ --eva-dark: #e0e0e0;
187
+ --eva-terminal-bg: rgba(255, 255, 255, 0.9);
188
+ --eva-text: #333333;
189
+ }
190
+
191
  body {
192
  background-color: var(--eva-black);
193
  color: var(--eva-text);
 
195
  margin: 0;
196
  padding: 0;
197
  line-height: 1.6;
198
+ transition: background-color 0.3s ease, color 0.3s ease;
199
  }
200
 
201
  .eva-container {
 
213
  align-items: center;
214
  position: sticky;
215
  top: 0;
216
+ background-color: var(--eva-black);
217
  z-index: 100;
218
  backdrop-filter: blur(5px);
219
  padding-top: 1rem;
220
+ transition: background-color 0.3s ease;
221
+ }
222
+
223
+ [data-theme="light"] .eva-header {
224
+ background-color: rgba(245, 245, 245, 0.95);
225
  }
226
 
227
  .eva-logo {
 
230
  color: var(--eva-green);
231
  text-transform: uppercase;
232
  letter-spacing: 2px;
233
+ text-shadow: 0 0 10px rgba(28, 115, 97, 0.5);
234
+ }
235
+
236
+ [data-theme="light"] .eva-logo {
237
+ text-shadow: 0 0 10px rgba(28, 115, 97, 0.3);
238
  }
239
 
240
  .eva-nav {
241
  display: flex;
242
  gap: 1.5rem;
243
+ align-items: center;
244
  }
245
 
246
  .eva-nav a {
247
+ color: var(--eva-text);
248
  text-decoration: none;
249
  text-transform: uppercase;
250
  font-size: 0.9rem;
 
269
  animation: scanline 1.5s linear infinite;
270
  }
271
 
272
+ .theme-toggle {
273
+ background: none;
274
+ border: none;
275
+ color: var(--eva-text);
276
+ cursor: pointer;
277
+ font-size: 1.2rem;
278
+ padding: 0.5rem;
279
+ margin-left: 1rem;
280
+ transition: color 0.3s;
281
+ }
282
+
283
+ .theme-toggle:hover {
284
+ color: var(--eva-green);
285
+ }
286
+
287
  .eva-hero {
288
  background-color: var(--eva-terminal-bg);
289
  border: 1px solid var(--eva-green);
 
299
  background-size: cover;
300
  background-position: center;
301
  background-blend-mode: overlay;
302
+ transition: background-color 0.3s ease, border-color 0.3s ease;
303
+ }
304
+
305
+ [data-theme="light"] .eva-hero {
306
+ background-image: linear-gradient(45deg, rgba(255, 255, 255, 0.9), rgba(255, 255, 255, 0.7)), url('https://raw.githubusercontent.com/marimo-team/marimo/main/docs/_static/marimo-logotype-thick.svg');
307
  }
308
 
309
  .eva-hero::before {
 
323
  color: var(--eva-green);
324
  text-transform: uppercase;
325
  letter-spacing: 2px;
326
+ text-shadow: 0 0 10px rgba(28, 115, 97, 0.5);
327
+ }
328
+
329
+ [data-theme="light"] .eva-hero h1 {
330
+ text-shadow: 0 0 10px rgba(28, 115, 97, 0.3);
331
  }
332
 
333
  .eva-hero p {
 
393
  background-color: var(--eva-green);
394
  }
395
 
396
+ /* Flashcard view for courses */
397
  .eva-courses {
398
+ display: grid;
399
+ grid-template-columns: repeat(auto-fill, minmax(350px, 1fr));
400
+ gap: 2rem;
401
  }
402
 
403
  .eva-course {
 
407
  transition: var(--eva-transition);
408
  position: relative;
409
  overflow: hidden;
410
+ height: 350px;
411
+ display: flex;
412
+ flex-direction: column;
413
  }
414
 
415
  .eva-course:hover {
 
428
  animation: scanline 2s linear infinite;
429
  }
430
 
431
+ .eva-course-badge {
432
+ position: absolute;
433
+ top: 15px;
434
+ right: -40px;
435
+ background: linear-gradient(135deg, var(--eva-orange) 0%, #ff9500 100%);
436
+ color: var(--eva-black);
437
+ font-size: 0.65rem;
438
+ padding: 0.3rem 2.5rem;
439
+ text-transform: uppercase;
440
+ font-weight: bold;
441
+ z-index: 3;
442
+ letter-spacing: 1px;
443
+ box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3);
444
+ transform: rotate(45deg);
445
+ text-shadow: 0 1px 1px rgba(255, 255, 255, 0.2);
446
+ border-top: 1px solid rgba(255, 255, 255, 0.3);
447
+ border-bottom: 1px solid rgba(0, 0, 0, 0.2);
448
+ white-space: nowrap;
449
+ overflow: hidden;
450
+ }
451
+
452
+ .eva-course-badge i {
453
+ margin-right: 4px;
454
+ font-size: 0.7rem;
455
+ }
456
+
457
+ [data-theme="light"] .eva-course-badge {
458
+ box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2);
459
+ text-shadow: 0 1px 1px rgba(255, 255, 255, 0.4);
460
+ }
461
+
462
+ .eva-course-badge::before {
463
+ content: '';
464
+ position: absolute;
465
+ left: 0;
466
+ top: 0;
467
+ width: 100%;
468
+ height: 100%;
469
+ background: linear-gradient(to right, transparent, rgba(255, 255, 255, 0.3), transparent);
470
+ animation: scanline 2s linear infinite;
471
+ }
472
+
473
  .eva-course-header {
474
+ padding: 1rem 1.5rem;
475
  cursor: pointer;
476
  display: flex;
477
  justify-content: space-between;
478
  align-items: center;
479
  border-bottom: 1px solid rgba(154, 30, 179, 0.3);
480
+ z-index: 2;
481
+ background-color: var(--eva-terminal-bg);
482
+ position: absolute;
483
+ top: 0;
484
+ left: 0;
485
+ width: 100%;
486
+ height: 3.5rem;
487
+ box-sizing: border-box;
488
  }
489
 
490
  .eva-course-title {
491
+ font-size: 1.3rem;
492
  color: var(--eva-purple);
493
  text-transform: uppercase;
494
  letter-spacing: 1px;
 
505
  transform: rotate(180deg);
506
  }
507
 
508
+ .eva-course-front {
509
+ display: flex;
510
+ flex-direction: column;
511
+ justify-content: space-between;
512
  padding: 1.5rem;
513
+ margin-top: 3.5rem;
514
+ transition: opacity 0.3s ease, transform 0.3s ease;
515
+ position: absolute;
516
+ top: 0;
517
+ left: 0;
518
+ width: 100%;
519
+ height: calc(100% - 3.5rem);
520
+ background-color: var(--eva-terminal-bg);
521
+ z-index: 1;
522
+ box-sizing: border-box;
523
  }
524
 
525
+ .eva-course.active .eva-course-front {
526
+ opacity: 0;
527
+ transform: translateY(-10px);
528
+ pointer-events: none;
529
  }
530
 
531
  .eva-course-description {
532
+ margin-top: 0.5rem;
533
  margin-bottom: 1.5rem;
534
  font-size: 0.9rem;
535
  line-height: 1.6;
536
+ flex-grow: 1;
537
+ overflow: hidden;
538
+ display: -webkit-box;
539
+ -webkit-line-clamp: 4;
540
+ -webkit-box-orient: vertical;
541
+ max-height: 150px;
542
+ }
543
+
544
+ .eva-course-stats {
545
+ display: flex;
546
+ justify-content: space-between;
547
+ font-size: 0.8rem;
548
+ color: var(--eva-text);
549
+ opacity: 0.7;
550
+ }
551
+
552
+ .eva-course-content {
553
+ position: absolute;
554
+ top: 3.5rem;
555
+ left: 0;
556
+ width: 100%;
557
+ height: calc(100% - 3.5rem);
558
+ padding: 1.5rem;
559
+ background-color: var(--eva-terminal-bg);
560
+ transition: opacity 0.3s ease, transform 0.3s ease;
561
+ opacity: 0;
562
+ transform: translateY(10px);
563
+ pointer-events: none;
564
+ overflow-y: auto;
565
+ z-index: 1;
566
+ box-sizing: border-box;
567
+ }
568
+
569
+ .eva-course.active .eva-course-content {
570
+ opacity: 1;
571
+ transform: translateY(0);
572
+ pointer-events: auto;
573
+ }
574
+
575
+ .eva-course.active {
576
+ height: auto;
577
+ min-height: 350px;
578
  }
579
 
580
  .eva-notebooks {
 
595
  border-radius: 0 var(--eva-border-radius) var(--eva-border-radius) 0;
596
  }
597
 
598
+ [data-theme="light"] .eva-notebook {
599
+ background-color: rgba(0, 0, 0, 0.05);
600
+ }
601
+
602
  .eva-notebook:hover {
603
  background-color: rgba(0, 102, 255, 0.1);
604
  padding-left: 1rem;
 
606
  }
607
 
608
  .eva-notebook a {
609
+ color: var(--eva-text);
610
  text-decoration: none;
611
  display: block;
612
  font-size: 0.9rem;
 
663
  left: 100%;
664
  }
665
 
666
+ .eva-course-button {
667
+ margin-top: 1rem;
668
+ margin-bottom: 1rem;
669
+ align-self: center;
670
+ }
671
+
672
  .eva-cta {
673
  background-color: var(--eva-terminal-bg);
674
  border: 1px solid var(--eva-orange);
 
708
  padding-top: 2rem;
709
  border-top: 2px solid var(--eva-green);
710
  display: flex;
711
+ flex-direction: column;
712
  align-items: center;
 
713
  gap: 2rem;
714
  }
715
 
716
+ .eva-footer-logo {
717
+ max-width: 200px;
718
+ margin-bottom: 1rem;
719
+ }
720
+
721
  .eva-footer-links {
722
  display: flex;
723
  gap: 1.5rem;
724
+ margin-bottom: 1.5rem;
725
  }
726
 
727
  .eva-footer-links a {
 
734
  color: var(--eva-green);
735
  }
736
 
737
+ .eva-social-links {
738
+ display: flex;
739
+ gap: 1.5rem;
740
+ margin-bottom: 1.5rem;
741
+ }
742
+
743
+ .eva-social-links a {
744
+ color: var(--eva-text);
745
+ font-size: 1.5rem;
746
+ transition: var(--eva-transition);
747
+ }
748
+
749
+ .eva-social-links a:hover {
750
+ color: var(--eva-green);
751
+ transform: translateY(-3px);
752
+ }
753
+
754
  .eva-footer-copyright {
755
  font-size: 0.9rem;
756
+ text-align: center;
757
  }
758
 
759
  .eva-search {
 
775
  }
776
 
777
  .eva-search input:focus {
778
+ box-shadow: 0 0 10px rgba(28, 115, 97, 0.3);
779
  }
780
 
781
  .eva-search input::placeholder {
 
871
  with open(index_path, "w", encoding="utf-8") as f:
872
  f.write(
873
  """<!DOCTYPE html>
874
+ <html lang="en" data-theme="dark">
875
  <head>
876
  <meta charset="UTF-8">
877
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
 
891
  <a href="#contribute">Contribute</a>
892
  <a href="https://docs.marimo.io" target="_blank">Documentation</a>
893
  <a href="https://github.com/marimo-team/learn" target="_blank">GitHub</a>
894
+ <button id="themeToggle" class="theme-toggle" aria-label="Toggle dark/light mode">
895
+ <i class="fas fa-moon"></i>
896
+ </button>
897
  </nav>
898
  </header>
899
 
 
933
  <div class="eva-search">
934
  <input type="text" id="courseSearch" placeholder="Search courses and notebooks...">
935
  <span class="eva-search-icon"><i class="fas fa-search"></i></span>
936
+ </div>
937
  <div class="eva-courses">
938
  """
939
  )
940
 
941
+ # Define the custom order for courses
942
+ course_order = ["python", "probability", "polars", "optimization", "functional_programming"]
943
+
944
+ # Create a dictionary of courses by ID for easy lookup
945
+ courses_by_id = {course["id"]: course for course in courses.values()}
946
+
947
+ # Determine which courses are "work in progress" based on description or notebook count
948
+ work_in_progress = set()
949
+ for course_id, course in courses_by_id.items():
950
+ # Consider a course as "work in progress" if it has few notebooks or contains specific phrases
951
+ if (len(course["notebooks"]) < 5 or
952
+ "work in progress" in course["description"].lower() or
953
+ "help us add" in course["description"].lower() or
954
+ "check back later" in course["description"].lower()):
955
+ work_in_progress.add(course_id)
956
+
957
+ # First output courses in the specified order
958
+ for course_id in course_order:
959
+ if course_id in courses_by_id:
960
+ course = courses_by_id[course_id]
961
+
962
+ # Skip if no notebooks
963
+ if not course["notebooks"]:
964
+ continue
965
+
966
+ # Count notebooks
967
+ notebook_count = len(course["notebooks"])
968
+
969
+ # Determine if this course is a work in progress
970
+ is_wip = course_id in work_in_progress
971
+
972
+ f.write(
973
+ f'<div class="eva-course" data-course-id="{course["id"]}">\n'
974
+ )
975
+
976
+ # Add WIP badge if needed
977
+ if is_wip:
978
+ f.write(f' <div class="eva-course-badge"><i class="fas fa-code-branch"></i> In Progress</div>\n')
979
+
980
+ f.write(
981
+ f' <div class="eva-course-header">\n'
982
+ f' <h2 class="eva-course-title">{course["title"]}</h2>\n'
983
+ f' <span class="eva-course-toggle"><i class="fas fa-chevron-down"></i></span>\n'
984
+ f' </div>\n'
985
+ f' <div class="eva-course-front">\n'
986
+ f' <p class="eva-course-description">{course["description"]}</p>\n'
987
+ f' <div class="eva-course-stats">\n'
988
+ f' <span><i class="fas fa-book"></i> {notebook_count} notebook{"s" if notebook_count != 1 else ""}</span>\n'
989
+ f' </div>\n'
990
+ f' <button class="eva-button eva-course-button">View Notebooks</button>\n'
991
+ f' </div>\n'
992
+ f' <div class="eva-course-content">\n'
993
+ f' <div class="eva-notebooks">\n'
994
+ )
995
+
996
+ for i, notebook in enumerate(course["notebooks"]):
997
+ # Use original file number instead of sequential numbering
998
+ notebook_number = notebook.get("original_number", f"{i+1:02d}")
999
+ f.write(
1000
+ f' <div class="eva-notebook">\n'
1001
+ f' <span class="eva-notebook-number">{notebook_number}</span>\n'
1002
+ f' <a href="{notebook["path"].replace(".py", ".html")}" data-notebook-title="{notebook["display_name"]}">{notebook["display_name"]}</a>\n'
1003
+ f' </div>\n'
1004
+ )
1005
+
1006
+ f.write(
1007
+ f' </div>\n'
1008
+ f' </div>\n'
1009
+ f'</div>\n'
1010
+ )
1011
+
1012
+ # Remove from the dictionary so we don't output it again
1013
+ del courses_by_id[course_id]
1014
+
1015
+ # Then output any remaining courses alphabetically
1016
+ sorted_remaining_courses = sorted(courses_by_id.values(), key=lambda x: x["title"])
1017
 
1018
+ for course in sorted_remaining_courses:
1019
  # Skip if no notebooks
1020
  if not course["notebooks"]:
1021
  continue
1022
+
1023
+ # Count notebooks
1024
+ notebook_count = len(course["notebooks"])
1025
+
1026
+ # Determine if this course is a work in progress
1027
+ is_wip = course["id"] in work_in_progress
1028
+
1029
  f.write(
1030
  f'<div class="eva-course" data-course-id="{course["id"]}">\n'
1031
+ )
1032
+
1033
+ # Add WIP badge if needed
1034
+ if is_wip:
1035
+ f.write(f' <div class="eva-course-badge"><i class="fas fa-code-branch"></i> In Progress</div>\n')
1036
+
1037
+ f.write(
1038
  f' <div class="eva-course-header">\n'
1039
  f' <h2 class="eva-course-title">{course["title"]}</h2>\n'
1040
  f' <span class="eva-course-toggle"><i class="fas fa-chevron-down"></i></span>\n'
1041
  f' </div>\n'
1042
+ f' <div class="eva-course-front">\n'
1043
  f' <p class="eva-course-description">{course["description"]}</p>\n'
1044
+ f' <div class="eva-course-stats">\n'
1045
+ f' <span><i class="fas fa-book"></i> {notebook_count} notebook{"s" if notebook_count != 1 else ""}</span>\n'
1046
+ f' </div>\n'
1047
+ f' <button class="eva-button eva-course-button">View Notebooks</button>\n'
1048
+ f' </div>\n'
1049
+ f' <div class="eva-course-content">\n'
1050
  f' <div class="eva-notebooks">\n'
1051
  )
1052
 
1053
  for i, notebook in enumerate(course["notebooks"]):
1054
+ # Use original file number instead of sequential numbering
1055
+ notebook_number = notebook.get("original_number", f"{i+1:02d}")
1056
  f.write(
1057
  f' <div class="eva-notebook">\n'
1058
  f' <span class="eva-notebook-number">{notebook_number}</span>\n'
 
1082
  </section>
1083
 
1084
  <footer class="eva-footer">
1085
+ <div class="eva-footer-logo">
1086
+ <a href="https://marimo.io" target="_blank">
1087
+ <img src="https://marimo.io/logotype-wide.svg" alt="Marimo" width="200">
1088
+ </a>
1089
+ </div>
1090
+ <div class="eva-social-links">
1091
+ <a href="https://github.com/marimo-team" target="_blank" aria-label="GitHub"><i class="fab fa-github"></i></a>
1092
+ <a href="https://marimo.io/discord?ref=learn" target="_blank" aria-label="Discord"><i class="fab fa-discord"></i></a>
1093
+ <a href="https://twitter.com/marimo_io" target="_blank" aria-label="Twitter"><i class="fab fa-twitter"></i></a>
1094
+ <a href="https://www.youtube.com/@marimo-io" target="_blank" aria-label="YouTube"><i class="fab fa-youtube"></i></a>
1095
+ <a href="https://www.linkedin.com/company/marimo-io" target="_blank" aria-label="LinkedIn"><i class="fab fa-linkedin"></i></a>
1096
  </div>
1097
  <div class="eva-footer-links">
1098
+ <a href="https://marimo.io" target="_blank">Website</a>
1099
  <a href="https://docs.marimo.io" target="_blank">Documentation</a>
1100
  <a href="https://github.com/marimo-team/learn" target="_blank">GitHub</a>
1101
  </div>
1102
+ <div class="eva-footer-copyright">
1103
+ © 2025 Marimo Inc. All rights reserved.
1104
+ </div>
1105
  </footer>
1106
  </div>
1107
 
1108
  <script>
1109
  document.addEventListener('DOMContentLoaded', function() {
1110
+ // Theme toggle functionality
1111
+ const themeToggle = document.getElementById('themeToggle');
1112
+ const themeIcon = themeToggle.querySelector('i');
1113
+
1114
+ // Check for saved theme preference or use default (dark)
1115
+ const savedTheme = localStorage.getItem('theme') || 'dark';
1116
+ document.documentElement.setAttribute('data-theme', savedTheme);
1117
+ updateThemeIcon(savedTheme);
1118
+
1119
+ // Toggle theme when button is clicked
1120
+ themeToggle.addEventListener('click', () => {
1121
+ const currentTheme = document.documentElement.getAttribute('data-theme');
1122
+ const newTheme = currentTheme === 'dark' ? 'light' : 'dark';
1123
+
1124
+ document.documentElement.setAttribute('data-theme', newTheme);
1125
+ localStorage.setItem('theme', newTheme);
1126
+ updateThemeIcon(newTheme);
1127
+ });
1128
+
1129
+ function updateThemeIcon(theme) {
1130
+ if (theme === 'dark') {
1131
+ themeIcon.className = 'fas fa-sun';
1132
+ } else {
1133
+ themeIcon.className = 'fas fa-moon';
1134
+ }
1135
+ }
1136
+
1137
  // Terminal typing effect for hero text
1138
  const heroTitle = document.querySelector('.eva-hero h1');
1139
  const heroText = document.querySelector('.eva-hero p');
 
1169
 
1170
  typeTitle();
1171
 
1172
+ // Course toggle functionality - flashcard style
1173
  const courseHeaders = document.querySelectorAll('.eva-course-header');
1174
+ const courseButtons = document.querySelectorAll('.eva-course-button');
1175
 
1176
+ // Function to toggle course
1177
+ function toggleCourse(course) {
1178
+ const isActive = course.classList.contains('active');
1179
+
1180
+ // First close all courses
1181
+ document.querySelectorAll('.eva-course').forEach(c => {
1182
+ c.classList.remove('active');
1183
+ });
1184
+
1185
+ // Toggle the clicked course
1186
+ if (!isActive) {
1187
+ course.classList.add('active');
1188
+
1189
+ // Check if the course has any notebooks
1190
+ const notebooks = course.querySelectorAll('.eva-notebook');
1191
+ const content = course.querySelector('.eva-course-content');
1192
+
1193
+ if (notebooks.length === 0 && !content.querySelector('.eva-empty-message')) {
1194
+ // If no notebooks, show a message
1195
+ const emptyMessage = document.createElement('p');
1196
+ emptyMessage.className = 'eva-empty-message';
1197
+ emptyMessage.textContent = 'No notebooks available in this course yet.';
1198
+ emptyMessage.style.color = 'var(--eva-text)';
1199
+ emptyMessage.style.fontStyle = 'italic';
1200
+ emptyMessage.style.opacity = '0.7';
1201
+ emptyMessage.style.textAlign = 'center';
1202
+ emptyMessage.style.padding = '1rem 0';
1203
+ content.appendChild(emptyMessage);
1204
+ }
1205
+ }
1206
+ }
1207
+
1208
+ // Add click event to course headers
1209
  courseHeaders.forEach(header => {
1210
  header.addEventListener('click', function(e) {
1211
  e.preventDefault();
1212
  e.stopPropagation();
1213
 
1214
+ const currentCourse = this.closest('.eva-course');
1215
+ toggleCourse(currentCourse);
1216
+ });
1217
+ });
1218
+
1219
+ // Add click event to course buttons
1220
+ courseButtons.forEach(button => {
1221
+ button.addEventListener('click', function(e) {
1222
+ e.preventDefault();
1223
+ e.stopPropagation();
1224
 
1225
+ const currentCourse = this.closest('.eva-course');
1226
+ toggleCourse(currentCourse);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1227
  });
1228
  });
1229
 
1230
+ // Search functionality with improved matching
1231
  const searchInput = document.getElementById('courseSearch');
1232
  const courses = document.querySelectorAll('.eva-course');
1233
  const notebooks = document.querySelectorAll('.eva-notebook');
 
1267
  // Then show courses and notebooks that match the search
1268
  let hasResults = false;
1269
 
1270
+ // Track which courses have matching notebooks
1271
+ const coursesWithMatchingNotebooks = new Set();
1272
+
1273
+ // First check notebooks
1274
  notebooks.forEach(notebook => {
1275
  const notebookTitle = notebook.querySelector('a').getAttribute('data-notebook-title').toLowerCase();
1276
  const matchesSearch = notebookTitle.includes(searchTerm);
 
1279
 
1280
  if (matchesSearch) {
1281
  const course = notebook.closest('.eva-course');
1282
+ coursesWithMatchingNotebooks.add(course.getAttribute('data-course-id'));
 
1283
  hasResults = true;
1284
  }
1285
  });
1286
 
1287
+ // Then check course titles and descriptions
1288
  courses.forEach(course => {
1289
+ const courseId = course.getAttribute('data-course-id');
1290
  const courseTitle = course.querySelector('.eva-course-title').textContent.toLowerCase();
1291
  const courseDescription = course.querySelector('.eva-course-description').textContent.toLowerCase();
1292
 
1293
+ const courseMatches = courseTitle.includes(searchTerm) || courseDescription.includes(searchTerm);
1294
+
1295
+ // Show course if it matches or has matching notebooks
1296
+ if (courseMatches || coursesWithMatchingNotebooks.has(courseId)) {
1297
  course.style.display = 'block';
1298
  course.classList.add('active');
1299
  hasResults = true;
1300
+
1301
+ // If course matches but doesn't have matching notebooks, show all its notebooks
1302
+ if (courseMatches && !coursesWithMatchingNotebooks.has(courseId)) {
1303
+ course.querySelectorAll('.eva-notebook').forEach(nb => {
1304
+ nb.style.display = 'flex';
1305
+ });
1306
+ }
1307
  }
1308
  });
1309
  });