phihung commited on
Commit
3b5501b
·
1 Parent(s): 1108474

refactor: Use attrs-first style

Browse files
src/tutorial/_01_click_to_edit.py CHANGED
@@ -15,27 +15,20 @@ def page():
15
 
16
  @app.get("/contact")
17
  def get_contact():
18
- return Div(
19
  Div(P(f"Name : {current.name}")),
20
  Div(P(f"Email : {current.email}")),
21
  Button("Click To Edit", hx_get=contact_edit.rt(), cls="btn primary"),
22
- hx_target="this",
23
- hx_swap="outerHTML",
24
- cls="container",
25
  )
26
 
27
 
28
  @app.get("/contact/edit")
29
  def contact_edit():
30
- return Form(
31
  Div(Label("Name"), Input(type="text", name="name", value=current.name)),
32
  Div(Label("Email"), Input(type="email", name="email", value=current.email)),
33
  Button("Submit", cls="btn"),
34
  Button("Cancel", hx_get=get_contact.rt(), cls="btn"),
35
- hx_put=put_contact.rt(),
36
- hx_target="this",
37
- hx_swap="outerHTML",
38
- cls="container",
39
  )
40
 
41
 
@@ -50,7 +43,7 @@ DESC = "Demonstrates inline editing of a data object"
50
  DOC = """
51
  The click to edit pattern provides a way to offer inline editing of all or part of a record without a page refresh.
52
 
53
- - This pattern starts with a UI that shows the details of a contact. The div has a button that will get the editing UI for the contact from /contact/1/edit
54
 
55
  ::get_contact::
56
 
@@ -58,7 +51,7 @@ The click to edit pattern provides a way to offer inline editing of all or part
58
 
59
  ::contact_edit::
60
 
61
- The form issues a PUT back to /contact/1, following the usual REST-ful pattern.
62
 
63
  ::put_contact::
64
  """
 
15
 
16
  @app.get("/contact")
17
  def get_contact():
18
+ return Div(hx_target="this", hx_swap="outerHTML", cls="container")(
19
  Div(P(f"Name : {current.name}")),
20
  Div(P(f"Email : {current.email}")),
21
  Button("Click To Edit", hx_get=contact_edit.rt(), cls="btn primary"),
 
 
 
22
  )
23
 
24
 
25
  @app.get("/contact/edit")
26
  def contact_edit():
27
+ return Form(hx_put=put_contact.rt(), hx_target="this", hx_swap="outerHTML", cls="container")(
28
  Div(Label("Name"), Input(type="text", name="name", value=current.name)),
29
  Div(Label("Email"), Input(type="email", name="email", value=current.email)),
30
  Button("Submit", cls="btn"),
31
  Button("Cancel", hx_get=get_contact.rt(), cls="btn"),
 
 
 
 
32
  )
33
 
34
 
 
43
  DOC = """
44
  The click to edit pattern provides a way to offer inline editing of all or part of a record without a page refresh.
45
 
46
+ - This pattern starts with a UI that shows the details of a contact. The div has a button that will get the editing UI for the contact from /contact/edit
47
 
48
  ::get_contact::
49
 
 
51
 
52
  ::contact_edit::
53
 
54
+ The form issues a PUT back to /contact, following the usual REST-ful pattern.
55
 
56
  ::put_contact::
57
  """
src/tutorial/_03_click_to_load.py CHANGED
@@ -6,14 +6,13 @@ app, rt = fast_app()
6
  # fmt: off
7
  @app.get
8
  def page():
9
- return Div(
10
  Table(
11
  Thead(Tr(Th("Name"), Th("ID"))),
12
  Tbody(load_contacts(page=1)),
13
- ),
14
- cls="container overflow-auto",
15
  )
16
- # fmt: on
17
 
18
 
19
  @app.get("/contacts")
@@ -23,18 +22,13 @@ def load_contacts(page: int, limit: int = 5):
23
 
24
 
25
  def make_last_row(page):
26
- return Tr(
27
- Td(
28
- Button(
29
- "Load More Agents...",
30
- hx_get=load_contacts.rt(page=page + 1),
31
- hx_swap="outerHTML",
32
- cls="btn primary",
33
- ),
34
- colspan="3",
35
  ),
36
- hx_target="this",
37
  )
 
38
 
39
 
40
  DESC = "Demonstrates clicking to load more rows in a table"
 
6
  # fmt: off
7
  @app.get
8
  def page():
