Srihari Thyagarajan commited on
Commit
a488e86
·
unverified ·
2 Parent(s): 5890b75 d3bf5bc

Merge pull request #67 from marimo-team/haleshot/add-docs-page

Browse files
.github/workflows/deploy.yml ADDED
@@ -0,0 +1,56 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ name: Deploy to GitHub Pages
2
+
3
+ on:
4
+ push:
5
+ branches: ['main']
6
+ workflow_dispatch:
7
+
8
+ concurrency:
9
+ group: 'pages'
10
+ cancel-in-progress: false
11
+
12
+ env:
13
+ UV_SYSTEM_PYTHON: 1
14
+
15
+ jobs:
16
+ build:
17
+ runs-on: ubuntu-latest
18
+ steps:
19
+ - uses: actions/checkout@v4
20
+
21
+ - name: 🚀 Install uv
22
+ uses: astral-sh/setup-uv@v4
23
+
24
+ - name: 🐍 Set up Python
25
+ uses: actions/setup-python@v5
26
+ with:
27
+ python-version: 3.12
28
+
29
+ - name: 📦 Install dependencies
30
+ run: |
31
+ uv pip install marimo
32
+
33
+ - name: 🛠️ Export notebooks
34
+ run: |
35
+ python scripts/build.py
36
+
37
+ - name: 📤 Upload artifact
38
+ uses: actions/upload-pages-artifact@v3
39
+ with:
40
+ path: _site
41
+
42
+ deploy:
43
+ needs: build
44
+
45
+ permissions:
46
+ pages: write
47
+ id-token: write
48
+
49
+ environment:
50
+ name: github-pages
51
+ url: ${{ steps.deployment.outputs.page_url }}
52
+ runs-on: ubuntu-latest
53
+ steps:
54
+ - name: 🚀 Deploy to GitHub Pages
55
+ id: deployment
56
+ uses: actions/deploy-pages@v4
.gitignore CHANGED
@@ -168,4 +168,7 @@ cython_debug/
168
  #.idea/
169
 
170
  # PyPI configuration file
171
- .pypirc
 
 
 
 
168
  #.idea/
169
 
170
  # PyPI configuration file
171
+ .pypirc
172
+
173
+ # Generated site content
174
+ _site/
README.md CHANGED
@@ -58,6 +58,21 @@ Here's a contribution checklist:
58
  If you aren't comfortable adding a new notebook or course, you can also request
59
  what you'd like to see by [filing an issue](https://github.com/marimo-team/learn/issues/new?template=example_request.yaml).
60
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
61
  ## Community
62
 
63
  We're building a community. Come hang out with us!
 
58
  If you aren't comfortable adding a new notebook or course, you can also request
59
  what you'd like to see by [filing an issue](https://github.com/marimo-team/learn/issues/new?template=example_request.yaml).
60
 
61
+ ## Building and Previewing
62
+
63
+ The site is built using a Python script that exports marimo notebooks to HTML and generates an index page.
64
+
65
+ ```bash
66
+ # Build the site
67
+ python scripts/build.py --output-dir _site
68
+
69
+ # Preview the site (builds first)
70
+ python scripts/preview.py
71
+
72
+ # Preview without rebuilding
73
+ python scripts/preview.py --no-build
74
+ ```
75
+
76
  ## Community
77
 
78
  We're building a community. Come hang out with us!
python/006_dictionaries.py CHANGED
@@ -196,13 +196,13 @@ def _():
196
 
197
  @app.cell
198
  def _(mo, nested_data):
199
- mo.md(f"Alice's age: {nested_data["users"]["alice"]["age"]}")
200
  return
201
 
202
 
203
  @app.cell
204
  def _(mo, nested_data):
205
- mo.md(f"Bob's interests: {nested_data["users"]["bob"]["interests"]}")
206
  return
207
 
208
 
 
196
 
197
  @app.cell
198
  def _(mo, nested_data):
199
+ mo.md(f"Alice's age: {nested_data['users']['alice']['age']}")
200
  return
201
 
202
 
203
  @app.cell
204
  def _(mo, nested_data):
205
+ mo.md(f"Bob's interests: {nested_data['users']['bob']['interests']}")
206
  return
207
 
208
 
