Spaces:
Running
Running
refactor: Use attrs-first style
Browse files- src/tutorial/_01_click_to_edit.py +4 -11
- src/tutorial/_03_click_to_load.py +8 -14
- src/tutorial/_04_delete_row.py +2 -6
- src/tutorial/_05_edit_row.py +4 -10
- src/tutorial/_06_lazy_loading.py +1 -4
- src/tutorial/_07_inline_validation.py +4 -9
- src/tutorial/_08_infinite_scroll.py +2 -6
- src/tutorial/_09_active_search.py +4 -9
- src/tutorial/_10_progress_bar.py +7 -19
- src/tutorial/_11_cascading_select.py +5 -8
- src/tutorial/_13_file_upload.py +3 -9
- src/tutorial/__init__.py +3 -6
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/
|
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
|
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 |
-
|
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 |
-
|
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 |
-
|
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 |
-
|
|
|
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 |
-
|
16 |
-
|
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="
|
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 |
-
|
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
|
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
|
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(
|
26 |
Img(width="20", src="/img/bars.svg", cls="htmx-indicator"),
|
27 |
),
|
28 |
),
|
29 |
-
cls="container",
|
30 |
)
|
31 |
|
32 |
|
33 |
@app.get
|
34 |
-
def
|
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 |
-
::
|
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 |
-
|
|
|
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):
|