9
+ return Div(cls="container overflow-auto")(
10
  Table(
11
  Thead(Tr(Th("Name"), Th("ID"))),
12
  Tbody(load_contacts(page=1)),
13
+ )
 
14
  )
15
+
16
 
17
 
18
  @app.get("/contacts")
 
22
 
23
 
24
  def make_last_row(page):
25
+ return Tr(hx_target="this")(
26
+ Td(colspan="3")(
27
+ Button("Load More Agents...",
28
+ hx_get=load_contacts.rt(page=page + 1), hx_swap="outerHTML", cls="btn primary"),
 
 
 
 
 
29
  ),
 
30
  )
31
+ # fmt: on
32
 
33
 
34
  DESC = "Demonstrates clicking to load more rows in a table"
src/tutorial/_04_delete_row.py CHANGED
@@ -12,10 +12,10 @@ app, rt = fast_app(hdrs=[Style(css)])
12
 
13
  @app.get
14
  def page():
15
- return Div(
16
  Table(
17
  Thead(Tr(Th("Name"), Th("Email"), Th())),
18
- Tbody(
19
  Tr(
20
  Td("Joe Smith"),
21
  Td("[email protected]"),
@@ -36,12 +36,8 @@ def page():
36
  Td("[email protected]"),
37
  Td(Button("Delete", hx_delete="/contacts/3", cls="btn secondary")),
38
  ),
39
- hx_confirm="Are you sure?",
40
- hx_target="closest tr",
41
- hx_swap="outerHTML swap:1s",
42
  ),
43
  ),
44
- cls="container overflow-auto",
45
  )
46
 
47
 
 
12
 
13
  @app.get
14
  def page():
15
+ return Div(cls="container overflow-auto")(
16
  Table(
17
  Thead(Tr(Th("Name"), Th("Email"), Th())),
18
+ Tbody(hx_confirm="Are you sure?", hx_target="closest tr", hx_swap="outerHTML swap:1s")(
19
  Tr(
20
  Td("Joe Smith"),
21
  Td("[email protected]"),
 
36
  Td("[email protected]"),
37
  Td(Button("Delete", hx_delete="/contacts/3", cls="btn secondary")),
38
  ),
 
 
 
39
  ),
40
  ),
 
41
  )
42
 
43
 