scripts/build.py ADDED
@@ -0,0 +1,1523 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #!/usr/bin/env python3
2
+
3
+ import os
4
+ import subprocess
5
+ import argparse
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:
13
+ """Export a single marimo notebook to HTML format.
14
+
15
+ Returns:
16
+ bool: True if export succeeded, False otherwise
17
+ """
18
+ output_path = notebook_path.replace(".py", ".html")
19
+
20
+ cmd = ["marimo", "export", "html-wasm"]
21
+ if as_app:
22
+ print(f"Exporting {notebook_path} to {output_path} as app")
23
+ cmd.extend(["--mode", "run", "--no-show-code"])
24
+ else:
25
+ print(f"Exporting {notebook_path} to {output_path} as notebook")
26
+ cmd.extend(["--mode", "edit"])
27
+
28
+ try:
29
+ output_file = os.path.join(output_dir, output_path)
30
+ os.makedirs(os.path.dirname(output_file), exist_ok=True)
31
+
32
+ cmd.extend([notebook_path, "-o", output_file])
33
+ print(f"Running command: {' '.join(cmd)}")
34
+
35
+ # Use Popen to handle interactive prompts
36
+ process = subprocess.Popen(
37
+ cmd,
38
+ stdin=subprocess.PIPE,
39
+ stdout=subprocess.PIPE,
40
+ stderr=subprocess.PIPE,
41
+ text=True
42
+ )
43
+
44
+ # Send 'Y' to the prompt
45
+ stdout, stderr = process.communicate(input="Y\n", timeout=60)
46
+
47
+ if process.returncode != 0:
48
+ print(f"Error exporting {notebook_path}:")
49
+ print(f"Command: {' '.join(cmd)}")
50
+ print(f"Return code: {process.returncode}")
51
+ print(f"Stdout: {stdout}")
52
+ print(f"Stderr: {stderr}")
53
+ return False
54
+
55
+ print(f"Successfully exported {notebook_path} to {output_file}")
56
+ return True
57
+ except subprocess.TimeoutExpired:
58
+ print(f"Timeout exporting {notebook_path} - command took too long to execute")
59
+ return False
60
+ except subprocess.CalledProcessError as e:
61
+ print(f"Error exporting {notebook_path}:")
62
+ print(e.stderr)
63
+ return False
64
+ except Exception as e:
65
+ print(f"Unexpected error exporting {notebook_path}: {e}")
66
+ return False
67
+
68
+
69
+ def get_course_metadata(course_dir: Path) -> Dict[str, Any]:
70
+ """Extract metadata from a course directory."""
71
+ metadata = {
72
+ "id": course_dir.name,
73
+ "title": course_dir.name.replace("_", " ").title(),
74
+ "description": "",
75
+ "notebooks": []
76
+ }
77
+
78
+ # Try to read README.md for description
79
+ readme_path = course_dir / "README.md"
80
+ if readme_path.exists():
81
+ with open(readme_path, "r", encoding="utf-8") as f:
82
+ content = f.read()
83
+ # Extract first paragraph as description
84
+ if content:
85
+ lines = content.split("\n")
86
+ # Skip title line if it exists
87
+ start_idx = 1 if lines and lines[0].startswith("#") else 0
88
+ description_lines = []
89
+ for line in lines[start_idx:]:
90
+ if line.strip() and not line.startswith("#"):
91
+ description_lines.append(line)
92
+ elif description_lines: # Stop at the next heading
93
+ break
94
+ description = " ".join(description_lines).strip()
95
+ # Clean up the description
96
+ description = description.replace("_", "")
97
+ description = description.replace("[work in progress]", "")
98
+ description = description.replace("(https://github.com/marimo-team/learn/issues/51)", "")
99
+ # Remove any other GitHub issue links
100
+ description = re.sub(r'\[.*?\]\(https://github\.com/.*?/issues/\d+\)', '', description)
101
+ description = re.sub(r'https://github\.com/.*?/issues/\d+', '', description)
102
+ # Clean up any double spaces
103
+ description = re.sub(r'\s+', ' ', description).strip()
104
+ metadata["description"] = description
105
+
106
+ return metadata
107
+
108
+
109
+ def organize_notebooks_by_course(all_notebooks: List[str]) -> Dict[str, Dict[str, Any]]:
110
+ """Organize notebooks by course."""
111
+ courses = {}
112
+
113
+ for notebook_path in all_notebooks:
114
+ path = Path(notebook_path)
115
+ course_id = path.parts[0]
116
+
117
+ if course_id not in courses:
118
+ course_dir = Path(course_id)
119
+ courses[course_id] = get_course_metadata(course_dir)
120
+
121
+ # Extract notebook info
122
+ filename = path.name
123
+ notebook_id = path.stem
124
+
125
+ # Try to extract order from filename (e.g., 001_numbers.py -> 1)
126
+ order = 999
127
+ if "_" in notebook_id:
128
+ try:
129
+ order_str = notebook_id.split("_")[0]
130
+ order = int(order_str)
131
+ except ValueError:
132
+ pass
133
+
134
+ # Create display name by removing order prefix and underscores
135
+ display_name = notebook_id
136
+ if "_" in notebook_id:
137
+ display_name = "_".join(notebook_id.split("_")[1:])
138
+
139
+ # Convert display name to title case, but handle italics properly
140
+ parts = display_name.split("_")
141
+ formatted_parts = []
142
+
143
+ i = 0
144
+ while i < len(parts):
145
+ if i + 1 < len(parts) and parts[i] == "" and parts[i+1] == "":
146
+ # Skip empty parts that might come from consecutive underscores
147
+ i += 2
148
+ continue
149
+
150
+ if i + 1 < len(parts) and (parts[i] == "" or parts[i+1] == ""):
151
+ # This is an italics marker
152
+ if parts[i] == "":
153
+ # Opening italics
154
+ text_part = parts[i+1].replace("_", " ").title()
155
+ formatted_parts.append(f"<em>{text_part}</em>")
156
+ i += 2
157
+ else:
158
+ # Text followed by italics marker
159
+ text_part = parts[i].replace("_", " ").title()
160
+ formatted_parts.append(text_part)
161
+ i += 1
162
+ else:
163
+ # Regular text
164
+ text_part = parts[i].replace("_", " ").title()
165
+ formatted_parts.append(text_part)
166
+ i += 1
167
+
168
+ display_name = " ".join(formatted_parts)
169
+
170
+ courses[course_id]["notebooks"].append({
171
+ "id": notebook_id,
172
+ "path": notebook_path,
173
+ "display_name": display_name,
174
+ "order": order,
175
+ "original_number": notebook_id.split("_")[0] if "_" in notebook_id else ""
176
+ })
177
+
178
+ # Sort notebooks by order
179
+ for course_id in courses:
180
+ courses[course_id]["notebooks"].sort(key=lambda x: x["order"])
181
+
182
+ return courses
183
+
184
+
185
+ def generate_eva_css() -> str:
186
+ """Generate Neon Genesis Evangelion inspired CSS with light/dark mode support."""
187
+ return """
188
+ :root {
189
+ /* Light mode colors (default) */
190
+ --eva-purple: #7209b7;
191
+ --eva-green: #1c7361;
192
+ --eva-orange: #e65100;
193
+ --eva-blue: #0039cb;
194
+ --eva-red: #c62828;
195
+ --eva-black: #f5f5f5;
196
+ --eva-dark: #e0e0e0;
197
+ --eva-terminal-bg: rgba(255, 255, 255, 0.9);
198
+ --eva-text: #333333;
199
+ --eva-border-radius: 4px;
200
+ --eva-transition: all 0.3s ease;
201
+ }
202
+
203
+ /* Dark mode colors */
204
+ [data-theme="dark"] {
205
+ --eva-purple: #9a1eb3;
206
+ --eva-green: #1c7361;
207
+ --eva-orange: #ff6600;
208
+ --eva-blue: #0066ff;
209
+ --eva-red: #ff0000;
210
+ --eva-black: #111111;
211
+ --eva-dark: #222222;
212
+ --eva-terminal-bg: rgba(0, 0, 0, 0.85);
213
+ --eva-text: #e0e0e0;
214
+ }
215
+
216
+ body {
217
+ background-color: var(--eva-black);
218
+ color: var(--eva-text);
219
+ font-family: 'Courier New', monospace;
220
+ margin: 0;
221
+ padding: 0;
222
+ line-height: 1.6;
223
+ transition: background-color 0.3s ease, color 0.3s ease;
224
+ }
225
+
226
+ .eva-container {
227
+ max-width: 1200px;
228
+ margin: 0 auto;
229
+ padding: 2rem;
230
+ }
231
+
232
+ .eva-header {
233
+ border-bottom: 2px solid var(--eva-green);
234
+ padding-bottom: 1rem;
235
+ margin-bottom: 2rem;
236
+ display: flex;
237
+ justify-content: space-between;
238
+ align-items: center;
239
+ position: sticky;
240
+ top: 0;
241
+ background-color: var(--eva-black);
242
+ z-index: 100;
243
+ backdrop-filter: blur(5px);
244
+ padding-top: 1rem;
245
+ transition: background-color 0.3s ease;
246
+ }
247
+
248
+ [data-theme="light"] .eva-header {
249
+ background-color: rgba(245, 245, 245, 0.95);
250
+ }
251
+
252
+ .eva-logo {
253
+ font-size: 2.5rem;
254
+ font-weight: bold;
255
+ color: var(--eva-green);
256
+ text-transform: uppercase;
257
+ letter-spacing: 2px;
258
+ text-shadow: 0 0 10px rgba(28, 115, 97, 0.5);
259
+ }
260
+
261
+ [data-theme="light"] .eva-logo {
262
+ text-shadow: 0 0 10px rgba(28, 115, 97, 0.3);
263
+ }
264
+
265
+ .eva-nav {
266
+ display: flex;
267
+ gap: 1.5rem;
268
+ align-items: center;
269
+ }
270
+
271
+ .eva-nav a {
272
+ color: var(--eva-text);
273
+ text-decoration: none;
274
+ text-transform: uppercase;
275
+ font-size: 0.9rem;
276
+ letter-spacing: 1px;
277
+ transition: color 0.3s;
278
+ position: relative;
279
+ padding: 0.5rem 0;
280
+ }
281
+
282
+ .eva-nav a:hover {
283
+ color: var(--eva-green);
284
+ }
285
+
286
+ .eva-nav a:hover::after {
287
+ content: '';
288
+ position: absolute;
289
+ bottom: -5px;
290
+ left: 0;
291
+ width: 100%;
292
+ height: 2px;
293
+ background-color: var(--eva-green);
294
+ animation: scanline 1.5s linear infinite;
295
+ }
296
+
297
+ .theme-toggle {
298
+ background: none;
299
+ border: none;
300
+ color: var(--eva-text);
301
+ cursor: pointer;
302
+ font-size: 1.2rem;
303
+ padding: 0.5rem;
304
+ margin-left: 1rem;
305
+ transition: color 0.3s;
306
+ }
307
+
308
+ .theme-toggle:hover {
309
+ color: var(--eva-green);
310
+ }
311
+
312
+ .eva-hero {
313
+ background-color: var(--eva-terminal-bg);
314
+ border: 1px solid var(--eva-green);
315
+ padding: 3rem 2rem;
316
+ margin-bottom: 3rem;
317
+ position: relative;
318
+ overflow: hidden;
319
+ border-radius: var(--eva-border-radius);
320
+ display: flex;
321
+ flex-direction: column;
322
+ align-items: flex-start;
323
+ background-image: linear-gradient(45deg, rgba(0, 0, 0, 0.9), rgba(0, 0, 0, 0.7)), url('https://raw.githubusercontent.com/marimo-team/marimo/main/docs/_static/marimo-logotype-thick.svg');
324
+ background-size: cover;
325
+ background-position: center;
326
+ background-blend-mode: overlay;
327
+ transition: background-color 0.3s ease, border-color 0.3s ease;
328
+ }
329
+
330
+ [data-theme="light"] .eva-hero {
331
+ 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');
332
+ }
333
+
334
+ .eva-hero::before {
335
+ content: '';
336
+ position: absolute;
337
+ top: 0;
338
+ left: 0;
339
+ width: 100%;
340
+ height: 2px;
341
+ background-color: var(--eva-green);
342
+ animation: scanline 3s linear infinite;
343
+ }
344
+
345
+ .eva-hero h1 {
346
+ font-size: 2.5rem;
347
+ margin-bottom: 1rem;
348
+ color: var(--eva-green);
349
+ text-transform: uppercase;
350
+ letter-spacing: 2px;
351
+ text-shadow: 0 0 10px rgba(28, 115, 97, 0.5);
352
+ }
353
+
354
+ [data-theme="light"] .eva-hero h1 {
355
+ text-shadow: 0 0 10px rgba(28, 115, 97, 0.3);
356
+ }
357
+
358
+ .eva-hero p {
359
+ font-size: 1.1rem;
360
+ max-width: 800px;
361
+ margin-bottom: 2rem;
362
+ line-height: 1.8;
363
+ }
364
+
365
+ .eva-features {
366
+ display: grid;
367
+ grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
368
+ gap: 2rem;
369
+ margin-bottom: 3rem;
370
+ }
371
+
372
+ .eva-feature {
373
+ background-color: var(--eva-terminal-bg);
374
+ border: 1px solid var(--eva-blue);
375
+ padding: 1.5rem;
376
+ border-radius: var(--eva-border-radius);
377
+ transition: var(--eva-transition);
378
+ position: relative;
379
+ overflow: hidden;
380
+ }
381
+
382
+ .eva-feature:hover {
383
+ transform: translateY(-5px);
384
+ box-shadow: 0 10px 20px rgba(0, 102, 255, 0.2);
385
+ }
386
+
387
+ .eva-feature-icon {
388
+ font-size: 2rem;
389
+ margin-bottom: 1rem;
390
+ color: var(--eva-blue);
391
+ }
392
+
393
+ .eva-feature h3 {
394
+ font-size: 1.3rem;
395
+ margin-bottom: 1rem;
396
+ color: var(--eva-blue);
397
+ }
398
+
399
+ .eva-section-title {
400
+ font-size: 2rem;
401
+ color: var(--eva-green);
402
+ margin-bottom: 2rem;
403
+ text-transform: uppercase;
404
+ letter-spacing: 2px;
405
+ text-align: center;
406
+ position: relative;
407
+ padding-bottom: 1rem;
408
+ }
409
+
410
+ .eva-section-title::after {
411
+ content: '';
412
+ position: absolute;
413
+ bottom: 0;
414
+ left: 50%;
415
+ transform: translateX(-50%);
416
+ width: 100px;
417
+ height: 2px;
418
+ background-color: var(--eva-green);
419
+ }
420
+
421
+ /* Flashcard view for courses */
422
+ .eva-courses {
423
+ display: grid;
424
+ grid-template-columns: repeat(auto-fill, minmax(350px, 1fr));
425
+ gap: 2rem;
426
+ }
427
+
428
+ .eva-course {
429
+ background-color: var(--eva-terminal-bg);
430
+ border: 1px solid var(--eva-purple);
431
+ border-radius: var(--eva-border-radius);
432
+ transition: var(--eva-transition), height 0.4s cubic-bezier(0.19, 1, 0.22, 1);
433
+ position: relative;
434
+ overflow: hidden;
435
+ height: 350px;
436
+ display: flex;
437
+ flex-direction: column;
438
+ }
439
+
440
+ .eva-course:hover {
441
+ transform: translateY(-5px);
442
+ box-shadow: 0 10px 20px rgba(154, 30, 179, 0.3);
443
+ }
444
+
445
+ .eva-course::after {
446
+ content: '';
447
+ position: absolute;
448
+ bottom: 0;
449
+ left: 0;
450
+ width: 100%;
451
+ height: 2px;
452
+ background-color: var(--eva-purple);
453
+ animation: scanline 2s linear infinite;
454
+ }
455
+
456
+ .eva-course-badge {
457
+ position: absolute;
458
+ top: 15px;
459
+ right: -40px;
460
+ background: linear-gradient(135deg, var(--eva-orange) 0%, #ff9500 100%);
461
+ color: var(--eva-black);
462
+ font-size: 0.65rem;
463
+ padding: 0.3rem 2.5rem;
464
+ text-transform: uppercase;
465
+ font-weight: bold;
466
+ z-index: 3;
467
+ letter-spacing: 1px;
468
+ box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3);
469
+ transform: rotate(45deg);
470
+ text-shadow: 0 1px 1px rgba(255, 255, 255, 0.2);
471
+ border-top: 1px solid rgba(255, 255, 255, 0.3);
472
+ border-bottom: 1px solid rgba(0, 0, 0, 0.2);
473
+ white-space: nowrap;
474
+ overflow: hidden;
475
+ }
476
+
477
+ .eva-course-badge i {
478
+ margin-right: 4px;
479
+ font-size: 0.7rem;
480
+ }
481
+
482
+ [data-theme="light"] .eva-course-badge {
483
+ box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2);
484
+ text-shadow: 0 1px 1px rgba(255, 255, 255, 0.4);
485
+ }
486
+
487
+ .eva-course-badge::before {
488
+ content: '';
489
+ position: absolute;
490
+ left: 0;
491
+ top: 0;
492
+ width: 100%;
493
+ height: 100%;
494
+ background: linear-gradient(to right, transparent, rgba(255, 255, 255, 0.3), transparent);
495
+ animation: scanline 2s linear infinite;
496
+ }
497
+
498
+ .eva-course-header {
499
+ padding: 1rem 1.5rem;
500
+ cursor: pointer;
501
+ display: flex;
502
+ justify-content: space-between;
503
+ align-items: center;
504
+ border-bottom: 1px solid rgba(154, 30, 179, 0.3);
505
+ z-index: 2;
506
+ background-color: var(--eva-terminal-bg);
507
+ position: absolute;
508
+ top: 0;
509
+ left: 0;
510
+ width: 100%;
511
+ height: 3.5rem;
512
+ box-sizing: border-box;
513
+ }
514
+
515
+ .eva-course-title {
516
+ font-size: 1.3rem;
517
+ color: var(--eva-purple);
518
+ text-transform: uppercase;
519
+ letter-spacing: 1px;
520
+ margin: 0;
521
+ }
522
+
523
+ .eva-course-toggle {
524
+ color: var(--eva-purple);
525
+ font-size: 1.5rem;
526
+ transition: transform 0.4s cubic-bezier(0.19, 1, 0.22, 1);
527
+ }
528
+
529
+ .eva-course.active .eva-course-toggle {
530
+ transform: rotate(180deg);
531
+ }
532
+
533
+ .eva-course-front {
534
+ display: flex;
535
+ flex-direction: column;
536
+ justify-content: space-between;
537
+ padding: 1.5rem;
538
+ margin-top: 3.5rem;
539
+ transition: opacity 0.3s ease, transform 0.3s ease;
540
+ position: absolute;
541
+ top: 0;
542
+ left: 0;
543
+ width: 100%;
544
+ height: calc(100% - 3.5rem);
545
+ background-color: var(--eva-terminal-bg);
546
+ z-index: 1;
547
+ box-sizing: border-box;
548
+ }
549
+
550
+ .eva-course.active .eva-course-front {
551
+ opacity: 0;
552
+ transform: translateY(-10px);
553
+ pointer-events: none;
554
+ }
555
+
556
+ .eva-course-description {
557
+ margin-top: 0.5rem;
558
+ margin-bottom: 1.5rem;
559
+ font-size: 0.9rem;
560
+ line-height: 1.6;
561
+ flex-grow: 1;
562
+ overflow: hidden;
563
+ display: -webkit-box;
564
+ -webkit-line-clamp: 4;
565
+ -webkit-box-orient: vertical;
566
+ max-height: 150px;
567
+ }
568
+
569
+ .eva-course-stats {
570
+ display: flex;
571
+ justify-content: space-between;
572
+ font-size: 0.8rem;
573
+ color: var(--eva-text);
574
+ opacity: 0.7;
575
+ }
576
+
577
+ .eva-course-content {
578
+ position: absolute;
579
+ top: 3.5rem;
580
+ left: 0;
581
+ width: 100%;
582
+ height: calc(100% - 3.5rem);
583
+ padding: 1.5rem;
584
+ background-color: var(--eva-terminal-bg);
585
+ transition: opacity 0.3s ease, transform 0.3s ease;
586
+ opacity: 0;
587
+ transform: translateY(10px);
588
+ pointer-events: none;
589
+ overflow-y: auto;
590
+ z-index: 1;
591
+ box-sizing: border-box;
592
+ }
593
+
594
+ .eva-course.active .eva-course-content {
595
+ opacity: 1;
596
+ transform: translateY(0);
597
+ pointer-events: auto;
598
+ }
599
+
600
+ .eva-course.active {
601
+ height: auto;
602
+ min-height: 350px;
603
+ max-height: 800px;
604
+ transition: height 0.4s cubic-bezier(0.19, 1, 0.22, 1), transform 0.3s ease, box-shadow 0.3s ease;
605
+ }
606
+
607
+ .eva-notebooks {
608
+ margin-top: 1rem;
609
+ display: grid;
610
+ grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
611
+ gap: 0.75rem;
612
+ }
613
+
614
+ .eva-notebook {
615
+ margin-bottom: 0.5rem;
616
+ padding: 0.75rem;
617
+ border-left: 2px solid var(--eva-blue);
618
+ transition: all 0.25s ease;
619
+ display: flex;
620
+ align-items: center;
621
+ background-color: rgba(0, 0, 0, 0.2);
622
+ border-radius: 0 var(--eva-border-radius) var(--eva-border-radius) 0;
623
+ opacity: 1;
624
+ transform: translateX(0);
625
+ }
626
+
627
+ [data-theme="light"] .eva-notebook {
628
+ background-color: rgba(0, 0, 0, 0.05);
629
+ }
630
+
631
+ .eva-notebook:hover {
632
+ background-color: rgba(0, 102, 255, 0.1);
633
+ padding-left: 1rem;
634
+ transform: translateX(3px);
635
+ }
636
+
637
+ .eva-notebook a {
638
+ color: var(--eva-text);
639
+ text-decoration: none;
640
+ display: block;
641
+ font-size: 0.9rem;
642
+ flex-grow: 1;
643
+ }
644
+
645
+ .eva-notebook a:hover {
646
+ color: var(--eva-blue);
647
+ }
648
+
649
+ .eva-notebook-number {
650
+ color: var(--eva-blue);
651
+ font-size: 0.8rem;
652
+ margin-right: 0.75rem;
653
+ opacity: 0.7;
654
+ min-width: 24px;
655
+ font-weight: bold;
656
+ }
657
+
658
+ .eva-button {
659
+ display: inline-block;
660
+ background-color: transparent;
661
+ color: var(--eva-green);
662
+ border: 1px solid var(--eva-green);
663
+ padding: 0.7rem 1.5rem;
664
+ text-decoration: none;
665
+ text-transform: uppercase;
666
+ font-size: 0.9rem;
667
+ letter-spacing: 1px;
668
+ transition: var(--eva-transition);
669
+ cursor: pointer;
670
+ border-radius: var(--eva-border-radius);
671
+ position: relative;
672
+ overflow: hidden;
673
+ }
674
+
675
+ .eva-button:hover {
676
+ background-color: var(--eva-green);
677
+ color: var(--eva-black);
678
+ }
679
+
680
+ .eva-button::after {
681
+ content: '';
682
+ position: absolute;
683
+ top: 0;
684
+ left: -100%;
685
+ width: 100%;
686
+ height: 100%;
687
+ background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.2), transparent);
688
+ transition: 0.5s;
689
+ }
690
+
691
+ .eva-button:hover::after {
692
+ left: 100%;
693
+ }
694
+
695
+ .eva-course-button {
696
+ margin-top: 1rem;
697
+ margin-bottom: 1rem;
698
+ align-self: center;
699
+ }
700
+
701
+ .eva-cta {
702
+ background-color: var(--eva-terminal-bg);
703
+ border: 1px solid var(--eva-orange);
704
+ padding: 3rem 2rem;
705
+ margin: 4rem 0;
706
+ text-align: center;
707
+ border-radius: var(--eva-border-radius);
708
+ position: relative;
709
+ overflow: hidden;
710
+ }
711
+
712
+ .eva-cta h2 {
713
+ font-size: 2rem;
714
+ color: var(--eva-orange);
715
+ margin-bottom: 1.5rem;
716
+ text-transform: uppercase;
717
+ }
718
+
719
+ .eva-cta p {
720
+ max-width: 600px;
721
+ margin: 0 auto 2rem;
722
+ font-size: 1.1rem;
723
+ }
724
+
725
+ .eva-cta .eva-button {
726
+ color: var(--eva-orange);
727
+ border-color: var(--eva-orange);
728
+ }
729
+
730
+ .eva-cta .eva-button:hover {
731
+ background-color: var(--eva-orange);
732
+ color: var(--eva-black);
733
+ }
734
+
735
+ .eva-footer {
736
+ margin-top: 4rem;
737
+ padding-top: 2rem;
738
+ border-top: 2px solid var(--eva-green);
739
+ display: flex;
740
+ flex-direction: column;
741
+ align-items: center;
742
+ gap: 2rem;
743
+ }
744
+
745
+ .eva-footer-logo {
746
+ max-width: 200px;
747
+ margin-bottom: 1rem;
748
+ }
749
+
750
+ .eva-footer-links {
751
+ display: flex;
752
+ gap: 1.5rem;
753
+ margin-bottom: 1.5rem;
754
+ }
755
+
756
+ .eva-footer-links a {
757
+ color: var(--eva-text);
758
+ text-decoration: none;
759
+ transition: var(--eva-transition);
760
+ }
761
+
762
+ .eva-footer-links a:hover {
763
+ color: var(--eva-green);
764
+ }
765
+
766
+ .eva-social-links {
767
+ display: flex;
768
+ gap: 1.5rem;
769
+ margin-bottom: 1.5rem;
770
+ }
771
+
772
+ .eva-social-links a {
773
+ color: var(--eva-text);
774
+ font-size: 1.5rem;
775
+ transition: var(--eva-transition);
776
+ }
777
+
778
+ .eva-social-links a:hover {
779
+ color: var(--eva-green);
780
+ transform: translateY(-3px);
781
+ }
782
+
783
+ .eva-footer-copyright {
784
+ font-size: 0.9rem;
785
+ text-align: center;
786
+ }
787
+
788
+ .eva-search {
789
+ position: relative;
790
+ margin-bottom: 3rem;
791
+ }
792
+
793
+ .eva-search input {
794
+ width: 100%;
795
+ padding: 1rem;
796
+ background-color: var(--eva-terminal-bg);
797
+ border: 1px solid var(--eva-green);
798
+ color: var(--eva-text);
799
+ font-family: 'Courier New', monospace;
800
+ font-size: 1rem;
801
+ border-radius: var(--eva-border-radius);
802
+ outline: none;
803
+ transition: var(--eva-transition);
804
+ }
805
+
806
+ .eva-search input:focus {
807
+ box-shadow: 0 0 10px rgba(28, 115, 97, 0.3);
808
+ }
809
+
810
+ .eva-search input::placeholder {
811
+ color: rgba(224, 224, 224, 0.5);
812
+ }
813
+
814
+ [data-theme="light"] .eva-search input::placeholder {
815
+ color: rgba(51, 51, 51, 0.5);
816
+ }
817
+
818
+ .eva-search-icon {
819
+ position: absolute;
820
+ right: 1rem;
821
+ top: 50%;
822
+ transform: translateY(-50%);
823
+ color: var(--eva-green);
824
+ font-size: 1.2rem;
825
+ }
826
+
827
+ @keyframes scanline {
828
+ 0% {
829
+ transform: translateX(-100%);
830
+ }
831
+ 100% {
832
+ transform: translateX(100%);
833
+ }
834
+ }
835
+
836
+ @keyframes blink {
837
+ 0%, 100% {
838
+ opacity: 1;
839
+ }
840
+ 50% {
841
+ opacity: 0;
842
+ }
843
+ }
844
+
845
+ .eva-cursor {
846
+ display: inline-block;
847
+ width: 10px;
848
+ height: 1.2em;
849
+ background-color: var(--eva-green);
850
+ margin-left: 2px;
851
+ animation: blink 1s infinite;
852
+ vertical-align: middle;
853
+ }
854
+
855
+ @media (max-width: 768px) {
856
+ .eva-courses {
857
+ grid-template-columns: 1fr;
858
+ }
859
+
860
+ .eva-header {
861
+ flex-direction: column;
862
+ align-items: flex-start;
863
+ padding: 1rem;
864
+ }
865
+
866
+ .eva-nav {
867
+ margin-top: 1rem;
868
+ flex-wrap: wrap;
869
+ }
870
+
871
+ .eva-hero {
872
+ padding: 2rem 1rem;
873
+ }
874
+
875
+ .eva-hero h1 {
876
+ font-size: 2rem;
877
+ }
878
+
879
+ .eva-features {
880
+ grid-template-columns: 1fr;
881
+ }
882
+
883
+ .eva-footer {
884
+ flex-direction: column;
885
+ align-items: center;
886
+ text-align: center;
887
+ }
888
+
889
+ .eva-notebooks {
890
+ grid-template-columns: 1fr;
891
+ }
892
+ }
893
+
894
+ .eva-course.closing .eva-course-content {
895
+ opacity: 0;
896
+ transform: translateY(10px);
897
+ transition: opacity 0.2s ease, transform 0.2s ease;
898
+ }
899
+
900
+ .eva-course.closing .eva-course-front {
901
+ opacity: 1;
902
+ transform: translateY(0);
903
+ transition: opacity 0.3s ease 0.1s, transform 0.3s ease 0.1s;
904
+ }
905
+ """
906
+
907
+
908
+ def get_html_header():
909
+ """Generate the HTML header with CSS and meta tags."""
910
+ return """<!DOCTYPE html>
911
+ <html lang="en" data-theme="light">
912
+ <head>
913
+ <meta charset="UTF-8">
914
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
915
+ <title>Marimo Learn - Interactive Educational Notebooks</title>
916
+ <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
917
+ <style>
918
+ {css}
919
+ </style>
920
+ </head>
921
+ <body>
922
+ <div class="eva-container">
923
+ <header class="eva-header">
924
+ <div class="eva-logo">MARIMO LEARN</div>
925
+ <nav class="eva-nav">
926
+ <a href="#features">Features</a>
927
+ <a href="#courses">Courses</a>
928
+ <a href="#contribute">Contribute</a>
929
+ <a href="https://docs.marimo.io" target="_blank">Documentation</a>
930
+ <a href="https://github.com/marimo-team/learn" target="_blank">GitHub</a>
931
+ <button id="themeToggle" class="theme-toggle" aria-label="Toggle dark/light mode">
932
+ <i class="fas fa-moon"></i>
933
+ </button>
934
+ </nav>
935
+ </header>"""
936
+
937
+
938
+ def get_html_hero_section():
939
+ """Generate the hero section of the page."""
940
+ return """
941
+ <section class="eva-hero">
942
+ <h1>Interactive Learning with Marimo<span class="eva-cursor"></span></h1>
943
+ <p>
944
+ A curated collection of educational notebooks covering computer science,
945
+ mathematics, data science, and more. Built with marimo - the reactive
946
+ Python notebook that makes data exploration delightful.
947
+ </p>
948
+ <a href="#courses" class="eva-button">Explore Courses</a>
949
+ </section>"""
950
+
951
+
952
+ def get_html_features_section():
953
+ """Generate the features section of the page."""
954
+ return """
955
+ <section id="features">
956
+ <h2 class="eva-section-title">Why Marimo Learn?</h2>
957
+ <div class="eva-features">
958
+ <div class="eva-feature">
959
+ <div class="eva-feature-icon"><i class="fas fa-bolt"></i></div>
960
+ <h3>Reactive Notebooks</h3>
961
+ <p>Experience the power of reactive programming with marimo notebooks that automatically update when dependencies change.</p>
962
+ </div>
963
+ <div class="eva-feature">
964
+ <div class="eva-feature-icon"><i class="fas fa-code"></i></div>
965
+ <h3>Learn by Doing</h3>
966
+ <p>Interactive examples and exercises help you understand concepts through hands-on practice.</p>
967
+ </div>
968
+ <div class="eva-feature">
969
+ <div class="eva-feature-icon"><i class="fas fa-graduation-cap"></i></div>
970
+ <h3>Comprehensive Courses</h3>
971
+ <p>From Python basics to advanced optimization techniques, our courses cover a wide range of topics.</p>
972
+ </div>
973
+ </div>
974
+ </section>"""
975
+
976
+
977
+ def get_html_courses_start():
978
+ """Generate the beginning of the courses section."""
979
+ return """
980
+ <section id="courses">
981
+ <h2 class="eva-section-title">Explore Courses</h2>
982
+ <div class="eva-search">
983
+ <input type="text" id="courseSearch" placeholder="Search courses and notebooks...">
984
+ <span class="eva-search-icon"><i class="fas fa-search"></i></span>
985
+ </div>
986
+ <div class="eva-courses">"""
987
+
988
+
989
+ def generate_course_card(course, notebook_count, is_wip):
990
+ """Generate HTML for a single course card."""
991
+ html = f'<div class="eva-course" data-course-id="{course["id"]}">\n'
992
+
993
+ # Add WIP badge if needed
994
+ if is_wip:
995
+ html += ' <div class="eva-course-badge"><i class="fas fa-code-branch"></i> In Progress</div>\n'
996
+
997
+ html += f''' <div class="eva-course-header">
998
+ <h2 class="eva-course-title">{course["title"]}</h2>
999
+ <span class="eva-course-toggle"><i class="fas fa-chevron-down"></i></span>
1000
+ </div>
1001
+ <div class="eva-course-front">
1002
+ <p class="eva-course-description">{course["description"]}</p>
1003
+ <div class="eva-course-stats">
1004
+ <span><i class="fas fa-book"></i> {notebook_count} notebook{"s" if notebook_count != 1 else ""}</span>
1005
+ </div>
1006
+ <button class="eva-button eva-course-button">View Notebooks</button>
1007
+ </div>
1008
+ <div class="eva-course-content">
1009
+ <div class="eva-notebooks">
1010
+ '''
1011
+
1012
+ # Add notebooks
1013
+ for i, notebook in enumerate(course["notebooks"]):
1014
+ notebook_number = notebook.get("original_number", f"{i+1:02d}")
1015
+ html += f''' <div class="eva-notebook">
1016
+ <span class="eva-notebook-number">{notebook_number}</span>
1017
+ <a href="{notebook["path"].replace(".py", ".html")}" data-notebook-title="{notebook["display_name"]}">{notebook["display_name"]}</a>
1018
+ </div>
1019
+ '''
1020
+
1021
+ html += ''' </div>
1022
+ </div>
1023
+ </div>
1024
+ '''
1025
+ return html
1026
+
1027
+
1028
+ def generate_course_cards(courses):
1029
+ """Generate HTML for all course cards."""
1030
+ html = ""
1031
+
1032
+ # Define the custom order for courses
1033
+ course_order = ["python", "probability", "polars", "optimization", "functional_programming"]
1034
+
1035
+ # Create a dictionary of courses by ID for easy lookup
1036
+ courses_by_id = {course["id"]: course for course in courses.values()}
1037
+
1038
+ # Determine which courses are "work in progress" based on description or notebook count
1039
+ work_in_progress = set()
1040
+ for course_id, course in courses_by_id.items():
1041
+ # Consider a course as "work in progress" if it has few notebooks or contains specific phrases
1042
+ if (len(course["notebooks"]) < 5 or
1043
+ "work in progress" in course["description"].lower() or
1044
+ "help us add" in course["description"].lower() or
1045
+ "check back later" in course["description"].lower()):
1046
+ work_in_progress.add(course_id)
1047
+
1048
+ # First output courses in the specified order
1049
+ for course_id in course_order:
1050
+ if course_id in courses_by_id:
1051
+ course = courses_by_id[course_id]
1052
+
1053
+ # Skip if no notebooks
1054
+ if not course["notebooks"]:
1055
+ continue
1056
+
1057
+ # Count notebooks
1058
+ notebook_count = len(course["notebooks"])
1059
+
1060
+ # Determine if this course is a work in progress
1061
+ is_wip = course_id in work_in_progress
1062
+
1063
+ html += generate_course_card(course, notebook_count, is_wip)
1064
+
1065
+ # Remove from the dictionary so we don't output it again
1066
+ del courses_by_id[course_id]
1067
+
1068
+ # Then output any remaining courses alphabetically
1069
+ sorted_remaining_courses = sorted(courses_by_id.values(), key=lambda x: x["title"])
1070
+
1071
+ for course in sorted_remaining_courses:
1072
+ # Skip if no notebooks
1073
+ if not course["notebooks"]:
1074
+ continue
1075
+
1076
+ # Count notebooks
1077
+ notebook_count = len(course["notebooks"])
1078
+
1079
+ # Determine if this course is a work in progress
1080
+ is_wip = course["id"] in work_in_progress
1081
+
1082
+ html += generate_course_card(course, notebook_count, is_wip)
1083
+
1084
+ return html
1085
+
1086
+
1087
+ def get_html_courses_end():
1088
+ """Generate the end of the courses section."""
1089
+ return """ </div>
1090
+ </section>"""
1091
+
1092
+
1093
+ def get_html_contribute_section():
1094
+ """Generate the contribute section."""
1095
+ return """
1096
+ <section id="contribute" class="eva-cta">
1097
+ <h2>Contribute to Marimo Learn</h2>
1098
+ <p>
1099
+ Help us expand our collection of educational notebooks. Whether you're an expert in machine learning,
1100
+ statistics, or any other field, your contributions are welcome!
1101
+ </p>
1102
+ <a href="https://github.com/marimo-team/learn" target="_blank" class="eva-button">
1103
+ <i class="fab fa-github"></i> Contribute on GitHub
1104
+ </a>
1105
+ </section>"""
1106
+
1107
+
1108
+ def get_html_footer():
1109
+ """Generate the page footer."""
1110
+ return """
1111
+ <footer class="eva-footer">
1112
+ <div class="eva-footer-logo">
1113
+ <a href="https://marimo.io" target="_blank">
1114
+ <img src="https://marimo.io/logotype-wide.svg" alt="Marimo" width="200">
1115
+ </a>
1116
+ </div>
1117
+ <div class="eva-social-links">
1118
+ <a href="https://github.com/marimo-team" target="_blank" aria-label="GitHub"><i class="fab fa-github"></i></a>
1119
+ <a href="https://marimo.io/discord?ref=learn" target="_blank" aria-label="Discord"><i class="fab fa-discord"></i></a>
1120
+ <a href="https://twitter.com/marimo_io" target="_blank" aria-label="Twitter"><i class="fab fa-twitter"></i></a>
1121
+ <a href="https://www.youtube.com/@marimo-team" target="_blank" aria-label="YouTube"><i class="fab fa-youtube"></i></a>
1122
+ <a href="https://www.linkedin.com/company/marimo-io" target="_blank" aria-label="LinkedIn"><i class="fab fa-linkedin"></i></a>
1123
+ </div>
1124
+ <div class="eva-footer-links">
1125
+ <a href="https://marimo.io" target="_blank">Website</a>
1126
+ <a href="https://docs.marimo.io" target="_blank">Documentation</a>
1127
+ <a href="https://github.com/marimo-team/learn" target="_blank">GitHub</a>
1128
+ </div>
1129
+ <div class="eva-footer-copyright">
1130
+ © 2025 Marimo Inc. All rights reserved.
1131
+ </div>
1132
+ </footer>"""
1133
+
1134
+
1135
+ def get_html_scripts():
1136
+ """Generate the JavaScript for the page."""
1137
+ return """
1138
+ <script>
1139
+ // Set light theme as default immediately
1140
+ document.documentElement.setAttribute('data-theme', 'light');
1141
+
1142
+ document.addEventListener('DOMContentLoaded', function() {
1143
+ // Theme toggle functionality
1144
+ const themeToggle = document.getElementById('themeToggle');
1145
+ const themeIcon = themeToggle.querySelector('i');
1146
+
1147
+ // Update theme icon based on current theme
1148
+ updateThemeIcon('light');
1149
+
1150
+ // Check localStorage for saved theme preference
1151
+ const savedTheme = localStorage.getItem('theme');
1152
+ if (savedTheme && savedTheme !== 'light') {
1153
+ document.documentElement.setAttribute('data-theme', savedTheme);
1154
+ updateThemeIcon(savedTheme);
1155
+ }
1156
+
1157
+ // Toggle theme when button is clicked
1158
+ themeToggle.addEventListener('click', () => {
1159
+ const currentTheme = document.documentElement.getAttribute('data-theme');
1160
+ const newTheme = currentTheme === 'dark' ? 'light' : 'dark';
1161
+
1162
+ document.documentElement.setAttribute('data-theme', newTheme);
1163
+ localStorage.setItem('theme', newTheme);
1164
+ updateThemeIcon(newTheme);
1165
+ });
1166
+
1167
+ function updateThemeIcon(theme) {
1168
+ if (theme === 'dark') {
1169
+ themeIcon.className = 'fas fa-sun';
1170
+ } else {
1171
+ themeIcon.className = 'fas fa-moon';
1172
+ }
1173
+ }
1174
+
1175
+ // Terminal typing effect for hero text
1176
+ const heroTitle = document.querySelector('.eva-hero h1');
1177
+ const heroText = document.querySelector('.eva-hero p');
1178
+ const cursor = document.querySelector('.eva-cursor');
1179
+
1180
+ const originalTitle = heroTitle.textContent;
1181
+ const originalText = heroText.textContent.trim();
1182
+
1183
+ heroTitle.textContent = '';
1184
+ heroText.textContent = '';
1185
+
1186
+ let titleIndex = 0;
1187
+ let textIndex = 0;
1188
+
1189
+ function typeTitle() {
1190
+ if (titleIndex < originalTitle.length) {
1191
+ heroTitle.textContent += originalTitle.charAt(titleIndex);
1192
+ titleIndex++;
1193
+ setTimeout(typeTitle, 50);
1194
+ } else {
1195
+ cursor.style.display = 'none';
1196
+ setTimeout(typeText, 500);
1197
+ }
1198
+ }
1199
+
1200
+ function typeText() {
1201
+ if (textIndex < originalText.length) {
1202
+ heroText.textContent += originalText.charAt(textIndex);
1203
+ textIndex++;
1204
+ setTimeout(typeText, 20);
1205
+ }
1206
+ }
1207
+
1208
+ typeTitle();
1209
+
1210
+ // Course toggle functionality - flashcard style
1211
+ const courseHeaders = document.querySelectorAll('.eva-course-header');
1212
+ const courseButtons = document.querySelectorAll('.eva-course-button');
1213
+
1214
+ // Function to toggle course
1215
+ function toggleCourse(course) {
1216
+ const isActive = course.classList.contains('active');
1217
+
1218
+ // First close all courses with a slight delay for better visual effect
1219
+ document.querySelectorAll('.eva-course.active').forEach(c => {
1220
+ if (c !== course) {
1221
+ // Add a closing class for animation
1222
+ c.classList.add('closing');
1223
+ // Remove active class after a short delay
1224
+ setTimeout(() => {
1225
+ c.classList.remove('active');
1226
+ c.classList.remove('closing');
1227
+ }, 300);
1228
+ }
1229
+ });
1230
+
1231
+ // Toggle the clicked course
1232
+ if (!isActive) {
1233
+ // Add a small delay before opening to allow others to close
1234
+ setTimeout(() => {
1235
+ course.classList.add('active');
1236
+
1237
+ // Check if the course has any notebooks
1238
+ const notebooks = course.querySelectorAll('.eva-notebook');
1239
+ const content = course.querySelector('.eva-course-content');
1240
+
1241
+ if (notebooks.length === 0 && !content.querySelector('.eva-empty-message')) {
1242
+ // If no notebooks, show a message
1243
+ const emptyMessage = document.createElement('p');
1244
+ emptyMessage.className = 'eva-empty-message';
1245
+ emptyMessage.textContent = 'No notebooks available in this course yet.';
1246
+ emptyMessage.style.color = 'var(--eva-text)';
1247
+ emptyMessage.style.fontStyle = 'italic';
1248
+ emptyMessage.style.opacity = '0.7';
1249
+ emptyMessage.style.textAlign = 'center';
1250
+ emptyMessage.style.padding = '1rem 0';
1251
+ content.appendChild(emptyMessage);
1252
+ }
1253
+
1254
+ // Animate notebooks to appear sequentially
1255
+ notebooks.forEach((notebook, index) => {
1256
+ notebook.style.opacity = '0';
1257
+ notebook.style.transform = 'translateX(-10px)';
1258
+ setTimeout(() => {
1259
+ notebook.style.opacity = '1';
1260
+ notebook.style.transform = 'translateX(0)';
1261
+ }, 50 + (index * 30)); // Stagger the animations
1262
+ });
1263
+ }, 100);
1264
+ }
1265
+ }
1266
+
1267
+ // Add click event to course headers
1268
+ courseHeaders.forEach(header => {
1269
+ header.addEventListener('click', function(e) {
1270
+ e.preventDefault();
1271
+ e.stopPropagation();
1272
+
1273
+ const currentCourse = this.closest('.eva-course');
1274
+ toggleCourse(currentCourse);
1275
+ });
1276
+ });
1277
+
1278
+ // Add click event to course buttons
1279
+ courseButtons.forEach(button => {
1280
+ button.addEventListener('click', function(e) {
1281
+ e.preventDefault();
1282
+ e.stopPropagation();
1283
+
1284
+ const currentCourse = this.closest('.eva-course');
1285
+ toggleCourse(currentCourse);
1286
+ });
1287
+ });
1288
+
1289
+ // Search functionality with improved matching
1290
+ const searchInput = document.getElementById('courseSearch');
1291
+ const courses = document.querySelectorAll('.eva-course');
1292
+ const notebooks = document.querySelectorAll('.eva-notebook');
1293
+
1294
+ searchInput.addEventListener('input', function() {
1295
+ const searchTerm = this.value.toLowerCase();
1296
+
1297
+ if (searchTerm === '') {
1298
+ // Reset all visibility
1299
+ courses.forEach(course => {
1300
+ course.style.display = 'block';
1301
+ course.classList.remove('active');
1302
+ });
1303
+
1304
+ notebooks.forEach(notebook => {
1305
+ notebook.style.display = 'flex';
1306
+ });
1307
+
1308
+ // Open the first course with notebooks by default when search is cleared
1309
+ for (let i = 0; i < courses.length; i++) {
1310
+ const courseNotebooks = courses[i].querySelectorAll('.eva-notebook');
1311
+ if (courseNotebooks.length > 0) {
1312
+ courses[i].classList.add('active');
1313
+ break;
1314
+ }
1315
+ }
1316
+
1317
+ return;
1318
+ }
1319
+
1320
+ // First hide all courses
1321
+ courses.forEach(course => {
1322
+ course.style.display = 'none';
1323
+ course.classList.remove('active');
1324
+ });
1325
+
1326
+ // Then show courses and notebooks that match the search
1327
+ let hasResults = false;
1328
+
1329
+ // Track which courses have matching notebooks
1330
+ const coursesWithMatchingNotebooks = new Set();
1331
+
1332
+ // First check notebooks
1333
+ notebooks.forEach(notebook => {
1334
+ const notebookTitle = notebook.querySelector('a').getAttribute('data-notebook-title').toLowerCase();
1335
+ const matchesSearch = notebookTitle.includes(searchTerm);
1336
+
1337
+ notebook.style.display = matchesSearch ? 'flex' : 'none';
1338
+
1339
+ if (matchesSearch) {
1340
+ const course = notebook.closest('.eva-course');
1341
+ coursesWithMatchingNotebooks.add(course.getAttribute('data-course-id'));
1342
+ hasResults = true;
1343
+ }
1344
+ });
1345
+
1346
+ // Then check course titles and descriptions
1347
+ courses.forEach(course => {
1348
+ const courseId = course.getAttribute('data-course-id');
1349
+ const courseTitle = course.querySelector('.eva-course-title').textContent.toLowerCase();
1350
+ const courseDescription = course.querySelector('.eva-course-description').textContent.toLowerCase();
1351
+
1352
+ const courseMatches = courseTitle.includes(searchTerm) || courseDescription.includes(searchTerm);
1353
+
1354
+ // Show course if it matches or has matching notebooks
1355
+ if (courseMatches || coursesWithMatchingNotebooks.has(courseId)) {
1356
+ course.style.display = 'block';
1357
+ course.classList.add('active');
1358
+ hasResults = true;
1359
+
1360
+ // If course matches but doesn't have matching notebooks, show all its notebooks
1361
+ if (courseMatches && !coursesWithMatchingNotebooks.has(courseId)) {
1362
+ course.querySelectorAll('.eva-notebook').forEach(nb => {
1363
+ nb.style.display = 'flex';
1364
+ });
1365
+ }
1366
+ }
1367
+ });
1368
+ });
1369
+
1370
+ // Open the first course with notebooks by default
1371
+ let firstCourseWithNotebooks = null;
1372
+ for (let i = 0; i < courses.length; i++) {
1373
+ const courseNotebooks = courses[i].querySelectorAll('.eva-notebook');
1374
+ if (courseNotebooks.length > 0) {
1375
+ firstCourseWithNotebooks = courses[i];
1376
+ break;
1377
+ }
1378
+ }
1379
+
1380
+ if (firstCourseWithNotebooks) {
1381
+ firstCourseWithNotebooks.classList.add('active');
1382
+ } else if (courses.length > 0) {
1383
+ // If no courses have notebooks, just open the first one
1384
+ courses[0].classList.add('active');
1385
+ }
1386
+
1387
+ // Smooth scrolling for anchor links
1388
+ document.querySelectorAll('a[href^="#"]').forEach(anchor => {
1389
+ anchor.addEventListener('click', function(e) {
1390
+ e.preventDefault();
1391
+
1392
+ const targetId = this.getAttribute('href');
1393
+ const targetElement = document.querySelector(targetId);
1394
+
1395
+ if (targetElement) {
1396
+ window.scrollTo({
1397
+ top: targetElement.offsetTop - 100,
1398
+ behavior: 'smooth'
1399
+ });
1400
+ }
1401
+ });
1402
+ });
1403
+ });
1404
+ </script>"""
1405
+
1406
+
1407
+ def get_html_footer_closing():
1408
+ """Generate closing HTML tags."""
1409
+ return """
1410
+ </div>
1411
+ </body>
1412
+ </html>"""
1413
+
1414
+
1415
+ def generate_index(courses: Dict[str, Dict[str, Any]], output_dir: str) -> None:
1416
+ """Generate the index.html file with Neon Genesis Evangelion aesthetics."""
1417
+ print("Generating index.html")
1418
+
1419
+ index_path = os.path.join(output_dir, "index.html")
1420
+ os.makedirs(output_dir, exist_ok=True)
1421
+
1422
+ try:
1423
+ with open(index_path, "w", encoding="utf-8") as f:
1424
+ # Build the page HTML from individual components
1425
+ header = get_html_header().format(css=generate_eva_css())
1426
+ hero = get_html_hero_section()
1427
+ features = get_html_features_section()
1428
+ courses_start = get_html_courses_start()
1429
+ course_cards = generate_course_cards(courses)
1430
+ courses_end = get_html_courses_end()
1431
+ contribute = get_html_contribute_section()
1432
+ footer = get_html_footer()
1433
+ scripts = get_html_scripts()
1434
+ closing = get_html_footer_closing()
1435
+
1436
+ # Write all elements to the file
1437
+ f.write(header)
1438
+ f.write(hero)
1439
+ f.write(features)
1440
+ f.write(courses_start)
1441
+ f.write(course_cards)
1442
+ f.write(courses_end)
1443
+ f.write(contribute)
1444
+ f.write(footer)
1445
+ f.write(scripts)
1446
+ f.write(closing)
1447
+
1448
+ except IOError as e:
1449
+ print(f"Error generating index.html: {e}")
1450
+
1451
+
1452
+ def main() -> None:
1453
+ parser = argparse.ArgumentParser(description="Build marimo notebooks")
1454
+ parser.add_argument(
1455
+ "--output-dir", default="_site", help="Output directory for built files"
1456
+ )
1457
+ parser.add_argument(
1458
+ "--course-dirs", nargs="+", default=None,
1459
+ help="Specific course directories to build (default: all directories with .py files)"
1460
+ )
1461
+ args = parser.parse_args()
1462
+
1463
+ # Find all course directories (directories containing .py files)
1464
+ all_notebooks: List[str] = []
1465
+
1466
+ # Directories to exclude from course detection
1467
+ excluded_dirs = ["scripts", "env", "__pycache__", ".git", ".github", "assets"]
1468
+
1469
+ if args.course_dirs:
1470
+ course_dirs = args.course_dirs
1471
+ else:
1472
+ # Automatically detect course directories (any directory with .py files)
1473
+ course_dirs = []
1474
+ for item in os.listdir("."):
1475
+ if (os.path.isdir(item) and
1476
+ not item.startswith(".") and
1477
+ not item.startswith("_") and
1478
+ item not in excluded_dirs):
1479
+ # Check if directory contains .py files
1480
+ if list(Path(item).glob("*.py")):
1481
+ course_dirs.append(item)
1482
+
1483
+ print(f"Found course directories: {', '.join(course_dirs)}")
1484
+
1485
+ for directory in course_dirs:
1486
+ dir_path = Path(directory)
1487
+ if not dir_path.exists():
1488
+ print(f"Warning: Directory not found: {dir_path}")
1489
+ continue
1490
+
1491
+ notebooks = [str(path) for path in dir_path.rglob("*.py")
1492
+ if not path.name.startswith("_") and "/__pycache__/" not in str(path)]
1493
+ all_notebooks.extend(notebooks)
1494
+
1495
+ if not all_notebooks:
1496
+ print("No notebooks found!")
1497
+ return
1498
+
1499
+ # Export notebooks sequentially
1500
+ successful_notebooks = []
1501
+ for nb in all_notebooks:
1502
+ # Determine if notebook should be exported as app or notebook
1503
+ # For now, export all as notebooks
1504
+ if export_html_wasm(nb, args.output_dir, as_app=False):
1505
+ successful_notebooks.append(nb)
1506
+
1507
+ # Organize notebooks by course (only include successfully exported notebooks)
1508
+ courses = organize_notebooks_by_course(successful_notebooks)
1509
+
1510
+ # Generate index with organized courses
1511
+ generate_index(courses, args.output_dir)
1512
+
1513
+ # Save course data as JSON for potential use by other tools
1514
+ courses_json_path = os.path.join(args.output_dir, "courses.json")
1515
+ with open(courses_json_path, "w", encoding="utf-8") as f:
1516
+ json.dump(courses, f, indent=2)
1517
+
1518
+ print(f"Build complete! Site generated in {args.output_dir}")
1519
+ print(f"Successfully exported {len(successful_notebooks)} out of {len(all_notebooks)} notebooks")
1520
+
1521
+
1522
+ if __name__ == "__main__":
1523
+ main()
scripts/preview.py ADDED
@@ -0,0 +1,76 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #!/usr/bin/env python3
2
+
3
+ import os
4
+ import subprocess
5
+ import argparse
6
+ import webbrowser
7
+ import time
8
+ import sys
9
+ from pathlib import Path
10
+
11
+ def main():
12
+ parser = argparse.ArgumentParser(description="Build and preview marimo notebooks site")
13
+ parser.add_argument(
14
+ "--port", default=8000, type=int, help="Port to run the server on"
15
+ )
16
+ parser.add_argument(
17
+ "--no-build", action="store_true", help="Skip building the site (just serve existing files)"
18
+ )
19
+ parser.add_argument(
20
+ "--output-dir", default="_site", help="Output directory for built files"
21
+ )
22
+ args = parser.parse_args()
23
+
24
+ # Store the current directory
25
+ original_dir = os.getcwd()
26
+
27
+ try:
28
+ # Build the site if not skipped
29
+ if not args.no_build:
30
+ print("Building site...")
31
+ build_script = Path("scripts/build.py")
32
+ if not build_script.exists():
33
+ print(f"Error: Build script not found at {build_script}")
34
+ return 1
35
+
36
+ result = subprocess.run(
37
+ [sys.executable, str(build_script), "--output-dir", args.output_dir],
38
+ check=False
39
+ )
40
+ if result.returncode != 0:
41
+ print("Warning: Build process completed with errors.")
42
+
43
+ # Check if the output directory exists
44
+ output_dir = Path(args.output_dir)
45
+ if not output_dir.exists():
46
+ print(f"Error: Output directory '{args.output_dir}' does not exist.")
47
+ return 1
48
+
49
+ # Change to the output directory
50
+ os.chdir(args.output_dir)
51
+
52
+ # Open the browser
53
+ url = f"http://localhost:{args.port}"
54
+ print(f"Opening {url} in your browser...")
55
+ webbrowser.open(url)
56
+
57
+ # Start the server
58
+ print(f"Starting server on port {args.port}...")
59
+ print("Press Ctrl+C to stop the server")
60
+
61
+ # Use the appropriate Python executable
62
+ subprocess.run([sys.executable, "-m", "http.server", str(args.port)])
63
+
64
+ return 0
65
+ except KeyboardInterrupt:
66
+ print("\nServer stopped.")
67
+ return 0
68
+ except Exception as e:
69
+ print(f"Error: {e}")
70
+ return 1
71
+ finally:
72
+ # Always return to the original directory
73
+ os.chdir(original_dir)
74
+
75
+ if __name__ == "__main__":
76
+ sys.exit(main())