src/tutorial/_05_edit_row.py CHANGED
@@ -12,16 +12,13 @@ DATA = [
12
 
13
  @app.get
14
  def page():
15
- return Div(
16
  Table(
17
  Thead(Tr(Th("Name"), Th("Email"), Th())),
18
- Tbody(
19
- *(get_contact(i) for i in range(4)),
20
- hx_target="closest tr",
21
- hx_swap="outerHTML",
22
  ),
23
  ),
24
- cls="container-fluid",
25
  )
26
 
27
 
@@ -38,16 +35,13 @@ def get_contact(idx: int):
38
  @app.get("/contact/{idx}/edit")
39
  def edit_view(idx: int):
40
  name, email = DATA[idx]
41
- return Tr(
42
  Td(Input(name="name", value=name)),
43
  Td(Input(name="email", value=email)),
44
  Td(
45
  Button("Cancel", hx_get=f"/contact/{idx}", cls="btn secondary"),
46
  Button("Save", hx_put=f"/contact/{idx}", hx_include="closest tr", cls="btn primary"),
47
  ),
48
- hx_trigger="cancel",
49
- hx_get=f"/contact/{idx}",
50
- cls="editing",
51
  )
52
 
53
 
 
12
 
13
  @app.get
14
  def page():
15
+ return Div(cls="container-fluid")(
16
  Table(
17
  Thead(Tr(Th("Name"), Th("Email"), Th())),
18
+ Tbody(hx_target="closest tr", hx_swap="outerHTML")(
19
+ tuple(get_contact(i) for i in range(4)),
 
 
20
  ),
21
  ),
 
22
  )
23
 
24
 
 
35
  @app.get("/contact/{idx}/edit")
36
  def edit_view(idx: int):
37
  name, email = DATA[idx]
38
+ return Tr(hx_trigger="cancel", hx_get=f"/contact/{idx}", cls="editing")(
39
  Td(Input(name="name", value=name)),
40
  Td(Input(name="email", value=email)),
41
  Td(
42
  Button("Cancel", hx_get=f"/contact/{idx}", cls="btn secondary"),
43
  Button("Save", hx_put=f"/contact/{idx}", hx_include="closest tr", cls="btn primary"),
44
  ),
 
 
 
45
  )
46
 
47
 
src/tutorial/_06_lazy_loading.py CHANGED
@@ -16,11 +16,8 @@ app, rt = fast_app(hdrs=[Style(css)])
16
 
17
  @app.get
18
  def page():
19
- return Div(
20
  Img(src="/img/bars.svg", alt="Result loading...", cls="htmx-indicator", width="150"),
21
- hx_get=get_content.rt(),
22
- hx_trigger="load",
23
- cls="container",
24
  )
25
 
26
 
 
16
 
17
  @app.get
18
  def page():
19
+ return Div(hx_get=get_content.rt(), hx_trigger="load", cls="container")(
20
  Img(src="/img/bars.svg", alt="Result loading...", cls="htmx-indicator", width="150"),
 
 
 
21
  )
22
 
23
 
src/tutorial/_07_inline_validation.py CHANGED
@@ -16,17 +16,14 @@ app, rt = fast_app(hdrs=[Style(css)])
16
 
17
  @app.get
18
  def page():
19
- return Div(
20
  H3("Signup Form"),
21
- NotStr(
22
- "Enter an email into the input below and on tab out it will be validated. Only <ins>[email protected]</ins> will pass."
23
- ),
24
  Form(
25
  make_email_field("", error=False, touched=False),
26
  Div(Label("Name"), Input(name="name")),
27
  Button("Submit", disabled="", cls="btn primary"),
28
  ),
29
- cls="container",
30
  )
31
 
32
 
@@ -37,14 +34,12 @@ def validate_email(email: str):
37
 
38
 
39
  def make_email_field(value: str, error: bool, touched: bool):
40
- return Div(
 
41
  Label("Email"),
42
  Input(name="email", hx_post=validate_email.rt(), hx_indicator="#ind", value=value),
43
  Img(id="ind", src="/img/bars.svg", cls="htmx-indicator"),
44
  Span("Please enter a valid email address", style="color:red;") if error else None,
45
- hx_target="this",
46
- hx_swap="outerHTML",
47
- cls="" if not touched else "error" if error else "valid",
48
  )
49
 
50
 
 
16
 
17
  @app.get
18
  def page():
19
+ return Div(cls="container")(
20
  H3("Signup Form"),
21
+ NotStr("Enter an email and on tab out it will be validated. Only <ins>[email protected]</ins> will pass."),
 
 
22
  Form(
23
  make_email_field("", error=False, touched=False),
24
  Div(Label("Name"), Input(name="name")),
25
  Button("Submit", disabled="", cls="btn primary"),
26
  ),
 
27
  )
28
 
29
 
 
34
 
35
 
36
  def make_email_field(value: str, error: bool, touched: bool):
37
+ cls = "" if not touched else "error" if error else "valid"
38
+ return Div(hx_target="this", hx_swap="outerHTML", cls=cls)(
39
  Label("Email"),
40
  Input(name="email", hx_post=validate_email.rt(), hx_indicator="#ind", value=value),
41
  Img(id="ind", src="/img/bars.svg", cls="htmx-indicator"),
42
  Span("Please enter a valid email address", style="color:red;") if error else None,
 
 
 
43
  )
44
 
45
 
src/tutorial/_08_infinite_scroll.py CHANGED
@@ -6,12 +6,11 @@ app, rt = fast_app()
6
  # fmt: off
7
  @app.get
8
  def page():
9
- return Div(
10
  Table(
11
  Thead(Tr(Th("Name"), Th("ID"))),
12
  Tbody(load_contacts(page=1)),
13
  ),
14
- cls="container",
15
  )
16
  # fmt: on
17
 
@@ -23,12 +22,9 @@ def load_contacts(page: int, limit: int = 5):
23
 
24
 
25
  def make_last_row(page, limit):
26
- return Tr(
27
  Td("Smith"),
28
  Td(page * limit),
29
- hx_trigger="revealed",
30
- hx_swap="afterend",
31
- hx_get=load_contacts.rt(page=page + 1),
32
  )
33
 
34
 
 
6
  # fmt: off
7
  @app.get
8
  def page():
9
+ return Div(cls="container")(
10
  Table(
11
  Thead(Tr(Th("Name"), Th("ID"))),
12
  Tbody(load_contacts(page=1)),
13
  ),
 
14
  )
15
  # fmt: on
16
 
 
22
 
23
 
24
  def make_last_row(page, limit):
25
+ return Tr(hx_trigger="revealed", hx_swap="afterend", hx_get=load_contacts.rt(page=page + 1))(
26
  Td("Smith"),
27
  Td(page * limit),
 
 
 
28
  )
29
 
30
 
src/tutorial/_09_active_search.py CHANGED
@@ -12,19 +12,14 @@ def page():
12
  return Div(
13
  H3("Search Contacts"),
14
  Input(
15
- type="search",
16
- name="query",
17
- placeholder="Begin Typing To Search Users...",
18
- hx_post=search.rt(),
19
- hx_trigger="input changed delay:500ms, search",
20
- hx_target="#search-results",
21
- hx_indicator=".htmx-indicator",
22
- cls="form-control",
23
  ),
24
  Span(Img(src="/img/bars.svg"), "Searching...", cls="htmx-indicator"),
25
  Table(
26
  Thead(Tr(Th("First Name"), Th("Last Name"), Th("Email"))),
27
- Tbody(id="search-results"),
28
  ),
29
  )
30
  # fmt: on
 
12
  return Div(
13
  H3("Search Contacts"),
14
  Input(
15
+ hx_post=search.rt(), hx_target="#results",
16
+ hx_trigger="input changed delay:500ms, search", hx_indicator=".htmx-indicator",
17
+ type="search", name="query", placeholder="Begin Typing To Search Users...",
 
 
 
 
 
18
  ),
19
  Span(Img(src="/img/bars.svg"), "Searching...", cls="htmx-indicator"),
20
  Table(
21
  Thead(Tr(Th("First Name"), Th("Last Name"), Th("Email"))),
22
+ Tbody(id="results"),
23
  ),
24
  )
25
  # fmt: on
src/tutorial/_10_progress_bar.py CHANGED
@@ -33,11 +33,9 @@ current = 1
33
 
34
  @app.get
35
  def page():
36
- return Div(
37
  H3("Start Progress"),
38
  Button("Start Job", hx_post=start.rt(), cls="btn primary"),
39
- hx_target="this",
40
- hx_swap="outerHTML",
41
  )
42
 
43
 
@@ -45,24 +43,16 @@ def page():
45
  def start():
46
  global current
47
  current = 1
48
- return Div(
49
  H3("Running", role="status", id="pblabel", tabindex="-1", autofocus=""),
50
- Div(
51
- get_progress(),
52
- hx_get=get_progress.rt(),
53
- hx_trigger="every 600ms",
54
- hx_target="this",
55
- hx_swap="innerHTML",
56
  ),
57
- hx_trigger="done",
58
- hx_get=job_finished.rt(),
59
- hx_swap="outerHTML",
60
- hx_target="this",
61
  )
62
 
63
 
64
  @app.get
65
- def get_progress():
66
  global current
67
  if current <= 100:
68
  current += 20
@@ -72,12 +62,10 @@ def get_progress():
72
 
73
  @app.get
74
  def job_finished():
75
- return Div(
76
  H3("Complete", role="status", id="pblabel", tabindex="-1", autofocus=""),
77
  Div(Div(style="width:100%", cls="progress-bar"), cls="progress"),
78
  Button("Restart Job", hx_post=start.rt(), cls="btn primary show"),
79
- hx_swap="outerHTML",
80
- hx_target="this",
81
  )
82
 
83
 
@@ -88,7 +76,7 @@ This example shows how to implement a smoothly scrolling progress bar.
88
  We start with an initial state with a button that issues a POST to /start to begin the job:
89
  ::page::
90
  This progress bar is updated every 600 milliseconds, with the “width” style attribute and aria-valuenow attributed set to current progress value. Because there is an id on the progress bar div, htmx will smoothly transition between requests by settling the style attribute into its new value. This, when coupled with CSS transitions, makes the visual transition continuous rather than jumpy.
91
- ::start get_progress::
92
  Finally, when the process is complete, a server returns HX-Trigger: done header, which triggers an update of the UI to “Complete” state with a restart button added to the UI:
93
  ::job_finished::
94
  """
 
33
 
34
  @app.get
35
  def page():
36
+ return Div(hx_target="this", hx_swap="outerHTML")(
37
  H3("Start Progress"),
38
  Button("Start Job", hx_post=start.rt(), cls="btn primary"),
 
 
39
  )
40
 
41
 
 
43
  def start():
44
  global current
45
  current = 1
46
+ return Div(hx_trigger="done", hx_get=job_finished.rt(), hx_swap="outerHTML", hx_target="this")(
47
  H3("Running", role="status", id="pblabel", tabindex="-1", autofocus=""),
48
+ Div(hx_get=progress_bar.rt(), hx_trigger="every 600ms", hx_target="this", hx_swap="innerHTML")(
49
+ progress_bar(),
 
 
 
 
50
  ),
 
 
 
 
51
  )
52
 
53
 
54
  @app.get
55
+ def progress_bar():
56
  global current
57
  if current <= 100:
58
  current += 20
 
62
 
63
  @app.get
64
  def job_finished():
65
+ return Div(hx_swap="outerHTML", hx_target="this")(
66
  H3("Complete", role="status", id="pblabel", tabindex="-1", autofocus=""),
67
  Div(Div(style="width:100%", cls="progress-bar"), cls="progress"),
68
  Button("Restart Job", hx_post=start.rt(), cls="btn primary show"),
 
 
69
  )
70
 
71
 
 
76
  We start with an initial state with a button that issues a POST to /start to begin the job:
77
  ::page::
78
  This progress bar is updated every 600 milliseconds, with the “width” style attribute and aria-valuenow attributed set to current progress value. Because there is an id on the progress bar div, htmx will smoothly transition between requests by settling the style attribute into its new value. This, when coupled with CSS transitions, makes the visual transition continuous rather than jumpy.
79
+ ::start progress_bar::
80
  Finally, when the process is complete, a server returns HX-Trigger: done header, which triggers an update of the UI to “Complete” state with a restart button added to the UI:
81
  ::job_finished::
82
  """
src/tutorial/_11_cascading_select.py CHANGED
@@ -7,14 +7,12 @@ app, rt = fast_app()
7
 
8
  @app.get
9
  def page():
10
- return Div(
11
  H3("Pick A Make/Model"),
12
  Form(
13
  Div(
14
  Label("Make"),
15
- Select(
16
- name="make", hx_get=load_models.rt(sleep=1), hx_target="#models", hx_indicator=".htmx-indicator"
17
- )(
18
  Option("Audi", value="audi"),
19
  Option("Toyota", value="toyota"),
20
  Option("BMW", value="bmw"),
@@ -22,16 +20,15 @@ def page():
22
  ),
23
  Div(
24
  Label("Model"),
25
- Select(load_models("audi"), id="models"),
26
  Img(width="20", src="/img/bars.svg", cls="htmx-indicator"),
27
  ),
28
  ),
29
- cls="container",
30
  )
31
 
32
 
33
  @app.get
34
- def load_models(make: str, sleep: int = 0):
35
  time.sleep(sleep)
36
  cars = {
37
  "audi": ["A1", "A4", "A6"],
@@ -52,7 +49,7 @@ Here is the code:
52
  ::page::
53
 
54
  When a request is made to the /models end point, we return the models for that make:
55
- ::load_models::
56
 
57
  And they become available in the model select.
58
  """
 
7
 
8
  @app.get
9
  def page():
10
+ return Div(cls="container")(
11
  H3("Pick A Make/Model"),
12
  Form(
13
  Div(
14
  Label("Make"),
15
+ Select(name="make", hx_get=models.rt(sleep=1), hx_target="#models", hx_indicator=".htmx-indicator")(
 
 
16
  Option("Audi", value="audi"),
17
  Option("Toyota", value="toyota"),
18
  Option("BMW", value="bmw"),
 
20
  ),
21
  Div(
22
  Label("Model"),
23
+ Select(models("audi"), id="models"),
24
  Img(width="20", src="/img/bars.svg", cls="htmx-indicator"),
25
  ),
26
  ),
 
27
  )
28
 
29
 
30
  @app.get
31
+ def models(make: str, sleep: int = 0):
32
  time.sleep(sleep)
33
  cars = {
34
  "audi": ["A1", "A4", "A6"],
 
49
  ::page::
50
 
51
  When a request is made to the /models end point, we return the models for that make:
52
+ ::models::
53
 
54
  And they become available in the model select.
55
  """
src/tutorial/_13_file_upload.py CHANGED
@@ -21,29 +21,23 @@ def method1():
21
  htmx.find('#progress').setAttribute('value', evt.detail.loaded/evt.detail.total * 100)
22
  });
23
  """
24
- form = Form(
25
  Input(type="file", name="file"),
26
  Button("Upload"),
27
  Progress(id="progress", value="0", max="100"),
28
  Div(id="output"),
29
- hx_target="#output",
30
- hx_post=upload.rt(),
31
- id="form",
32
  )
33
  return form, Script(js)
34
 
35
 
36
  @app.get
37
  def method2():
38
- return Form(
 
39
  Input(type="file", name="file"),
40
  Button("Upload"),
41
  Progress(id="progress2", value="0", max="100"),
42
  Div(id="output2"),
43
- hx_target="#output2",
44
- hx_encoding="multipart/form-data",
45
- hx_post=upload.rt(),
46
- _="on htmx:xhr:progress(loaded, total) set #progress2.value to (loaded/total)*100",
47
  )
48
 
49
 
 
21
  htmx.find('#progress').setAttribute('value', evt.detail.loaded/evt.detail.total * 100)
22
  });
23
  """
24
+ form = Form(hx_target="#output", hx_post=upload.rt(), id="form")(
25
  Input(type="file", name="file"),
26
  Button("Upload"),
27
  Progress(id="progress", value="0", max="100"),
28
  Div(id="output"),
 
 
 
29
  )
30
  return form, Script(js)
31
 
32
 
33
  @app.get
34
  def method2():
35
+ script = "on htmx:xhr:progress(loaded, total) set #progress2.value to (loaded/total)*100"
36
+ return Form(_=script, hx_target="#output2", hx_post=upload.rt())(
37
  Input(type="file", name="file"),
38
  Button("Upload"),
39
  Progress(id="progress2", value="0", max="100"),
40
  Div(id="output2"),
 
 
 
 
41
  )
42
 
43
 
src/tutorial/__init__.py CHANGED
@@ -50,13 +50,12 @@ The code can be found on [GitHub](https://github.com/phihung/fasthtml_examples).
50
  @app.get("/")
51
  def homepage():
52
  ls = [get_example(name) for name in examples]
53
- return Main(
54
  Div(INTRO, cls="marked"),
55
  Table(
56
  Thead(Tr(Th("Pattern"), Th("Description"))),
57
  Tbody(tuple(Tr(Td(A(ex.title, href="/" + ex.slug)), Td(ex.desc)) for ex in ls)),
58
  ),
59
- cls="container",
60
  )
61
 
62
 
@@ -118,7 +117,7 @@ class Example:
118
  doc = re.sub("::([a-zA-Z_0-9\s]+)::", lambda x: code_block(module, x.group(1)), self.doc)
119
  content = Div(doc, cls="marked")
120
 
121
- return Main(
122
  Hgroup(H1(self.title), P(self.desc)),
123
  Div(
124
  A("Back", href="/"),
@@ -129,12 +128,10 @@ class Example:
129
  "|",
130
  A("Htmx Docs", href=self.htmx_url),
131
  ),
132
- Div(
133
  Div(content, style="height:80vh;overflow:scroll"),
134
  Div(P(A("Direct url", href=self.start_url)), Iframe(src=self.start_url, height="500px", width="100%")),
135
- cls="grid",
136
  ),
137
- cls="container",
138
  )
139
 
140
  def _fix_url(self):
 
50
  @app.get("/")
51
  def homepage():
52
  ls = [get_example(name) for name in examples]
53
+ return Main(cls="container")(
54
  Div(INTRO, cls="marked"),
55
  Table(
56
  Thead(Tr(Th("Pattern"), Th("Description"))),
57
  Tbody(tuple(Tr(Td(A(ex.title, href="/" + ex.slug)), Td(ex.desc)) for ex in ls)),
58
  ),
 
59
  )
60
 
61
 
 
117
  doc = re.sub("::([a-zA-Z_0-9\s]+)::", lambda x: code_block(module, x.group(1)), self.doc)
118
  content = Div(doc, cls="marked")
119
 
120
+ return Main(cls="container")(
121
  Hgroup(H1(self.title), P(self.desc)),
122
  Div(
123
  A("Back", href="/"),
 
128
  "|",
129
  A("Htmx Docs", href=self.htmx_url),
130
  ),
131
+ Div(cls="grid")(
132
  Div(content, style="height:80vh;overflow:scroll"),
133
  Div(P(A("Direct url", href=self.start_url)), Iframe(src=self.start_url, height="500px", width="100%")),
 
134
  ),
 
135
  )
136
 
137
  def _fix_url(self):