From c1dda7127f1bab5544b82f9c42b18be8b8d005a2 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sun, 12 Jan 2025 17:39:29 +0000 Subject: [PATCH] deploy: aa02c2e68c42969cc27fbdf649a2e7f619342c46 --- .nojekyll | 0 CNAME | 1 + api/README.txt | 2 + api/cli.html | 832 +++ api/cli.html.md | 53 + api/components.html | 1260 +++++ api/components.html.md | 491 ++ api/core.html | 2201 ++++++++ api/core.html.md | 1489 +++++ api/js.html | 1144 ++++ api/js.html.md | 365 ++ api/jupyter.html | 1075 ++++ api/jupyter.html.md | 299 + api/oauth.html | 1024 ++++ api/oauth.html.md | 243 + api/pico.html | 1066 ++++ api/pico.html.md | 243 + api/svg.html | 1380 +++++ api/svg.html.md | 639 +++ api/xtend.html | 1228 +++++ api/xtend.html.md | 458 ++ apilist.txt | 467 ++ explains/explaining_xt_components.html | 992 ++++ explains/explaining_xt_components.html.md | 215 + explains/faq.html | 897 +++ explains/faq.html.md | 133 + explains/imgs/gh-oauth.png | Bin 0 -> 122885 bytes explains/minidataapi.html | 1367 +++++ explains/minidataapi.html.md | 668 +++ explains/oauth.html | 1086 ++++ explains/oauth.html.md | 480 ++ explains/routes.html | 992 ++++ explains/routes.html.md | 217 + explains/websockets.html | 948 ++++ explains/websockets.html.md | 197 + favicon.ico | Bin 0 -> 15086 bytes index.html | 938 ++++ index.html.md | 203 + listings.json | 11 + llms-ctx-full.txt | 4857 +++++++++++++++++ llms-ctx.txt | 3261 +++++++++++ llms.txt | 33 + logo.svg | 32 + ref/defining_xt_component.html | 974 ++++ ref/defining_xt_component.md | 202 + ref/handlers.html | 1653 ++++++ ref/handlers.html.md | 1207 ++++ ref/live_reload.html | 855 +++ ref/live_reload.html.md | 61 + robots.txt | 1 + search.json | 1528 ++++++ ...p-f060f2c01c87989ca9870cd8c49a312f.min.css | 12 + site_libs/bootstrap/bootstrap-icons.css | 2078 +++++++ site_libs/bootstrap/bootstrap-icons.woff | Bin 0 -> 176200 bytes site_libs/bootstrap/bootstrap.min.js | 7 + site_libs/clipboard/clipboard.min.js | 7 + site_libs/quarto-html/anchor.min.js | 9 + site_libs/quarto-html/popper.min.js | 6 + ...hting-549806ee2085284f45b00abea8c6df48.css | 205 + site_libs/quarto-html/quarto.js | 911 ++++ site_libs/quarto-html/tippy.css | 1 + site_libs/quarto-html/tippy.umd.min.js | 2 + site_libs/quarto-listing/list.min.js | 2 + site_libs/quarto-listing/quarto-listing.js | 254 + site_libs/quarto-nav/headroom.min.js | 7 + site_libs/quarto-nav/quarto-nav.js | 325 ++ site_libs/quarto-search/autocomplete.umd.js | 3 + site_libs/quarto-search/fuse.min.js | 9 + site_libs/quarto-search/quarto-search.js | 1290 +++++ sitemap.xml | 103 + styles.css | 47 + tutorials/by_example.html | 1902 +++++++ tutorials/by_example.html.md | 1563 ++++++ .../figure-commonmark/cell-101-1-image.png | Bin 0 -> 93632 bytes .../figure-commonmark/cell-53-1-image.png | Bin 0 -> 45599 bytes .../figure-commonmark/cell-58-1-image.png | Bin 0 -> 1411382 bytes .../figure-html/cell-101-1-image.png | Bin 0 -> 93632 bytes .../figure-html/cell-53-1-image.png | Bin 0 -> 45599 bytes .../figure-html/cell-58-1-image.png | Bin 0 -> 1411382 bytes tutorials/e2e.html | 1242 +++++ tutorials/e2e.html.md | 499 ++ tutorials/imgs/quickdraw.png | Bin 0 -> 67243 bytes tutorials/index.html | 894 +++ tutorials/index.md | 5 + tutorials/jupyter_and_fasthtml.html | 902 +++ tutorials/jupyter_and_fasthtml.html.md | 107 + .../quickstart-fasthtml.png | Bin 0 -> 13843 bytes tutorials/quickstart_for_web_devs.html | 1988 +++++++ tutorials/quickstart_for_web_devs.html.md | 1301 +++++ unpublished/tutorial_for_web_devs.html | 1077 ++++ unpublished/tutorial_for_web_devs.html.md | 319 ++ .../web-dev-tut/random-list-letters.png | Bin 0 -> 18818 bytes 92 files changed, 57045 insertions(+) create mode 100644 .nojekyll create mode 100644 CNAME create mode 100644 api/README.txt create mode 100644 api/cli.html create mode 100644 api/cli.html.md create mode 100644 api/components.html create mode 100644 api/components.html.md create mode 100644 api/core.html create mode 100644 api/core.html.md create mode 100644 api/js.html create mode 100644 api/js.html.md create mode 100644 api/jupyter.html create mode 100644 api/jupyter.html.md create mode 100644 api/oauth.html create mode 100644 api/oauth.html.md create mode 100644 api/pico.html create mode 100644 api/pico.html.md create mode 100644 api/svg.html create mode 100644 api/svg.html.md create mode 100644 api/xtend.html create mode 100644 api/xtend.html.md create mode 100644 apilist.txt create mode 100644 explains/explaining_xt_components.html create mode 100644 explains/explaining_xt_components.html.md create mode 100644 explains/faq.html create mode 100644 explains/faq.html.md create mode 100644 explains/imgs/gh-oauth.png create mode 100644 explains/minidataapi.html create mode 100644 explains/minidataapi.html.md create mode 100644 explains/oauth.html create mode 100644 explains/oauth.html.md create mode 100644 explains/routes.html create mode 100644 explains/routes.html.md create mode 100644 explains/websockets.html create mode 100644 explains/websockets.html.md create mode 100644 favicon.ico create mode 100644 index.html create mode 100644 index.html.md create mode 100644 listings.json create mode 100644 llms-ctx-full.txt create mode 100644 llms-ctx.txt create mode 100644 llms.txt create mode 100644 logo.svg create mode 100644 ref/defining_xt_component.html create mode 100644 ref/defining_xt_component.md create mode 100644 ref/handlers.html create mode 100644 ref/handlers.html.md create mode 100644 ref/live_reload.html create mode 100644 ref/live_reload.html.md create mode 100644 robots.txt create mode 100644 search.json create mode 100644 site_libs/bootstrap/bootstrap-f060f2c01c87989ca9870cd8c49a312f.min.css create mode 100644 site_libs/bootstrap/bootstrap-icons.css create mode 100644 site_libs/bootstrap/bootstrap-icons.woff create mode 100644 site_libs/bootstrap/bootstrap.min.js create mode 100644 site_libs/clipboard/clipboard.min.js create mode 100644 site_libs/quarto-html/anchor.min.js create mode 100644 site_libs/quarto-html/popper.min.js create mode 100644 site_libs/quarto-html/quarto-syntax-highlighting-549806ee2085284f45b00abea8c6df48.css create mode 100644 site_libs/quarto-html/quarto.js create mode 100644 site_libs/quarto-html/tippy.css create mode 100644 site_libs/quarto-html/tippy.umd.min.js create mode 100644 site_libs/quarto-listing/list.min.js create mode 100644 site_libs/quarto-listing/quarto-listing.js create mode 100644 site_libs/quarto-nav/headroom.min.js create mode 100644 site_libs/quarto-nav/quarto-nav.js create mode 100644 site_libs/quarto-search/autocomplete.umd.js create mode 100644 site_libs/quarto-search/fuse.min.js create mode 100644 site_libs/quarto-search/quarto-search.js create mode 100644 sitemap.xml create mode 100644 styles.css create mode 100644 tutorials/by_example.html create mode 100644 tutorials/by_example.html.md create mode 100644 tutorials/by_example_files/figure-commonmark/cell-101-1-image.png create mode 100644 tutorials/by_example_files/figure-commonmark/cell-53-1-image.png create mode 100644 tutorials/by_example_files/figure-commonmark/cell-58-1-image.png create mode 100644 tutorials/by_example_files/figure-html/cell-101-1-image.png create mode 100644 tutorials/by_example_files/figure-html/cell-53-1-image.png create mode 100644 tutorials/by_example_files/figure-html/cell-58-1-image.png create mode 100644 tutorials/e2e.html create mode 100644 tutorials/e2e.html.md create mode 100644 tutorials/imgs/quickdraw.png create mode 100644 tutorials/index.html create mode 100644 tutorials/index.md create mode 100644 tutorials/jupyter_and_fasthtml.html create mode 100644 tutorials/jupyter_and_fasthtml.html.md create mode 100644 tutorials/quickstart-web-dev/quickstart-fasthtml.png create mode 100644 tutorials/quickstart_for_web_devs.html create mode 100644 tutorials/quickstart_for_web_devs.html.md create mode 100644 unpublished/tutorial_for_web_devs.html create mode 100644 unpublished/tutorial_for_web_devs.html.md create mode 100644 unpublished/web-dev-tut/random-list-letters.png diff --git a/.nojekyll b/.nojekyll new file mode 100644 index 00000000..e69de29b diff --git a/CNAME b/CNAME new file mode 100644 index 00000000..fe2e8862 --- /dev/null +++ b/CNAME @@ -0,0 +1 @@ +docs.fastht.ml diff --git a/api/README.txt b/api/README.txt new file mode 100644 index 00000000..7a38fd9b --- /dev/null +++ b/api/README.txt @@ -0,0 +1,2 @@ +These are the source notebooks for FastHTML. + diff --git a/api/cli.html b/api/cli.html new file mode 100644 index 00000000..9ad30a98 --- /dev/null +++ b/api/cli.html @@ -0,0 +1,832 @@ + + + + + + + + + +Command Line Tools – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Command Line Tools

+
+ + + +
+ + + + +
+ + + +
+ + + +
+

source

+ +
+

railway_deploy

+
+
 railway_deploy (name:str, mount:<function bool_arg>=True)
+
+

Deploy a FastHTML app to Railway

+ + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
namestrThe project name to deploy
mountbool_argTrueCreate a mounted volume at /app/data?
+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/cli.html.md b/api/cli.html.md new file mode 100644 index 00000000..841a46bd --- /dev/null +++ b/api/cli.html.md @@ -0,0 +1,53 @@ +# Command Line Tools + + + + +------------------------------------------------------------------------ + +source + +### railway_link + +> railway_link () + +*Link the current directory to the current project’s Railway service* + +------------------------------------------------------------------------ + +source + +### railway_deploy + +> railway_deploy (name:str, mount:=True) + +*Deploy a FastHTML app to Railway* + + + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
namestrThe project name to deploy
mountbool_argTrueCreate a mounted volume at /app/data?
diff --git a/api/components.html b/api/components.html new file mode 100644 index 00000000..8629e9ce --- /dev/null +++ b/api/components.html @@ -0,0 +1,1260 @@ + + + + + + + + + + +Components – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Components

+
+ +
+
+ ft_html and ft_hx functions to add some conveniences to ft, along with a full set of basic HTML components, and functions to work with forms and FT conversion +
+
+ + +
+ + + + +
+ + + +
+ + + +
+
from lxml import html as lx
+from pprint import pprint
+
+
+

Str, show and repr

+
+

source

+
+
+

show

+
+
 show (ft, *rest)
+
+

Renders FT Components into HTML within a Jupyter notebook.

+
+
sentence = P(Strong("FastHTML is ", I("Fast")), id='sentence_id')
+
+

When placed within the show() function, this will render the HTML in Jupyter notebooks.

+
+
show(sentence)
+
+

+FastHTML is Fast

+
+
+

In notebooks, FT components are rendered as their syntax highlighted XML/HTML:

+
+
sentence
+
+
<p id="sentence_id">
+<strong>FastHTML is <i>Fast</i></strong></p>
+
+
+

Elsewhere, they are represented as their underlying data structure:

+
+
print(repr(sentence))
+
+
p((strong(('FastHTML is ', i(('Fast',),{})),{}),),{'id': 'sentence_id'})
+
+
+
+

source

+
+
+

FT.__str__

+
+
 FT.__str__ ()
+
+

Return str(self).

+

If they have an id, then that id is used as the component’s str representation:

+
+
f'hx_target=#{sentence}'
+
+
'hx_target=#sentence_id'
+
+
+
+

source

+
+
+

FT.__radd__

+
+
 FT.__radd__ (b)
+
+
+
'hx_target=#' + sentence
+
+
'hx_target=#sentence_id'
+
+
+
+

source

+
+
+

FT.__add__

+
+
 FT.__add__ (b)
+
+
+
sentence + '...'
+
+
'sentence_id...'
+
+
+
+
+

fh_html and fh_hx

+
+

source

+
+
+

attrmap_x

+
+
 attrmap_x (o)
+
+
+

source

+
+
+

ft_html

+
+
 ft_html (tag:str, *c, id=None, cls=None, title=None, style=None,
+          attrmap=None, valmap=None, ft_cls=None, **kwargs)
+
+
+
ft_html('a', **{'@click.away':1})
+
+
<a @click.away="1"></a>
+
+
+
+
ft_html('a', {'@click.away':1})
+
+
<a @click.away="1"></a>
+
+
+
+
c = Div(id='someid')
+
+
+
ft_html('a', id=c)
+
+
<a id="someid" name="someid"></a>
+
+
+
+

source

+
+
+

ft_hx

+
+
 ft_hx (tag:str, *c, target_id=None, hx_vals=None, hx_target=None,
+        id=None, cls=None, title=None, style=None, accesskey=None,
+        contenteditable=None, dir=None, draggable=None, enterkeyhint=None,
+        hidden=None, inert=None, inputmode=None, lang=None, popover=None,
+        spellcheck=None, tabindex=None, translate=None, hx_get=None,
+        hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,
+        hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None,
+        hx_select=None, hx_select_oob=None, hx_indicator=None,
+        hx_push_url=None, hx_confirm=None, hx_disable=None,
+        hx_replace_url=None, hx_disabled_elt=None, hx_ext=None,
+        hx_headers=None, hx_history=None, hx_history_elt=None,
+        hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None,
+        hx_request=None, hx_sync=None, hx_validate=None, **kwargs)
+
+
+
ft_hx('a', hx_vals={'a':1})
+
+
<a hx-vals='{"a": 1}'></a>
+
+
+
+
ft_hx('a', hx_target=c)
+
+
<a hx-target="#someid"></a>
+
+
+

For tags that have a name attribute, it will be set to the value of id if not provided explicitly:

+
+
Form(Button(target_id='foo', id='btn'),
+     hx_post='/', target_id='tgt', id='frm')
+
+
<form hx-post="/" hx-target="#tgt" id="frm" name="frm"><button hx-target="#foo" id="btn" name="btn"></button></form>
+
+
+
+

source

+
+
+

File

+
+
 File (fname)
+
+

Use the unescaped text in file fname directly

+
+
a = Input(name='nm')
+a
+
+
<input name="nm">
+
+
+
+
a(hx_swap_oob='true')
+
+
<input name="nm" hx-swap-oob="true">
+
+
+
+
a
+
+
<input name="nm" hx-swap-oob="true">
+
+
+
+
+

fill_form and find_inputs

+
+

source

+
+
+

fill_form

+
+
 fill_form (form:fastcore.xml.FT, obj)
+
+

Fills named items in form using attributes in obj

+
+
@dataclass
+class TodoItem:
+    title:str; id:int; done:bool; details:str; opt:str='a'
+
+todo = TodoItem(id=2, title="Profit", done=True, details="Details", opt='b')
+check = Label(Input(type="checkbox", cls="checkboxer", name="done", data_foo="bar"), "Done", cls='px-2')
+form = Form(Fieldset(Input(cls="char", id="title", value="a"), check, Input(type="hidden", id="id"),
+                     Select(Option(value='a'), Option(value='b'), name='opt'),
+                     Textarea(id='details'), Button("Save"),
+                     name="stuff"))
+form = fill_form(form, todo)
+assert '<textarea id="details" name="details">Details</textarea>' in to_xml(form)
+form
+
+
<form><fieldset name="stuff">    <input value="Profit" id="title" class="char" name="title">
+<label class="px-2">      <input type="checkbox" name="done" data-foo="bar" class="checkboxer" checked="1">
+Done</label>    <input type="hidden" id="id" name="id" value="2">
+<select name="opt"><option value="a"></option><option value="b" selected="1"></option></select><textarea id="details" name="details">Details</textarea><button>Save</button></fieldset></form>
+
+
+
+
@dataclass
+class MultiSelect:
+    items: list[str]
+
+multiselect = MultiSelect(items=['a', 'c'])
+multiform = Form(Select(Option('a', value='a'), Option('b', value='b'), Option('c', value='c'), multiple='1', name='items'))
+multiform = fill_form(multiform, multiselect)
+assert '<option value="a" selected="1">a</option>' in to_xml(multiform)
+assert '<option value="b">b</option>' in to_xml(multiform)
+assert '<option value="c" selected="1">c</option>' in to_xml(multiform)
+multiform
+
+
<form><select multiple="1" name="items"><option value="a" selected="1">a</option><option value="b">b</option><option value="c" selected="1">c</option></select></form>
+
+
+
+
@dataclass
+class MultiCheck:
+    items: list[str]
+
+multicheck = MultiCheck(items=['a', 'c'])
+multiform = Form(Fieldset(Label(Input(type='checkbox', name='items', value='a'), 'a'),
+                          Label(Input(type='checkbox', name='items', value='b'), 'b'),
+                          Label(Input(type='checkbox', name='items', value='c'), 'c')))
+multiform = fill_form(multiform, multicheck)
+assert '<input type="checkbox" name="items" value="a" checked="1">' in to_xml(multiform)
+assert '<input type="checkbox" name="items" value="b">' in to_xml(multiform)
+assert '<input type="checkbox" name="items" value="c" checked="1">' in to_xml(multiform)
+multiform
+
+
<form><fieldset><label>      <input type="checkbox" name="items" value="a" checked="1">
+a</label><label>      <input type="checkbox" name="items" value="b">
+b</label><label>      <input type="checkbox" name="items" value="c" checked="1">
+c</label></fieldset></form>
+
+
+
+

source

+
+
+

fill_dataclass

+
+
 fill_dataclass (src, dest)
+
+

Modifies dataclass in-place and returns it

+
+
nt = TodoItem('', 0, False, '')
+fill_dataclass(todo, nt)
+nt
+
+
TodoItem(title='Profit', id=2, done=True, details='Details', opt='b')
+
+
+
+

source

+
+
+

find_inputs

+
+
 find_inputs (e, tags='input', **kw)
+
+

Recursively find all elements in e with tags and attrs matching kw

+
+
inps = find_inputs(form, id='title')
+test_eq(len(inps), 1)
+inps
+
+
[input((),{'value': 'Profit', 'id': 'title', 'class': 'char', 'name': 'title'})]
+
+
+

You can also use lxml for more sophisticated searching:

+
+
elem = lx.fromstring(to_xml(form))
+test_eq(elem.xpath("//input[@id='title']/@value"), ['Profit'])
+
+
+

source

+
+
+

getattr

+
+
 __getattr__ (tag)
+
+
+
+

html2ft

+
+

source

+
+
+

html2ft

+
+
 html2ft (html, attr1st=False)
+
+

Convert HTML to an ft expression

+
+
h = to_xml(form)
+hl_md(html2ft(h), 'python')
+
+
Form(
+    Fieldset(
+        Input(value='Profit', id='title', name='title', cls='char'),
+        Label(
+            Input(type='checkbox', name='done', data_foo='bar', checked='1', cls='checkboxer'),
+            'Done',
+            cls='px-2'
+        ),
+        Input(type='hidden', id='id', name='id', value='2'),
+        Select(
+            Option(value='a'),
+            Option(value='b', selected='1'),
+            name='opt'
+        ),
+        Textarea('Details', id='details', name='details'),
+        Button('Save'),
+        name='stuff'
+    )
+)
+
+
+
+
hl_md(html2ft(h, attr1st=True), 'python')
+
+
Form(
+    Fieldset(name='stuff')(
+        Input(value='Profit', id='title', name='title', cls='char')(),
+        Label(cls='px-2')(
+            Input(type='checkbox', name='done', data_foo='bar', checked='1', cls='checkboxer')(),
+            'Done'
+        ),
+        Input(type='hidden', id='id', name='id', value='2')(),
+        Select(name='opt')(
+            Option(value='a')(),
+            Option(value='b', selected='1')()
+        ),
+        Textarea(id='details', name='details')('Details'),
+        Button()('Save')
+    )
+)
+
+
+
+

source

+
+
+

sse_message

+
+
 sse_message (elm, event='message')
+
+

Convert element elm into a format suitable for SSE streaming

+
+
print(sse_message(Div(P('hi'), P('there'))))
+
+
event: message
+data: <div>
+data:   <p>hi</p>
+data:   <p>there</p>
+data: </div>
+
+
+
+
+
+
+

Tests

+
+
test_html2ft('<input value="Profit" name="title" id="title" class="char">', attr1st=True)
+test_html2ft('<input value="Profit" name="title" id="title" class="char">')
+test_html2ft('<div id="foo"></div>')
+test_html2ft('<div id="foo">hi</div>')
+test_html2ft('<div x-show="open" x-transition:enter="transition duration-300" x-transition:enter-start="opacity-0 scale-90">Hello 👋</div>')
+test_html2ft('<div x-transition:enter.scale.80 x-transition:leave.scale.90>hello</div>')
+
+
+
assert html2ft('<div id="foo">hi</div>', attr1st=True) == "Div(id='foo')('hi')"
+assert html2ft("""
+  <div x-show="open" x-transition:enter="transition duration-300" x-transition:enter-start="opacity-0 scale-90">Hello 👋</div>
+""") == "Div('Hello 👋', x_show='open', **{'x-transition:enter': 'transition duration-300', 'x-transition:enter-start': 'opacity-0 scale-90'})"
+assert html2ft('<div x-transition:enter.scale.80 x-transition:leave.scale.90>hello</div>') == "Div('hello', **{'x-transition:enter.scale.80': True, 'x-transition:leave.scale.90': True})"
+assert html2ft("<img alt=' ' />") == "Img(alt=' ')"
+
+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/components.html.md b/api/components.html.md new file mode 100644 index 00000000..64df08dd --- /dev/null +++ b/api/components.html.md @@ -0,0 +1,491 @@ +# Components + + + + +``` python +from lxml import html as lx +from pprint import pprint +``` + +### Str, show and repr + +------------------------------------------------------------------------ + +source + +### show + +> show (ft, *rest) + +*Renders FT Components into HTML within a Jupyter notebook.* + +``` python +sentence = P(Strong("FastHTML is ", I("Fast")), id='sentence_id') +``` + +When placed within the +[`show()`](https://AnswerDotAI.github.io/fasthtml/api/jupyter.html#show) +function, this will render the HTML in Jupyter notebooks. + +``` python +show(sentence) +``` + +

+FastHTML is Fast

+ +In notebooks, FT components are rendered as their syntax highlighted +XML/HTML: + +``` python +sentence +``` + +``` html +

+FastHTML is Fast

+``` + +Elsewhere, they are represented as their underlying data structure: + +``` python +print(repr(sentence)) +``` + + p((strong(('FastHTML is ', i(('Fast',),{})),{}),),{'id': 'sentence_id'}) + +------------------------------------------------------------------------ + +source + +### FT.\_\_str\_\_ + +> FT.__str__ () + +*Return str(self).* + +If they have an id, then that id is used as the component’s str +representation: + +``` python +f'hx_target=#{sentence}' +``` + + 'hx_target=#sentence_id' + +------------------------------------------------------------------------ + +source + +### FT.\_\_radd\_\_ + +> FT.__radd__ (b) + +``` python +'hx_target=#' + sentence +``` + + 'hx_target=#sentence_id' + +------------------------------------------------------------------------ + +source + +### FT.\_\_add\_\_ + +> FT.__add__ (b) + +``` python +sentence + '...' +``` + + 'sentence_id...' + +### fh_html and fh_hx + +------------------------------------------------------------------------ + +source + +### attrmap_x + +> attrmap_x (o) + +------------------------------------------------------------------------ + +source + +### ft_html + +> ft_html (tag:str, *c, id=None, cls=None, title=None, style=None, +> attrmap=None, valmap=None, ft_cls=None, **kwargs) + +``` python +ft_html('a', **{'@click.away':1}) +``` + +``` html + +``` + +``` python +ft_html('a', {'@click.away':1}) +``` + +``` html + +``` + +``` python +c = Div(id='someid') +``` + +``` python +ft_html('a', id=c) +``` + +``` html + +``` + +------------------------------------------------------------------------ + +source + +### ft_hx + +> ft_hx (tag:str, *c, target_id=None, hx_vals=None, hx_target=None, +> id=None, cls=None, title=None, style=None, accesskey=None, +> contenteditable=None, dir=None, draggable=None, enterkeyhint=None, +> hidden=None, inert=None, inputmode=None, lang=None, popover=None, +> spellcheck=None, tabindex=None, translate=None, hx_get=None, +> hx_post=None, hx_put=None, hx_delete=None, hx_patch=None, +> hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None, +> hx_select=None, hx_select_oob=None, hx_indicator=None, +> hx_push_url=None, hx_confirm=None, hx_disable=None, +> hx_replace_url=None, hx_disabled_elt=None, hx_ext=None, +> hx_headers=None, hx_history=None, hx_history_elt=None, +> hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None, **kwargs) + +``` python +ft_hx('a', hx_vals={'a':1}) +``` + +``` html + +``` + +``` python +ft_hx('a', hx_target=c) +``` + +``` html + +``` + +For tags that have a `name` attribute, it will be set to the value of +`id` if not provided explicitly: + +``` python +Form(Button(target_id='foo', id='btn'), + hx_post='/', target_id='tgt', id='frm') +``` + +``` html +
+``` + +------------------------------------------------------------------------ + +source + +### File + +> File (fname) + +*Use the unescaped text in file `fname` directly* + +``` python +a = Input(name='nm') +a +``` + +``` html + +``` + +``` python +a(hx_swap_oob='true') +``` + +``` html + +``` + +``` python +a +``` + +``` html + +``` + +### fill_form and find_inputs + +------------------------------------------------------------------------ + +source + +### fill_form + +> fill_form (form:fastcore.xml.FT, obj) + +*Fills named items in `form` using attributes in `obj`* + +``` python +@dataclass +class TodoItem: + title:str; id:int; done:bool; details:str; opt:str='a' + +todo = TodoItem(id=2, title="Profit", done=True, details="Details", opt='b') +check = Label(Input(type="checkbox", cls="checkboxer", name="done", data_foo="bar"), "Done", cls='px-2') +form = Form(Fieldset(Input(cls="char", id="title", value="a"), check, Input(type="hidden", id="id"), + Select(Option(value='a'), Option(value='b'), name='opt'), + Textarea(id='details'), Button("Save"), + name="stuff")) +form = fill_form(form, todo) +assert '' in to_xml(form) +form +``` + +``` html +
+ +
+``` + +``` python +@dataclass +class MultiSelect: + items: list[str] + +multiselect = MultiSelect(items=['a', 'c']) +multiform = Form(Select(Option('a', value='a'), Option('b', value='b'), Option('c', value='c'), multiple='1', name='items')) +multiform = fill_form(multiform, multiselect) +assert '' in to_xml(multiform) +assert '' in to_xml(multiform) +assert '' in to_xml(multiform) +multiform +``` + +``` html +
+``` + +``` python +@dataclass +class MultiCheck: + items: list[str] + +multicheck = MultiCheck(items=['a', 'c']) +multiform = Form(Fieldset(Label(Input(type='checkbox', name='items', value='a'), 'a'), + Label(Input(type='checkbox', name='items', value='b'), 'b'), + Label(Input(type='checkbox', name='items', value='c'), 'c'))) +multiform = fill_form(multiform, multicheck) +assert '' in to_xml(multiform) +assert '' in to_xml(multiform) +assert '' in to_xml(multiform) +multiform +``` + +``` html +
+``` + +------------------------------------------------------------------------ + +source + +### fill_dataclass + +> fill_dataclass (src, dest) + +*Modifies dataclass in-place and returns it* + +``` python +nt = TodoItem('', 0, False, '') +fill_dataclass(todo, nt) +nt +``` + + TodoItem(title='Profit', id=2, done=True, details='Details', opt='b') + +------------------------------------------------------------------------ + +source + +### find_inputs + +> find_inputs (e, tags='input', **kw) + +*Recursively find all elements in `e` with `tags` and attrs matching +`kw`* + +``` python +inps = find_inputs(form, id='title') +test_eq(len(inps), 1) +inps +``` + + [input((),{'value': 'Profit', 'id': 'title', 'class': 'char', 'name': 'title'})] + +You can also use lxml for more sophisticated searching: + +``` python +elem = lx.fromstring(to_xml(form)) +test_eq(elem.xpath("//input[@id='title']/@value"), ['Profit']) +``` + +------------------------------------------------------------------------ + +source + +### **getattr** + +> __getattr__ (tag) + +### html2ft + +------------------------------------------------------------------------ + +source + +### html2ft + +> html2ft (html, attr1st=False) + +*Convert HTML to an `ft` expression* + +``` python +h = to_xml(form) +hl_md(html2ft(h), 'python') +``` + +``` python +Form( + Fieldset( + Input(value='Profit', id='title', name='title', cls='char'), + Label( + Input(type='checkbox', name='done', data_foo='bar', checked='1', cls='checkboxer'), + 'Done', + cls='px-2' + ), + Input(type='hidden', id='id', name='id', value='2'), + Select( + Option(value='a'), + Option(value='b', selected='1'), + name='opt' + ), + Textarea('Details', id='details', name='details'), + Button('Save'), + name='stuff' + ) +) +``` + +``` python +hl_md(html2ft(h, attr1st=True), 'python') +``` + +``` python +Form( + Fieldset(name='stuff')( + Input(value='Profit', id='title', name='title', cls='char')(), + Label(cls='px-2')( + Input(type='checkbox', name='done', data_foo='bar', checked='1', cls='checkboxer')(), + 'Done' + ), + Input(type='hidden', id='id', name='id', value='2')(), + Select(name='opt')( + Option(value='a')(), + Option(value='b', selected='1')() + ), + Textarea(id='details', name='details')('Details'), + Button()('Save') + ) +) +``` + +------------------------------------------------------------------------ + +source + +### sse_message + +> sse_message (elm, event='message') + +*Convert element `elm` into a format suitable for SSE streaming* + +``` python +print(sse_message(Div(P('hi'), P('there')))) +``` + + event: message + data:
+ data:

hi

+ data:

there

+ data:
+ +## Tests + +``` python +test_html2ft('', attr1st=True) +test_html2ft('') +test_html2ft('
') +test_html2ft('
hi
') +test_html2ft('
Hello 👋
') +test_html2ft('
hello
') +``` + +``` python +assert html2ft('
hi
', attr1st=True) == "Div(id='foo')('hi')" +assert html2ft(""" +
Hello 👋
+""") == "Div('Hello 👋', x_show='open', **{'x-transition:enter': 'transition duration-300', 'x-transition:enter-start': 'opacity-0 scale-90'})" +assert html2ft('
hello
') == "Div('hello', **{'x-transition:enter.scale.80': True, 'x-transition:leave.scale.90': True})" +assert html2ft(" ") == "Img(alt=' ')" +``` diff --git a/api/core.html b/api/core.html new file mode 100644 index 00000000..93de0ff4 --- /dev/null +++ b/api/core.html @@ -0,0 +1,2201 @@ + + + + + + + + + + +Core – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Core

+
+ +
+
+ The FastHTML subclass of Starlette, along with the RouterX and RouteX classes it automatically uses. +
+
+ + +
+ + + + +
+ + + +
+ + + +

This is the source code to fasthtml. You won’t need to read this unless you want to understand how things are built behind the scenes, or need full details of a particular API. The notebook is converted to the Python module fasthtml/core.py using nbdev.

+
+

Imports and utils

+
+
import time
+
+from IPython import display
+from enum import Enum
+from pprint import pprint
+
+from fastcore.test import *
+from starlette.testclient import TestClient
+from starlette.requests import Headers
+from starlette.datastructures import UploadFile
+
+

We write source code first, and then tests come after. The tests serve as both a means to confirm that the code works and also serves as working examples. The first exported function, parsed_date, is an example of this pattern.

+
+

source

+
+

parsed_date

+
+
 parsed_date (s:str)
+
+

Convert s to a datetime

+
+
parsed_date('2pm')
+
+
datetime.datetime(2025, 1, 12, 14, 0)
+
+
+
+
isinstance(date.fromtimestamp(0), date)
+
+
True
+
+
+
+

source

+
+
+

snake2hyphens

+
+
 snake2hyphens (s:str)
+
+

Convert s from snake case to hyphenated and capitalised

+
+
snake2hyphens("snake_case")
+
+
'Snake-Case'
+
+
+
+

source

+
+
+

HtmxHeaders

+
+
 HtmxHeaders (boosted:str|None=None, current_url:str|None=None,
+              history_restore_request:str|None=None, prompt:str|None=None,
+              request:str|None=None, target:str|None=None,
+              trigger_name:str|None=None, trigger:str|None=None)
+
+
+
def test_request(url: str='/', headers: dict={}, method: str='get') -> Request:
+    scope = {
+        'type': 'http',
+        'method': method,
+        'path': url,
+        'headers': Headers(headers).raw,
+        'query_string': b'',
+        'scheme': 'http',
+        'client': ('127.0.0.1', 8000),
+        'server': ('127.0.0.1', 8000),
+    }
+    receive = lambda: {"body": b"", "more_body": False}
+    return Request(scope, receive)
+
+
+
h = test_request(headers=Headers({'HX-Request':'1'}))
+_get_htmx(h.headers)
+
+
HtmxHeaders(boosted=None, current_url=None, history_restore_request=None, prompt=None, request='1', target=None, trigger_name=None, trigger=None)
+
+
+
+
+
+

Request and response

+
+
test_eq(_fix_anno(Union[str,None], 'a'), 'a')
+test_eq(_fix_anno(float, 0.9), 0.9)
+test_eq(_fix_anno(int, '1'), 1)
+test_eq(_fix_anno(int, ['1','2']), 2)
+test_eq(_fix_anno(list[int], ['1','2']), [1,2])
+test_eq(_fix_anno(list[int], '1'), [1])
+
+
+
d = dict(k=int, l=List[int])
+test_eq(_form_arg('k', "1", d), 1)
+test_eq(_form_arg('l', "1", d), [1])
+test_eq(_form_arg('l', ["1","2"], d), [1,2])
+
+
+

source

+
+

HttpHeader

+
+
 HttpHeader (k:str, v:str)
+
+
+
_to_htmx_header('trigger_after_settle')
+
+
'HX-Trigger-After-Settle'
+
+
+
+

source

+
+
+

HtmxResponseHeaders

+
+
 HtmxResponseHeaders (location=None, push_url=None, redirect=None,
+                      refresh=None, replace_url=None, reswap=None,
+                      retarget=None, reselect=None, trigger=None,
+                      trigger_after_settle=None, trigger_after_swap=None)
+
+

HTMX response headers

+
+
HtmxResponseHeaders(trigger_after_settle='hi')
+
+
HttpHeader(k='HX-Trigger-After-Settle', v='hi')
+
+
+
+

source

+
+
+

form2dict

+
+
 form2dict (form:starlette.datastructures.FormData)
+
+

Convert starlette form data to a dict

+
+
d = [('a',1),('a',2),('b',0)]
+fd = FormData(d)
+res = form2dict(fd)
+test_eq(res['a'], [1,2])
+test_eq(res['b'], 0)
+
+
+

source

+
+
+

parse_form

+
+
 parse_form (req:starlette.requests.Request)
+
+

Starlette errors on empty multipart forms, so this checks for that situation

+
+
async def f(req):
+    def _f(p:HttpHeader): ...
+    p = first(_params(_f).values())
+    result = await _from_body(req, p)
+    return JSONResponse(result.__dict__)
+
+client = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])]))
+
+d = dict(k='value1',v=['value2','value3'])
+response = client.post('/', data=d)
+print(response.json())
+
+
{'k': 'value1', 'v': 'value3'}
+
+
+
+
async def f(req): return Response(str(req.query_params.getlist('x')))
+client = TestClient(Starlette(routes=[Route('/', f, methods=['GET'])]))
+client.get('/?x=1&x=2').text
+
+
"['1', '2']"
+
+
+
+
def g(req, this:Starlette, a:str, b:HttpHeader): ...
+
+async def f(req):
+    a = await _wrap_req(req, _params(g))
+    return Response(str(a))
+
+client = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])]))
+response = client.post('/?a=1', data=d)
+print(response.text)
+
+
[<starlette.requests.Request object>, <starlette.applications.Starlette object>, '1', HttpHeader(k='value1', v='value3')]
+
+
+
+
def g(req, this:Starlette, a:str, b:HttpHeader): ...
+
+async def f(req):
+    a = await _wrap_req(req, _params(g))
+    return Response(str(a))
+
+client = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])]))
+response = client.post('/?a=1', data=d)
+print(response.text)
+
+
[<starlette.requests.Request object>, <starlette.applications.Starlette object>, '1', HttpHeader(k='value1', v='value3')]
+
+
+
+

source

+
+
+

flat_xt

+
+
 flat_xt (lst)
+
+

Flatten lists

+
+
x = ft('a',1)
+test_eq(flat_xt([x, x, [x,x]]), (x,)*4)
+test_eq(flat_xt(x), (x,))
+
+
+

source

+
+
+

Beforeware

+
+
 Beforeware (f, skip=None)
+
+

Initialize self. See help(type(self)) for accurate signature.

+
+
+
+

Websockets / SSE

+
+
def on_receive(self, msg:str): return f"Message text was: {msg}"
+c = _ws_endp(on_receive)
+cli = TestClient(Starlette(routes=[WebSocketRoute('/', _ws_endp(on_receive))]))
+with cli.websocket_connect('/') as ws:
+    ws.send_text('{"msg":"Hi!"}')
+    data = ws.receive_text()
+    assert data == 'Message text was: Hi!'
+
+
+

source

+
+

EventStream

+
+
 EventStream (s)
+
+

Create a text/event-stream response from s

+
+

source

+
+
+

signal_shutdown

+
+
 signal_shutdown ()
+
+
+
+
+

Routing and application

+
+

source

+
+

uri

+
+
 uri (_arg, **kwargs)
+
+
+

source

+
+
+

decode_uri

+
+
 decode_uri (s)
+
+
+

source

+
+
+

StringConvertor.to_string

+
+
 StringConvertor.to_string (value:str)
+
+
+

source

+
+
+

HTTPConnection.url_path_for

+
+
 HTTPConnection.url_path_for (name:str, **path_params)
+
+
+

source

+
+
+

flat_tuple

+
+
 flat_tuple (o)
+
+

Flatten lists

+
+

source

+
+
+

noop_body

+
+
 noop_body (c, req)
+
+

Default Body wrap function which just returns the content

+
+

source

+
+
+

respond

+
+
 respond (req, heads, bdy)
+
+

Default FT response creation function

+
+

source

+
+
+

Redirect

+
+
 Redirect (loc)
+
+

Use HTMX or Starlette RedirectResponse as required to redirect to loc

+
+

source

+
+
+

get_key

+
+
 get_key (key=None, fname='.sesskey')
+
+
+
get_key()
+
+
'5a5e5544-5ee8-46f2-836e-924976ce8b58'
+
+
+
+

source

+
+
+

qp

+
+
 qp (p:str, **kw)
+
+

Add query parameters to path p

+
+
qp('/foo', a=None, b=False, c=[1,2], d='bar')
+
+
'/foo?a=&b=&c=1&c=2&d=bar'
+
+
+
+

source

+
+
+

def_hdrs

+
+
 def_hdrs (htmx=True, surreal=True)
+
+

Default headers for a FastHTML app

+
+

source

+
+
+

FastHTML

+
+
 FastHTML (debug=False, routes=None, middleware=None, title:str='FastHTML
+           page', exception_handlers=None, on_startup=None,
+           on_shutdown=None, lifespan=None, hdrs=None, ftrs=None,
+           exts=None, before=None, after=None, surreal=True, htmx=True,
+           default_hdrs=True, sess_cls=<class
+           'starlette.middleware.sessions.SessionMiddleware'>,
+           secret_key=None, session_cookie='session_', max_age=31536000,
+           sess_path='/', same_site='lax', sess_https_only=False,
+           sess_domain=None, key_fname='.sesskey', body_wrap=<function
+           noop_body>, htmlkw=None, nb_hdrs=False, **bodykw)
+
+

Creates an Starlette application.

+
+

source

+
+
+

FastHTML.ws

+
+
 FastHTML.ws (path:str, conn=None, disconn=None, name=None,
+              middleware=None)
+
+

Add a websocket route at path

+
+

source

+
+
+

nested_name

+
+
 nested_name (f)
+
+

*Get name of function f using ’_’ to join nested function names*

+
+
def f():
+    def g(): ...
+    return g
+
+
+
func = f()
+nested_name(func)
+
+
'f_g'
+
+
+
+

source

+
+
+

FastHTML.route

+
+
 FastHTML.route (path:str=None, methods=None, name=None,
+                 include_in_schema=True, body_wrap=None)
+
+

Add a route at path

+
+
app = FastHTML()
+@app.get
+def foo(a:str, b:list[int]): ...
+
+print(app.routes)
+foo.to(a='bar', b=[1,2])
+
+
[Route(path='/foo', name='foo', methods=['GET', 'HEAD'])]
+
+
+
'/foo?a=bar&b=1&b=2'
+
+
+
+

source

+
+
+

serve

+
+
 serve (appname=None, app='app', host='0.0.0.0', port=None, reload=True,
+        reload_includes:list[str]|str|None=None,
+        reload_excludes:list[str]|str|None=None)
+
+

Run the app in an async server, with live reload set as the default.

+ ++++++ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
appnameNoneTypeNoneName of the module
appstrappApp instance to be served
hoststr0.0.0.0If host is 0.0.0.0 will convert to localhost
portNoneTypeNoneIf port is None it will default to 5001 or the PORT environment variable
reloadboolTrueDefault is to reload the app upon code changes
reload_includeslist[str] | str | NoneNoneAdditional files to watch for changes
reload_excludeslist[str] | str | NoneNoneFiles to ignore for changes
+
+

source

+
+
+

Client

+
+
 Client (app, url='http://testserver')
+
+

A simple httpx ASGI client that doesn’t require async

+
+
app = FastHTML(routes=[Route('/', lambda _: Response('test'))])
+cli = Client(app)
+
+cli.get('/').text
+
+
'test'
+
+
+

Note that you can also use Starlette’s TestClient instead of FastHTML’s Client. They should be largely interchangable.

+
+
+
+

FastHTML Tests

+
+
def get_cli(app): return app,TestClient(app),app.route
+
+
+
app,cli,rt = get_cli(FastHTML(secret_key='soopersecret'))
+
+
+
app,cli,rt = get_cli(FastHTML(title="My Custom Title"))
+@app.get
+def foo(): return Div("Hello World")
+
+print(app.routes)
+
+response = cli.get('/foo')
+assert '<title>My Custom Title</title>' in response.text
+
+foo.to(param='value')
+
+
[Route(path='/foo', name='foo', methods=['GET', 'HEAD'])]
+
+
+
'/foo?param=value'
+
+
+
+
app,cli,rt = get_cli(FastHTML())
+
+@rt('/xt2')
+def get(): return H1('bar')
+
+txt = cli.get('/xt2').text
+assert '<title>FastHTML page</title>' in txt and '<h1>bar</h1>' in txt and '<html>' in txt
+
+
+
@rt("/hi")
+def get(): return 'Hi there'
+
+r = cli.get('/hi')
+r.text
+
+
'Hi there'
+
+
+
+
@rt("/hi")
+def post(): return 'Postal'
+
+cli.post('/hi').text
+
+
'Postal'
+
+
+
+
@app.get("/hostie")
+def show_host(req): return req.headers['host']
+
+cli.get('/hostie').text
+
+
'testserver'
+
+
+
+
@app.get("/setsess")
+def set_sess(session):
+   session['foo'] = 'bar'
+   return 'ok'
+
+@app.ws("/ws")
+def ws(self, msg:str, ws:WebSocket, session): return f"Message text was: {msg} with session {session.get('foo')}, from client: {ws.client}"
+
+cli.get('/setsess')
+with cli.websocket_connect('/ws') as ws:
+    ws.send_text('{"msg":"Hi!"}')
+    data = ws.receive_text()
+assert 'Message text was: Hi! with session bar' in data
+print(data)
+
+
Message text was: Hi! with session bar, from client: Address(host='testclient', port=50000)
+
+
+
+
@rt
+def yoyo(): return 'a yoyo'
+
+cli.post('/yoyo').text
+
+
'a yoyo'
+
+
+
+
@app.get
+def autopost(): return Html(Div('Text.', hx_post=yoyo()))
+print(cli.get('/autopost').text)
+
+
 <!doctype html>
+ <html>
+   <div hx-post="a yoyo">Text.</div>
+ </html>
+
+
+
+
+
@app.get
+def autopost2(): return Html(Body(Div('Text.', cls='px-2', hx_post=show_host.to(a='b'))))
+print(cli.get('/autopost2').text)
+
+
 <!doctype html>
+ <html>
+   <body>
+     <div class="px-2" hx-post="/hostie?a=b">Text.</div>
+   </body>
+ </html>
+
+
+
+
+
@app.get
+def autoget2(): return Html(Div('Text.', hx_get=show_host))
+print(cli.get('/autoget2').text)
+
+
 <!doctype html>
+ <html>
+   <div hx-get="/hostie">Text.</div>
+ </html>
+
+
+
+
+
@rt('/user/{nm}', name='gday')
+def get(nm:str=''): return f"Good day to you, {nm}!"
+cli.get('/user/Alexis').text
+
+
'Good day to you, Alexis!'
+
+
+
+
@app.get
+def autolink(): return Html(Div('Text.', link=uri('gday', nm='Alexis')))
+print(cli.get('/autolink').text)
+
+
 <!doctype html>
+ <html>
+   <div href="/user/Alexis">Text.</div>
+ </html>
+
+
+
+
+
@rt('/link')
+def get(req): return f"{req.url_for('gday', nm='Alexis')}; {req.url_for('show_host')}"
+
+cli.get('/link').text
+
+
'http://testserver/user/Alexis; http://testserver/hostie'
+
+
+
+
@app.get("/background")
+async def background_task(request):
+    async def long_running_task():
+        await asyncio.sleep(0.1)
+        print("Background task completed!")
+    return P("Task started"), BackgroundTask(long_running_task)
+
+response = cli.get("/background")
+
+
Background task completed!
+
+
+
+
test_eq(app.router.url_path_for('gday', nm='Jeremy'), '/user/Jeremy')
+
+
+
hxhdr = {'headers':{'hx-request':"1"}}
+
+@rt('/ft')
+def get(): return Title('Foo'),H1('bar')
+
+txt = cli.get('/ft').text
+assert '<title>Foo</title>' in txt and '<h1>bar</h1>' in txt and '<html>' in txt
+
+@rt('/xt2')
+def get(): return H1('bar')
+
+txt = cli.get('/xt2').text
+assert '<title>FastHTML page</title>' in txt and '<h1>bar</h1>' in txt and '<html>' in txt
+
+assert cli.get('/xt2', **hxhdr).text.strip() == '<h1>bar</h1>'
+
+@rt('/xt3')
+def get(): return Html(Head(Title('hi')), Body(P('there')))
+
+txt = cli.get('/xt3').text
+assert '<title>FastHTML page</title>' not in txt and '<title>hi</title>' in txt and '<p>there</p>' in txt
+
+
+
@rt('/oops')
+def get(nope): return nope
+test_warns(lambda: cli.get('/oops?nope=1'))
+
+
+
def test_r(cli, path, exp, meth='get', hx=False, **kwargs):
+    if hx: kwargs['headers'] = {'hx-request':"1"}
+    test_eq(getattr(cli, meth)(path, **kwargs).text, exp)
+
+ModelName = str_enum('ModelName', "alexnet", "resnet", "lenet")
+fake_db = [{"name": "Foo"}, {"name": "Bar"}]
+
+
+
@rt('/html/{idx}')
+async def get(idx:int): return Body(H4(f'Next is {idx+1}.'))
+
+
+
@rt("/models/{nm}")
+def get(nm:ModelName): return nm
+
+@rt("/files/{path}")
+async def get(path: Path): return path.with_suffix('.txt')
+
+@rt("/items/")
+def get(idx:int|None = 0): return fake_db[idx]
+
+@rt("/idxl/")
+def get(idx:list[int]): return str(idx)
+
+
+
r = cli.get('/html/1', headers={'hx-request':"1"})
+assert '<h4>Next is 2.</h4>' in r.text
+test_r(cli, '/models/alexnet', 'alexnet')
+test_r(cli, '/files/foo', 'foo.txt')
+test_r(cli, '/items/?idx=1', '{"name":"Bar"}')
+test_r(cli, '/items/', '{"name":"Foo"}')
+assert cli.get('/items/?idx=g').text=='404 Not Found'
+assert cli.get('/items/?idx=g').status_code == 404
+test_r(cli, '/idxl/?idx=1&idx=2', '[1, 2]')
+assert cli.get('/idxl/?idx=1&idx=g').status_code == 404
+
+
+
app = FastHTML()
+rt = app.route
+cli = TestClient(app)
+@app.route(r'/static/{path:path}.jpg')
+def index(path:str): return f'got {path}'
+cli.get('/static/sub/a.b.jpg').text
+
+
'got sub/a.b'
+
+
+
+
app.chk = 'foo'
+
+
+
@app.get("/booly/")
+def _(coming:bool=True): return 'Coming' if coming else 'Not coming'
+
+@app.get("/datie/")
+def _(d:parsed_date): return d
+
+@app.get("/ua")
+async def _(user_agent:str): return user_agent
+
+@app.get("/hxtest")
+def _(htmx): return htmx.request
+
+@app.get("/hxtest2")
+def _(foo:HtmxHeaders, req): return foo.request
+
+@app.get("/app")
+def _(app): return app.chk
+
+@app.get("/app2")
+def _(foo:FastHTML): return foo.chk,HttpHeader("mykey", "myval")
+
+@app.get("/app3")
+def _(foo:FastHTML): return HtmxResponseHeaders(location="http://example.org")
+
+@app.get("/app4")
+def _(foo:FastHTML): return Redirect("http://example.org")
+
+
+
test_r(cli, '/booly/?coming=true', 'Coming')
+test_r(cli, '/booly/?coming=no', 'Not coming')
+date_str = "17th of May, 2024, 2p"
+test_r(cli, f'/datie/?d={date_str}', '2024-05-17 14:00:00')
+test_r(cli, '/ua', 'FastHTML', headers={'User-Agent':'FastHTML'})
+test_r(cli, '/hxtest' , '1', headers={'HX-Request':'1'})
+test_r(cli, '/hxtest2', '1', headers={'HX-Request':'1'})
+test_r(cli, '/app' , 'foo')
+
+
+
r = cli.get('/app2', **hxhdr)
+test_eq(r.text, 'foo')
+test_eq(r.headers['mykey'], 'myval')
+
+
+
r = cli.get('/app3')
+test_eq(r.headers['HX-Location'], 'http://example.org')
+
+
+
r = cli.get('/app4', follow_redirects=False)
+test_eq(r.status_code, 303)
+
+
+
r = cli.get('/app4', headers={'HX-Request':'1'})
+test_eq(r.headers['HX-Redirect'], 'http://example.org')
+
+
+
@rt
+def meta():
+    return ((Title('hi'),H1('hi')),
+        (Meta(property='image'), Meta(property='site_name'))
+    )
+
+t = cli.post('/meta').text
+assert re.search(r'<body>\s*<h1>hi</h1>\s*</body>', t)
+assert '<meta' in t
+
+
+
@app.post('/profile/me')
+def profile_update(username: str): return username
+
+test_r(cli, '/profile/me', 'Alexis', 'post', data={'username' : 'Alexis'})
+test_r(cli, '/profile/me', 'Missing required field: username', 'post', data={})
+
+
+
# Example post request with parameter that has a default value
+@app.post('/pet/dog')
+def pet_dog(dogname: str = None): return dogname
+
+# Working post request with optional parameter
+test_r(cli, '/pet/dog', '', 'post', data={})
+
+
+
@dataclass
+class Bodie: a:int;b:str
+
+@rt("/bodie/{nm}")
+def post(nm:str, data:Bodie):
+    res = asdict(data)
+    res['nm'] = nm
+    return res
+
+@app.post("/bodied/")
+def bodied(data:dict): return data
+
+nt = namedtuple('Bodient', ['a','b'])
+
+@app.post("/bodient/")
+def bodient(data:nt): return asdict(data)
+
+class BodieTD(TypedDict): a:int;b:str='foo'
+
+@app.post("/bodietd/")
+def bodient(data:BodieTD): return data
+
+class Bodie2:
+    a:int|None; b:str
+    def __init__(self, a, b='foo'): store_attr()
+
+@rt("/bodie2/", methods=['get','post'])
+def bodie(d:Bodie2): return f"a: {d.a}; b: {d.b}"
+
+
+
from fasthtml.xtend import Titled
+
+
+
d = dict(a=1, b='foo')
+
+test_r(cli, '/bodie/me', '{"a":1,"b":"foo","nm":"me"}', 'post', data=dict(a=1, b='foo', nm='me'))
+test_r(cli, '/bodied/', '{"a":"1","b":"foo"}', 'post', data=d)
+test_r(cli, '/bodie2/', 'a: 1; b: foo', 'post', data={'a':1})
+test_r(cli, '/bodie2/?a=1&b=foo&nm=me', 'a: 1; b: foo')
+test_r(cli, '/bodient/', '{"a":"1","b":"foo"}', 'post', data=d)
+test_r(cli, '/bodietd/', '{"a":1,"b":"foo"}', 'post', data=d)
+
+
+
# Testing POST with Content-Type: application/json
+@app.post("/")
+def index(it: Bodie): return Titled("It worked!", P(f"{it.a}, {it.b}"))
+
+s = json.dumps({"b": "Lorem", "a": 15})
+response = cli.post('/', headers={"Content-Type": "application/json"}, data=s).text
+assert "<title>It worked!</title>" in response and "<p>15, Lorem</p>" in response
+
+
+
# Testing POST with Content-Type: application/json
+@app.post("/bodytext")
+def index(body): return body
+
+response = cli.post('/bodytext', headers={"Content-Type": "application/json"}, data=s).text
+test_eq(response, '{"b": "Lorem", "a": 15}')
+
+
+
files = [ ('files', ('file1.txt', b'content1')),
+         ('files', ('file2.txt', b'content2')) ]
+
+
+
@rt("/uploads")
+async def post(files:list[UploadFile]):
+    return ','.join([(await file.read()).decode() for file in files])
+
+res = cli.post('/uploads', files=files)
+print(res.status_code)
+print(res.text)
+
+
200
+content1,content2
+
+
+
+
res = cli.post('/uploads', files=[files[0]])
+print(res.status_code)
+print(res.text)
+
+
200
+content1
+
+
+
+
@rt("/setsess")
+def get(sess, foo:str=''):
+    now = datetime.now()
+    sess['auth'] = str(now)
+    return f'Set to {now}'
+
+@rt("/getsess")
+def get(sess): return f'Session time: {sess["auth"]}'
+
+print(cli.get('/setsess').text)
+time.sleep(0.01)
+
+cli.get('/getsess').text
+
+
Set to 2025-01-12 14:12:46.576323
+
+
+
'Session time: 2025-01-12 14:12:46.576323'
+
+
+
+
@rt("/sess-first")
+def post(sess, name: str):
+    sess["name"] = name
+    return str(sess)
+
+cli.post('/sess-first', data={'name': 2})
+
+@rt("/getsess-all")
+def get(sess): return sess['name']
+
+test_eq(cli.get('/getsess-all').text, '2')
+
+
+
@rt("/upload")
+async def post(uf:UploadFile): return (await uf.read()).decode()
+
+with open('../../CHANGELOG.md', 'rb') as f:
+    print(cli.post('/upload', files={'uf':f}, data={'msg':'Hello'}).text[:15])
+
+
# Release notes
+
+
+
+
@rt("/form-submit/{list_id}")
+def options(list_id: str):
+    headers = {
+        'Access-Control-Allow-Origin': '*',
+        'Access-Control-Allow-Methods': 'POST',
+        'Access-Control-Allow-Headers': '*',
+    }
+    return Response(status_code=200, headers=headers)
+
+
+
h = cli.options('/form-submit/2').headers
+test_eq(h['Access-Control-Allow-Methods'], 'POST')
+
+
+
from fasthtml.authmw import user_pwd_auth
+
+
+
def _not_found(req, exc): return Div('nope')
+
+app,cli,rt = get_cli(FastHTML(exception_handlers={404:_not_found}))
+
+txt = cli.get('/').text
+assert '<div>nope</div>' in txt
+assert '<!doctype html>' in txt
+
+
+
app,cli,rt = get_cli(FastHTML())
+
+@rt("/{name}/{age}")
+def get(name: str, age: int):
+    return Titled(f"Hello {name.title()}, age {age}")
+
+assert '<title>Hello Uma, age 5</title>' in cli.get('/uma/5').text
+assert '404 Not Found' in cli.get('/uma/five').text
+
+
+
auth = user_pwd_auth(testuser='spycraft')
+app,cli,rt = get_cli(FastHTML(middleware=[auth]))
+
+@rt("/locked")
+def get(auth): return 'Hello, ' + auth
+
+test_eq(cli.get('/locked').text, 'not authenticated')
+test_eq(cli.get('/locked', auth=("testuser","spycraft")).text, 'Hello, testuser')
+
+
+
auth = user_pwd_auth(testuser='spycraft')
+app,cli,rt = get_cli(FastHTML(middleware=[auth]))
+
+@rt("/locked")
+def get(auth): return 'Hello, ' + auth
+
+test_eq(cli.get('/locked').text, 'not authenticated')
+test_eq(cli.get('/locked', auth=("testuser","spycraft")).text, 'Hello, testuser')
+
+
+
+

APIRouter

+
+

source

+
+

RouteFuncs

+
+
 RouteFuncs ()
+
+

Initialize self. See help(type(self)) for accurate signature.

+
+

source

+
+
+

APIRouter

+
+
 APIRouter (prefix:str|None=None, body_wrap=<function noop_body>)
+
+

Add routes to an app

+
+
ar = APIRouter()
+
+
+
@ar("/hi")
+def get(): return 'Hi there'
+@ar("/hi")
+def post(): return 'Postal'
+@ar
+def ho(): return 'Ho ho'
+@ar("/hostie")
+def show_host(req): return req.headers['host']
+@ar
+def yoyo(): return 'a yoyo'
+@ar
+def index(): return "home page"
+
+@ar.ws("/ws")
+def ws(self, msg:str): return f"Message text was: {msg}"
+
+
+
app,cli,_ = get_cli(FastHTML())
+ar.to_app(app)
+
+
+
assert str(yoyo) == '/yoyo'
+# ensure route functions are properly discoverable on `APIRouter` and `APIRouter.rt_funcs`
+assert ar.prefix == ''
+assert str(ar.rt_funcs.index) == '/'
+assert str(ar.index) == '/'
+with ExceptionExpected(): ar.blah()
+with ExceptionExpected(): ar.rt_funcs.blah()
+# ensure any route functions named using an HTTPMethod are not discoverable via `rt_funcs`
+assert "get" not in ar.rt_funcs._funcs.keys()
+
+
+
test_eq(cli.get('/hi').text, 'Hi there')
+test_eq(cli.post('/hi').text, 'Postal')
+test_eq(cli.get('/hostie').text, 'testserver')
+test_eq(cli.post('/yoyo').text, 'a yoyo')
+
+test_eq(cli.get('/ho').text, 'Ho ho')
+test_eq(cli.post('/ho').text, 'Ho ho')
+
+
+
with cli.websocket_connect('/ws') as ws:
+    ws.send_text('{"msg":"Hi!"}')
+    data = ws.receive_text()
+    assert data == 'Message text was: Hi!'
+
+
+
ar2 = APIRouter("/products")
+
+
+
@ar2("/hi")
+def get(): return 'Hi there'
+@ar2("/hi")
+def post(): return 'Postal'
+@ar2
+def ho(): return 'Ho ho'
+@ar2("/hostie")
+def show_host(req): return req.headers['host']
+@ar2
+def yoyo(): return 'a yoyo'
+@ar2
+def index(): return "home page"
+
+@ar2.ws("/ws")
+def ws(self, msg:str): return f"Message text was: {msg}"
+
+
+
app,cli,_ = get_cli(FastHTML())
+ar2.to_app(app)
+
+
+
assert str(yoyo) == '/products/yoyo'
+assert ar2.prefix == '/products'
+assert str(ar2.rt_funcs.index) == '/products/'
+assert str(ar2.index) == '/products/'
+assert str(ar.index) == '/'
+with ExceptionExpected(): ar2.blah()
+with ExceptionExpected(): ar2.rt_funcs.blah()
+assert "get" not in ar2.rt_funcs._funcs.keys()
+
+
+
test_eq(cli.get('/products/hi').text, 'Hi there')
+test_eq(cli.post('/products/hi').text, 'Postal')
+test_eq(cli.get('/products/hostie').text, 'testserver')
+test_eq(cli.post('/products/yoyo').text, 'a yoyo')
+
+test_eq(cli.get('/products/ho').text, 'Ho ho')
+test_eq(cli.post('/products/ho').text, 'Ho ho')
+
+
+
with cli.websocket_connect('/products/ws') as ws:
+    ws.send_text('{"msg":"Hi!"}')
+    data = ws.receive_text()
+    assert data == 'Message text was: Hi!'
+
+
+
@ar.get
+def hi2(): return 'Hi there'
+@ar.get("/hi3")
+def _(): return 'Hi there'
+@ar.post("/post2")
+def _(): return 'Postal'
+
+@ar2.get
+def hi2(): return 'Hi there'
+@ar2.get("/hi3")
+def _(): return 'Hi there'
+@ar2.post("/post2")
+def _(): return 'Postal'
+
+
+
+
+

Extras

+
+
app,cli,rt = get_cli(FastHTML(secret_key='soopersecret'))
+
+
+

source

+ +
+

reg_re_param

+
+
 reg_re_param (m, s)
+
+
+

source

+
+
+

FastHTML.static_route_exts

+
+
 FastHTML.static_route_exts (prefix='/', static_path='.', exts='static')
+
+

Add a static route at URL path prefix with files from static_path and exts defined by reg_re_param()

+
+
reg_re_param("imgext", "ico|gif|jpg|jpeg|webm|pdf")
+
+@rt(r'/static/{path:path}{fn}.{ext:imgext}')
+def get(fn:str, path:str, ext:str): return f"Getting {fn}.{ext} from /{path}"
+
+test_r(cli, '/static/foo/jph.me.ico', 'Getting jph.me.ico from /foo/')
+
+
+
app.static_route_exts()
+assert 'These are the source notebooks for FastHTML' in cli.get('/README.txt').text
+
+
+

source

+
+
+

FastHTML.static_route

+
+
 FastHTML.static_route (ext='', prefix='/', static_path='.')
+
+

Add a static route at URL path prefix with files from static_path and single ext (including the ‘.’)

+
+
app.static_route('.md', static_path='../..')
+assert 'THIS FILE WAS AUTOGENERATED' in cli.get('/README.md').text
+
+
+

source

+
+
+

MiddlewareBase

+
+
 MiddlewareBase ()
+
+

Initialize self. See help(type(self)) for accurate signature.

+
+

source

+
+
+

FtResponse

+
+
 FtResponse (content, status_code:int=200, headers=None, cls=<class
+             'starlette.responses.HTMLResponse'>,
+             media_type:str|None=None)
+
+

Wrap an FT response with any Starlette Response

+
+
@rt('/ftr')
+def get():
+    cts = Title('Foo'),H1('bar')
+    return FtResponse(cts, status_code=201, headers={'Location':'/foo/1'})
+
+r = cli.get('/ftr')
+
+test_eq(r.status_code, 201)
+test_eq(r.headers['location'], '/foo/1')
+txt = r.text
+assert '<title>Foo</title>' in txt and '<h1>bar</h1>' in txt and '<html>' in txt
+
+
+

source

+
+
+

unqid

+
+
 unqid ()
+
+
+

source

+
+
+

setup_ws

+
+
 setup_ws (app, f=<function noop>)
+
+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/core.html.md b/api/core.html.md new file mode 100644 index 00000000..0a6b86ad --- /dev/null +++ b/api/core.html.md @@ -0,0 +1,1489 @@ +# Core + + + + +This is the source code to fasthtml. You won’t need to read this unless +you want to understand how things are built behind the scenes, or need +full details of a particular API. The notebook is converted to the +Python module +[fasthtml/core.py](https://github.com/AnswerDotAI/fasthtml/blob/main/fasthtml/core.py) +using [nbdev](https://nbdev.fast.ai/). + +## Imports and utils + +``` python +import time + +from IPython import display +from enum import Enum +from pprint import pprint + +from fastcore.test import * +from starlette.testclient import TestClient +from starlette.requests import Headers +from starlette.datastructures import UploadFile +``` + +We write source code *first*, and then tests come *after*. The tests +serve as both a means to confirm that the code works and also serves as +working examples. The first exported function, +[`parsed_date`](https://AnswerDotAI.github.io/fasthtml/api/core.html#parsed_date), +is an example of this pattern. + +------------------------------------------------------------------------ + +source + +### parsed_date + +> parsed_date (s:str) + +*Convert `s` to a datetime* + +``` python +parsed_date('2pm') +``` + + datetime.datetime(2025, 1, 12, 14, 0) + +``` python +isinstance(date.fromtimestamp(0), date) +``` + + True + +------------------------------------------------------------------------ + +source + +### snake2hyphens + +> snake2hyphens (s:str) + +*Convert `s` from snake case to hyphenated and capitalised* + +``` python +snake2hyphens("snake_case") +``` + + 'Snake-Case' + +------------------------------------------------------------------------ + +source + +### HtmxHeaders + +> HtmxHeaders (boosted:str|None=None, current_url:str|None=None, +> history_restore_request:str|None=None, prompt:str|None=None, +> request:str|None=None, target:str|None=None, +> trigger_name:str|None=None, trigger:str|None=None) + +``` python +def test_request(url: str='/', headers: dict={}, method: str='get') -> Request: + scope = { + 'type': 'http', + 'method': method, + 'path': url, + 'headers': Headers(headers).raw, + 'query_string': b'', + 'scheme': 'http', + 'client': ('127.0.0.1', 8000), + 'server': ('127.0.0.1', 8000), + } + receive = lambda: {"body": b"", "more_body": False} + return Request(scope, receive) +``` + +``` python +h = test_request(headers=Headers({'HX-Request':'1'})) +_get_htmx(h.headers) +``` + + HtmxHeaders(boosted=None, current_url=None, history_restore_request=None, prompt=None, request='1', target=None, trigger_name=None, trigger=None) + +## Request and response + +``` python +test_eq(_fix_anno(Union[str,None], 'a'), 'a') +test_eq(_fix_anno(float, 0.9), 0.9) +test_eq(_fix_anno(int, '1'), 1) +test_eq(_fix_anno(int, ['1','2']), 2) +test_eq(_fix_anno(list[int], ['1','2']), [1,2]) +test_eq(_fix_anno(list[int], '1'), [1]) +``` + +``` python +d = dict(k=int, l=List[int]) +test_eq(_form_arg('k', "1", d), 1) +test_eq(_form_arg('l', "1", d), [1]) +test_eq(_form_arg('l', ["1","2"], d), [1,2]) +``` + +------------------------------------------------------------------------ + +source + +### HttpHeader + +> HttpHeader (k:str, v:str) + +``` python +_to_htmx_header('trigger_after_settle') +``` + + 'HX-Trigger-After-Settle' + +------------------------------------------------------------------------ + +source + +### HtmxResponseHeaders + +> HtmxResponseHeaders (location=None, push_url=None, redirect=None, +> refresh=None, replace_url=None, reswap=None, +> retarget=None, reselect=None, trigger=None, +> trigger_after_settle=None, trigger_after_swap=None) + +*HTMX response headers* + +``` python +HtmxResponseHeaders(trigger_after_settle='hi') +``` + + HttpHeader(k='HX-Trigger-After-Settle', v='hi') + +------------------------------------------------------------------------ + +source + +### form2dict + +> form2dict (form:starlette.datastructures.FormData) + +*Convert starlette form data to a dict* + +``` python +d = [('a',1),('a',2),('b',0)] +fd = FormData(d) +res = form2dict(fd) +test_eq(res['a'], [1,2]) +test_eq(res['b'], 0) +``` + +------------------------------------------------------------------------ + +source + +### parse_form + +> parse_form (req:starlette.requests.Request) + +*Starlette errors on empty multipart forms, so this checks for that +situation* + +``` python +async def f(req): + def _f(p:HttpHeader): ... + p = first(_params(_f).values()) + result = await _from_body(req, p) + return JSONResponse(result.__dict__) + +client = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])])) + +d = dict(k='value1',v=['value2','value3']) +response = client.post('/', data=d) +print(response.json()) +``` + + {'k': 'value1', 'v': 'value3'} + +``` python +async def f(req): return Response(str(req.query_params.getlist('x'))) +client = TestClient(Starlette(routes=[Route('/', f, methods=['GET'])])) +client.get('/?x=1&x=2').text +``` + + "['1', '2']" + +``` python +def g(req, this:Starlette, a:str, b:HttpHeader): ... + +async def f(req): + a = await _wrap_req(req, _params(g)) + return Response(str(a)) + +client = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])])) +response = client.post('/?a=1', data=d) +print(response.text) +``` + + [, , '1', HttpHeader(k='value1', v='value3')] + +``` python +def g(req, this:Starlette, a:str, b:HttpHeader): ... + +async def f(req): + a = await _wrap_req(req, _params(g)) + return Response(str(a)) + +client = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])])) +response = client.post('/?a=1', data=d) +print(response.text) +``` + + [, , '1', HttpHeader(k='value1', v='value3')] + +------------------------------------------------------------------------ + +source + +### flat_xt + +> flat_xt (lst) + +*Flatten lists* + +``` python +x = ft('a',1) +test_eq(flat_xt([x, x, [x,x]]), (x,)*4) +test_eq(flat_xt(x), (x,)) +``` + +------------------------------------------------------------------------ + +source + +### Beforeware + +> Beforeware (f, skip=None) + +*Initialize self. See help(type(self)) for accurate signature.* + +## Websockets / SSE + +``` python +def on_receive(self, msg:str): return f"Message text was: {msg}" +c = _ws_endp(on_receive) +cli = TestClient(Starlette(routes=[WebSocketRoute('/', _ws_endp(on_receive))])) +with cli.websocket_connect('/') as ws: + ws.send_text('{"msg":"Hi!"}') + data = ws.receive_text() + assert data == 'Message text was: Hi!' +``` + +------------------------------------------------------------------------ + +source + +### EventStream + +> EventStream (s) + +*Create a text/event-stream response from `s`* + +------------------------------------------------------------------------ + +source + +### signal_shutdown + +> signal_shutdown () + +## Routing and application + +------------------------------------------------------------------------ + +source + +### uri + +> uri (_arg, **kwargs) + +------------------------------------------------------------------------ + +source + +### decode_uri + +> decode_uri (s) + +------------------------------------------------------------------------ + +source + +### StringConvertor.to_string + +> StringConvertor.to_string (value:str) + +------------------------------------------------------------------------ + +source + +### HTTPConnection.url_path_for + +> HTTPConnection.url_path_for (name:str, **path_params) + +------------------------------------------------------------------------ + +source + +### flat_tuple + +> flat_tuple (o) + +*Flatten lists* + +------------------------------------------------------------------------ + +source + +### noop_body + +> noop_body (c, req) + +*Default Body wrap function which just returns the content* + +------------------------------------------------------------------------ + +source + +### respond + +> respond (req, heads, bdy) + +*Default FT response creation function* + +------------------------------------------------------------------------ + +source + +### Redirect + +> Redirect (loc) + +*Use HTMX or Starlette RedirectResponse as required to redirect to +`loc`* + +------------------------------------------------------------------------ + +source + +### get_key + +> get_key (key=None, fname='.sesskey') + +``` python +get_key() +``` + + '5a5e5544-5ee8-46f2-836e-924976ce8b58' + +------------------------------------------------------------------------ + +source + +### qp + +> qp (p:str, **kw) + +*Add query parameters to path p* + +``` python +qp('/foo', a=None, b=False, c=[1,2], d='bar') +``` + + '/foo?a=&b=&c=1&c=2&d=bar' + +------------------------------------------------------------------------ + +source + +### def_hdrs + +> def_hdrs (htmx=True, surreal=True) + +*Default headers for a FastHTML app* + +------------------------------------------------------------------------ + +source + +### FastHTML + +> FastHTML (debug=False, routes=None, middleware=None, title:str='FastHTML +> page', exception_handlers=None, on_startup=None, +> on_shutdown=None, lifespan=None, hdrs=None, ftrs=None, +> exts=None, before=None, after=None, surreal=True, htmx=True, +> default_hdrs=True, sess_cls= 'starlette.middleware.sessions.SessionMiddleware'>, +> secret_key=None, session_cookie='session_', max_age=31536000, +> sess_path='/', same_site='lax', sess_https_only=False, +> sess_domain=None, key_fname='.sesskey', body_wrap= noop_body>, htmlkw=None, nb_hdrs=False, **bodykw) + +*Creates an Starlette application.* + +------------------------------------------------------------------------ + +source + +### FastHTML.ws + +> FastHTML.ws (path:str, conn=None, disconn=None, name=None, +> middleware=None) + +*Add a websocket route at `path`* + +------------------------------------------------------------------------ + +source + +### nested_name + +> nested_name (f) + +\*Get name of function `f` using ’\_’ to join nested function names\* + +``` python +def f(): + def g(): ... + return g +``` + +``` python +func = f() +nested_name(func) +``` + + 'f_g' + +------------------------------------------------------------------------ + +source + +### FastHTML.route + +> FastHTML.route (path:str=None, methods=None, name=None, +> include_in_schema=True, body_wrap=None) + +*Add a route at `path`* + +``` python +app = FastHTML() +@app.get +def foo(a:str, b:list[int]): ... + +print(app.routes) +foo.to(a='bar', b=[1,2]) +``` + + [Route(path='/foo', name='foo', methods=['GET', 'HEAD'])] + + '/foo?a=bar&b=1&b=2' + +------------------------------------------------------------------------ + +source + +### serve + +> serve (appname=None, app='app', host='0.0.0.0', port=None, reload=True, +> reload_includes:list[str]|str|None=None, +> reload_excludes:list[str]|str|None=None) + +*Run the app in an async server, with live reload set as the default.* + + ++++++ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
appnameNoneTypeNoneName of the module
appstrappApp instance to be served
hoststr0.0.0.0If host is 0.0.0.0 will convert to localhost
portNoneTypeNoneIf port is None it will default to 5001 or the PORT environment +variable
reloadboolTrueDefault is to reload the app upon code changes
reload_includeslist[str] | str | NoneNoneAdditional files to watch for changes
reload_excludeslist[str] | str | NoneNoneFiles to ignore for changes
+ +------------------------------------------------------------------------ + +source + +### Client + +> Client (app, url='http://testserver') + +*A simple httpx ASGI client that doesn’t require `async`* + +``` python +app = FastHTML(routes=[Route('/', lambda _: Response('test'))]) +cli = Client(app) + +cli.get('/').text +``` + + 'test' + +Note that you can also use Starlette’s `TestClient` instead of +FastHTML’s +[`Client`](https://AnswerDotAI.github.io/fasthtml/api/core.html#client). +They should be largely interchangable. + +## FastHTML Tests + +``` python +def get_cli(app): return app,TestClient(app),app.route +``` + +``` python +app,cli,rt = get_cli(FastHTML(secret_key='soopersecret')) +``` + +``` python +app,cli,rt = get_cli(FastHTML(title="My Custom Title")) +@app.get +def foo(): return Div("Hello World") + +print(app.routes) + +response = cli.get('/foo') +assert 'My Custom Title' in response.text + +foo.to(param='value') +``` + + [Route(path='/foo', name='foo', methods=['GET', 'HEAD'])] + + '/foo?param=value' + +``` python +app,cli,rt = get_cli(FastHTML()) + +@rt('/xt2') +def get(): return H1('bar') + +txt = cli.get('/xt2').text +assert 'FastHTML page' in txt and '

bar

' in txt and '' in txt +``` + +``` python +@rt("/hi") +def get(): return 'Hi there' + +r = cli.get('/hi') +r.text +``` + + 'Hi there' + +``` python +@rt("/hi") +def post(): return 'Postal' + +cli.post('/hi').text +``` + + 'Postal' + +``` python +@app.get("/hostie") +def show_host(req): return req.headers['host'] + +cli.get('/hostie').text +``` + + 'testserver' + +``` python +@app.get("/setsess") +def set_sess(session): + session['foo'] = 'bar' + return 'ok' + +@app.ws("/ws") +def ws(self, msg:str, ws:WebSocket, session): return f"Message text was: {msg} with session {session.get('foo')}, from client: {ws.client}" + +cli.get('/setsess') +with cli.websocket_connect('/ws') as ws: + ws.send_text('{"msg":"Hi!"}') + data = ws.receive_text() +assert 'Message text was: Hi! with session bar' in data +print(data) +``` + + Message text was: Hi! with session bar, from client: Address(host='testclient', port=50000) + +``` python +@rt +def yoyo(): return 'a yoyo' + +cli.post('/yoyo').text +``` + + 'a yoyo' + +``` python +@app.get +def autopost(): return Html(Div('Text.', hx_post=yoyo())) +print(cli.get('/autopost').text) +``` + + + +
Text.
+ + +``` python +@app.get +def autopost2(): return Html(Body(Div('Text.', cls='px-2', hx_post=show_host.to(a='b')))) +print(cli.get('/autopost2').text) +``` + + + + +
Text.
+ + + +``` python +@app.get +def autoget2(): return Html(Div('Text.', hx_get=show_host)) +print(cli.get('/autoget2').text) +``` + + + +
Text.
+ + +``` python +@rt('/user/{nm}', name='gday') +def get(nm:str=''): return f"Good day to you, {nm}!" +cli.get('/user/Alexis').text +``` + + 'Good day to you, Alexis!' + +``` python +@app.get +def autolink(): return Html(Div('Text.', link=uri('gday', nm='Alexis'))) +print(cli.get('/autolink').text) +``` + + + +
Text.
+ + +``` python +@rt('/link') +def get(req): return f"{req.url_for('gday', nm='Alexis')}; {req.url_for('show_host')}" + +cli.get('/link').text +``` + + 'http://testserver/user/Alexis; http://testserver/hostie' + +``` python +@app.get("/background") +async def background_task(request): + async def long_running_task(): + await asyncio.sleep(0.1) + print("Background task completed!") + return P("Task started"), BackgroundTask(long_running_task) + +response = cli.get("/background") +``` + + Background task completed! + +``` python +test_eq(app.router.url_path_for('gday', nm='Jeremy'), '/user/Jeremy') +``` + +``` python +hxhdr = {'headers':{'hx-request':"1"}} + +@rt('/ft') +def get(): return Title('Foo'),H1('bar') + +txt = cli.get('/ft').text +assert 'Foo' in txt and '

bar

' in txt and '' in txt + +@rt('/xt2') +def get(): return H1('bar') + +txt = cli.get('/xt2').text +assert 'FastHTML page' in txt and '

bar

' in txt and '' in txt + +assert cli.get('/xt2', **hxhdr).text.strip() == '

bar

' + +@rt('/xt3') +def get(): return Html(Head(Title('hi')), Body(P('there'))) + +txt = cli.get('/xt3').text +assert 'FastHTML page' not in txt and 'hi' in txt and '

there

' in txt +``` + +``` python +@rt('/oops') +def get(nope): return nope +test_warns(lambda: cli.get('/oops?nope=1')) +``` + +``` python +def test_r(cli, path, exp, meth='get', hx=False, **kwargs): + if hx: kwargs['headers'] = {'hx-request':"1"} + test_eq(getattr(cli, meth)(path, **kwargs).text, exp) + +ModelName = str_enum('ModelName', "alexnet", "resnet", "lenet") +fake_db = [{"name": "Foo"}, {"name": "Bar"}] +``` + +``` python +@rt('/html/{idx}') +async def get(idx:int): return Body(H4(f'Next is {idx+1}.')) +``` + +``` python +@rt("/models/{nm}") +def get(nm:ModelName): return nm + +@rt("/files/{path}") +async def get(path: Path): return path.with_suffix('.txt') + +@rt("/items/") +def get(idx:int|None = 0): return fake_db[idx] + +@rt("/idxl/") +def get(idx:list[int]): return str(idx) +``` + +``` python +r = cli.get('/html/1', headers={'hx-request':"1"}) +assert '

Next is 2.

' in r.text +test_r(cli, '/models/alexnet', 'alexnet') +test_r(cli, '/files/foo', 'foo.txt') +test_r(cli, '/items/?idx=1', '{"name":"Bar"}') +test_r(cli, '/items/', '{"name":"Foo"}') +assert cli.get('/items/?idx=g').text=='404 Not Found' +assert cli.get('/items/?idx=g').status_code == 404 +test_r(cli, '/idxl/?idx=1&idx=2', '[1, 2]') +assert cli.get('/idxl/?idx=1&idx=g').status_code == 404 +``` + +``` python +app = FastHTML() +rt = app.route +cli = TestClient(app) +@app.route(r'/static/{path:path}.jpg') +def index(path:str): return f'got {path}' +cli.get('/static/sub/a.b.jpg').text +``` + + 'got sub/a.b' + +``` python +app.chk = 'foo' +``` + +``` python +@app.get("/booly/") +def _(coming:bool=True): return 'Coming' if coming else 'Not coming' + +@app.get("/datie/") +def _(d:parsed_date): return d + +@app.get("/ua") +async def _(user_agent:str): return user_agent + +@app.get("/hxtest") +def _(htmx): return htmx.request + +@app.get("/hxtest2") +def _(foo:HtmxHeaders, req): return foo.request + +@app.get("/app") +def _(app): return app.chk + +@app.get("/app2") +def _(foo:FastHTML): return foo.chk,HttpHeader("mykey", "myval") + +@app.get("/app3") +def _(foo:FastHTML): return HtmxResponseHeaders(location="http://example.org") + +@app.get("/app4") +def _(foo:FastHTML): return Redirect("http://example.org") +``` + +``` python +test_r(cli, '/booly/?coming=true', 'Coming') +test_r(cli, '/booly/?coming=no', 'Not coming') +date_str = "17th of May, 2024, 2p" +test_r(cli, f'/datie/?d={date_str}', '2024-05-17 14:00:00') +test_r(cli, '/ua', 'FastHTML', headers={'User-Agent':'FastHTML'}) +test_r(cli, '/hxtest' , '1', headers={'HX-Request':'1'}) +test_r(cli, '/hxtest2', '1', headers={'HX-Request':'1'}) +test_r(cli, '/app' , 'foo') +``` + +``` python +r = cli.get('/app2', **hxhdr) +test_eq(r.text, 'foo') +test_eq(r.headers['mykey'], 'myval') +``` + +``` python +r = cli.get('/app3') +test_eq(r.headers['HX-Location'], 'http://example.org') +``` + +``` python +r = cli.get('/app4', follow_redirects=False) +test_eq(r.status_code, 303) +``` + +``` python +r = cli.get('/app4', headers={'HX-Request':'1'}) +test_eq(r.headers['HX-Redirect'], 'http://example.org') +``` + +``` python +@rt +def meta(): + return ((Title('hi'),H1('hi')), + (Meta(property='image'), Meta(property='site_name')) + ) + +t = cli.post('/meta').text +assert re.search(r'\s*

hi

\s*', t) +assert 'It worked!" in response and "

15, Lorem

" in response +``` + +``` python +# Testing POST with Content-Type: application/json +@app.post("/bodytext") +def index(body): return body + +response = cli.post('/bodytext', headers={"Content-Type": "application/json"}, data=s).text +test_eq(response, '{"b": "Lorem", "a": 15}') +``` + +``` python +files = [ ('files', ('file1.txt', b'content1')), + ('files', ('file2.txt', b'content2')) ] +``` + +``` python +@rt("/uploads") +async def post(files:list[UploadFile]): + return ','.join([(await file.read()).decode() for file in files]) + +res = cli.post('/uploads', files=files) +print(res.status_code) +print(res.text) +``` + + 200 + content1,content2 + +``` python +res = cli.post('/uploads', files=[files[0]]) +print(res.status_code) +print(res.text) +``` + + 200 + content1 + +``` python +@rt("/setsess") +def get(sess, foo:str=''): + now = datetime.now() + sess['auth'] = str(now) + return f'Set to {now}' + +@rt("/getsess") +def get(sess): return f'Session time: {sess["auth"]}' + +print(cli.get('/setsess').text) +time.sleep(0.01) + +cli.get('/getsess').text +``` + + Set to 2025-01-12 14:12:46.576323 + + 'Session time: 2025-01-12 14:12:46.576323' + +``` python +@rt("/sess-first") +def post(sess, name: str): + sess["name"] = name + return str(sess) + +cli.post('/sess-first', data={'name': 2}) + +@rt("/getsess-all") +def get(sess): return sess['name'] + +test_eq(cli.get('/getsess-all').text, '2') +``` + +``` python +@rt("/upload") +async def post(uf:UploadFile): return (await uf.read()).decode() + +with open('../../CHANGELOG.md', 'rb') as f: + print(cli.post('/upload', files={'uf':f}, data={'msg':'Hello'}).text[:15]) +``` + + # Release notes + +``` python +@rt("/form-submit/{list_id}") +def options(list_id: str): + headers = { + 'Access-Control-Allow-Origin': '*', + 'Access-Control-Allow-Methods': 'POST', + 'Access-Control-Allow-Headers': '*', + } + return Response(status_code=200, headers=headers) +``` + +``` python +h = cli.options('/form-submit/2').headers +test_eq(h['Access-Control-Allow-Methods'], 'POST') +``` + +``` python +from fasthtml.authmw import user_pwd_auth +``` + +``` python +def _not_found(req, exc): return Div('nope') + +app,cli,rt = get_cli(FastHTML(exception_handlers={404:_not_found})) + +txt = cli.get('/').text +assert '
nope
' in txt +assert '' in txt +``` + +``` python +app,cli,rt = get_cli(FastHTML()) + +@rt("/{name}/{age}") +def get(name: str, age: int): + return Titled(f"Hello {name.title()}, age {age}") + +assert 'Hello Uma, age 5' in cli.get('/uma/5').text +assert '404 Not Found' in cli.get('/uma/five').text +``` + +``` python +auth = user_pwd_auth(testuser='spycraft') +app,cli,rt = get_cli(FastHTML(middleware=[auth])) + +@rt("/locked") +def get(auth): return 'Hello, ' + auth + +test_eq(cli.get('/locked').text, 'not authenticated') +test_eq(cli.get('/locked', auth=("testuser","spycraft")).text, 'Hello, testuser') +``` + +``` python +auth = user_pwd_auth(testuser='spycraft') +app,cli,rt = get_cli(FastHTML(middleware=[auth])) + +@rt("/locked") +def get(auth): return 'Hello, ' + auth + +test_eq(cli.get('/locked').text, 'not authenticated') +test_eq(cli.get('/locked', auth=("testuser","spycraft")).text, 'Hello, testuser') +``` + +## APIRouter + +------------------------------------------------------------------------ + +source + +### RouteFuncs + +> RouteFuncs () + +*Initialize self. See help(type(self)) for accurate signature.* + +------------------------------------------------------------------------ + +source + +### APIRouter + +> APIRouter (prefix:str|None=None, body_wrap=) + +*Add routes to an app* + +``` python +ar = APIRouter() +``` + +``` python +@ar("/hi") +def get(): return 'Hi there' +@ar("/hi") +def post(): return 'Postal' +@ar +def ho(): return 'Ho ho' +@ar("/hostie") +def show_host(req): return req.headers['host'] +@ar +def yoyo(): return 'a yoyo' +@ar +def index(): return "home page" + +@ar.ws("/ws") +def ws(self, msg:str): return f"Message text was: {msg}" +``` + +``` python +app,cli,_ = get_cli(FastHTML()) +ar.to_app(app) +``` + +``` python +assert str(yoyo) == '/yoyo' +# ensure route functions are properly discoverable on `APIRouter` and `APIRouter.rt_funcs` +assert ar.prefix == '' +assert str(ar.rt_funcs.index) == '/' +assert str(ar.index) == '/' +with ExceptionExpected(): ar.blah() +with ExceptionExpected(): ar.rt_funcs.blah() +# ensure any route functions named using an HTTPMethod are not discoverable via `rt_funcs` +assert "get" not in ar.rt_funcs._funcs.keys() +``` + +``` python +test_eq(cli.get('/hi').text, 'Hi there') +test_eq(cli.post('/hi').text, 'Postal') +test_eq(cli.get('/hostie').text, 'testserver') +test_eq(cli.post('/yoyo').text, 'a yoyo') + +test_eq(cli.get('/ho').text, 'Ho ho') +test_eq(cli.post('/ho').text, 'Ho ho') +``` + +``` python +with cli.websocket_connect('/ws') as ws: + ws.send_text('{"msg":"Hi!"}') + data = ws.receive_text() + assert data == 'Message text was: Hi!' +``` + +``` python +ar2 = APIRouter("/products") +``` + +``` python +@ar2("/hi") +def get(): return 'Hi there' +@ar2("/hi") +def post(): return 'Postal' +@ar2 +def ho(): return 'Ho ho' +@ar2("/hostie") +def show_host(req): return req.headers['host'] +@ar2 +def yoyo(): return 'a yoyo' +@ar2 +def index(): return "home page" + +@ar2.ws("/ws") +def ws(self, msg:str): return f"Message text was: {msg}" +``` + +``` python +app,cli,_ = get_cli(FastHTML()) +ar2.to_app(app) +``` + +``` python +assert str(yoyo) == '/products/yoyo' +assert ar2.prefix == '/products' +assert str(ar2.rt_funcs.index) == '/products/' +assert str(ar2.index) == '/products/' +assert str(ar.index) == '/' +with ExceptionExpected(): ar2.blah() +with ExceptionExpected(): ar2.rt_funcs.blah() +assert "get" not in ar2.rt_funcs._funcs.keys() +``` + +``` python +test_eq(cli.get('/products/hi').text, 'Hi there') +test_eq(cli.post('/products/hi').text, 'Postal') +test_eq(cli.get('/products/hostie').text, 'testserver') +test_eq(cli.post('/products/yoyo').text, 'a yoyo') + +test_eq(cli.get('/products/ho').text, 'Ho ho') +test_eq(cli.post('/products/ho').text, 'Ho ho') +``` + +``` python +with cli.websocket_connect('/products/ws') as ws: + ws.send_text('{"msg":"Hi!"}') + data = ws.receive_text() + assert data == 'Message text was: Hi!' +``` + +``` python +@ar.get +def hi2(): return 'Hi there' +@ar.get("/hi3") +def _(): return 'Hi there' +@ar.post("/post2") +def _(): return 'Postal' + +@ar2.get +def hi2(): return 'Hi there' +@ar2.get("/hi3") +def _(): return 'Hi there' +@ar2.post("/post2") +def _(): return 'Postal' +``` + +## Extras + +``` python +app,cli,rt = get_cli(FastHTML(secret_key='soopersecret')) +``` + +------------------------------------------------------------------------ + +source + +### cookie + +> cookie (key:str, value='', max_age=None, expires=None, path='/', +> domain=None, secure=False, httponly=False, samesite='lax') + +*Create a ‘set-cookie’ +[`HttpHeader`](https://AnswerDotAI.github.io/fasthtml/api/core.html#httpheader)* + +``` python +@rt("/setcookie") +def get(req): return cookie('now', datetime.now()) + +@rt("/getcookie") +def get(now:parsed_date): return f'Cookie was set at time {now.time()}' + +print(cli.get('/setcookie').text) +time.sleep(0.01) +cli.get('/getcookie').text +``` + + 'Cookie was set at time 14:12:47.159530' + +------------------------------------------------------------------------ + +source + +### reg_re_param + +> reg_re_param (m, s) + +------------------------------------------------------------------------ + +source + +### FastHTML.static_route_exts + +> FastHTML.static_route_exts (prefix='/', static_path='.', exts='static') + +*Add a static route at URL path `prefix` with files from `static_path` +and `exts` defined by +[`reg_re_param()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#reg_re_param)* + +``` python +reg_re_param("imgext", "ico|gif|jpg|jpeg|webm|pdf") + +@rt(r'/static/{path:path}{fn}.{ext:imgext}') +def get(fn:str, path:str, ext:str): return f"Getting {fn}.{ext} from /{path}" + +test_r(cli, '/static/foo/jph.me.ico', 'Getting jph.me.ico from /foo/') +``` + +``` python +app.static_route_exts() +assert 'These are the source notebooks for FastHTML' in cli.get('/README.txt').text +``` + +------------------------------------------------------------------------ + +source + +### FastHTML.static_route + +> FastHTML.static_route (ext='', prefix='/', static_path='.') + +*Add a static route at URL path `prefix` with files from `static_path` +and single `ext` (including the ‘.’)* + +``` python +app.static_route('.md', static_path='../..') +assert 'THIS FILE WAS AUTOGENERATED' in cli.get('/README.md').text +``` + +------------------------------------------------------------------------ + +source + +### MiddlewareBase + +> MiddlewareBase () + +*Initialize self. See help(type(self)) for accurate signature.* + +------------------------------------------------------------------------ + +source + +### FtResponse + +> FtResponse (content, status_code:int=200, headers=None, cls= 'starlette.responses.HTMLResponse'>, +> media_type:str|None=None) + +*Wrap an FT response with any Starlette `Response`* + +``` python +@rt('/ftr') +def get(): + cts = Title('Foo'),H1('bar') + return FtResponse(cts, status_code=201, headers={'Location':'/foo/1'}) + +r = cli.get('/ftr') + +test_eq(r.status_code, 201) +test_eq(r.headers['location'], '/foo/1') +txt = r.text +assert 'Foo' in txt and '

bar

' in txt and '' in txt +``` + +------------------------------------------------------------------------ + +source + +### unqid + +> unqid () + +------------------------------------------------------------------------ + +source + +### setup_ws + +> setup_ws (app, f=) diff --git a/api/js.html b/api/js.html new file mode 100644 index 00000000..4ed06f42 --- /dev/null +++ b/api/js.html @@ -0,0 +1,1144 @@ + + + + + + + + + + +Javascript examples – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Javascript examples

+
+ +
+
+ Basic external Javascript lib wrappers +
+
+ + +
+ + + + +
+ + + +
+ + + +

To expedite fast development, FastHTML comes with several built-in Javascript and formatting components. These are largely provided to demonstrate FastHTML JS patterns. There’s far too many JS libs for FastHTML to wrap them all, and as shown here the code to add FastHTML support is very simple anyway.

+
+

source

+
+

light_media

+
+
 light_media (css:str)
+
+

Render light media for day mode views

+ + + + + + + + + + + + + + + +
TypeDetails
cssstrCSS to be included in the light media query
+
+
light_media('.body {color: green;}')
+
+
<style>@media (prefers-color-scheme: light) {.body {color: green;}}</style>
+
+
+
+

source

+
+
+

dark_media

+
+
 dark_media (css:str)
+
+

Render dark media for night mode views

+ + + + + + + + + + + + + + + +
TypeDetails
cssstrCSS to be included in the dark media query
+
+
dark_media('.body {color: white;}')
+
+
<style>@media (prefers-color-scheme:  dark) {.body {color: white;}}</style>
+
+
+
+

source

+
+
+

MarkdownJS

+
+
 MarkdownJS (sel='.marked')
+
+

Implements browser-based markdown rendering.

+ + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstr.markedCSS selector for markdown elements
+

Usage example here.

+
+
__file__ = '../../fasthtml/katex.js'
+
+
+

source

+
+
+

KatexMarkdownJS

+
+
 KatexMarkdownJS (sel='.marked', inline_delim='$', display_delim='$$',
+                  math_envs=None)
+
+ ++++++ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstr.markedCSS selector for markdown elements
inline_delimstr$Delimiter for inline math
display_delimstr$$Delimiter for long math
math_envsNoneTypeNoneList of environments to render as display math
+

KatexMarkdown usage example:

+
longexample = r"""
+Long example:
+
+$$\begin{array}{c}
+
+\nabla \times \vec{\mathbf{B}} -\, \frac1c\, \frac{\partial\vec{\mathbf{E}}}{\partial t} &
+= \frac{4\pi}{c}\vec{\mathbf{j}}    \nabla \cdot \vec{\mathbf{E}} & = 4 \pi \rho \\
+
+\nabla \times \vec{\mathbf{E}}\, +\, \frac1c\, \frac{\partial\vec{\mathbf{B}}}{\partial t} & = \vec{\mathbf{0}} \\
+
+\nabla \cdot \vec{\mathbf{B}} & = 0
+
+\end{array}$$
+"""
+
+app, rt = fast_app(hdrs=[KatexMarkdownJS()])
+
+@rt('/')
+def get():
+    return Titled("Katex Examples", 
+        # Assigning 'marked' class to components renders content as markdown
+        P(cls='marked')("Inline example: $\sqrt{3x-1}+(1+x)^2$"),
+        Div(cls='marked')(longexample)
+    )
+
+

source

+
+
+

HighlightJS

+
+
 HighlightJS (sel='pre code:not([data-highlighted="yes"])',
+              langs:str|list|tuple='python', light='atom-one-light',
+              dark='atom-one-dark')
+
+

Implements browser-based syntax highlighting. Usage example here.

+ ++++++ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstrpre code:not([data-highlighted=“yes”])CSS selector for code elements. Default is industry standard, be careful before adjusting it
langsstr | list | tuplepythonLanguage(s) to highlight
lightstratom-one-lightLight theme
darkstratom-one-darkDark theme
+
+

source

+
+
+

SortableJS

+
+
 SortableJS (sel='.sortable', ghost_class='blue-background-class')
+
+ ++++++ + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstr.sortableCSS selector for sortable elements
ghost_classstrblue-background-classWhen an element is being dragged, this is the class used to distinguish it from the rest
+
+

source

+
+
+

MermaidJS

+
+
 MermaidJS (sel='.language-mermaid', theme='base')
+
+

Implements browser-based Mermaid diagram rendering.

+ + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstr.language-mermaidCSS selector for mermaid elements
themestrbaseMermaid theme to use
+
app, rt = fast_app(hdrs=[MermaidJS()])
+@rt('/')
+def get():
+    return Titled("Mermaid Examples", 
+        # Assigning 'marked' class to components renders content as markdown
+        Pre(Code(cls ="language-mermaid")('''flowchart TD
+            A[main] --> B["fact(5)"] --> C["fact(4)"] --> D["fact(3)"] --> E["fact(2)"] --> F["fact(1)"] --> G["fact(0)"]
+           ''')))
+

In a markdown file, just like a code cell you can define

+

```mermaid

+
    graph TD
+    A --> B 
+    B --> C 
+    C --> E
+

```

+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/js.html.md b/api/js.html.md new file mode 100644 index 00000000..021e51b4 --- /dev/null +++ b/api/js.html.md @@ -0,0 +1,365 @@ +# Javascript examples + + + + +To expedite fast development, FastHTML comes with several built-in +Javascript and formatting components. These are largely provided to +demonstrate FastHTML JS patterns. There’s far too many JS libs for +FastHTML to wrap them all, and as shown here the code to add FastHTML +support is very simple anyway. + +------------------------------------------------------------------------ + +source + +### light_media + +> light_media (css:str) + +*Render light media for day mode views* + + + + + + + + + + + + + + + + +
TypeDetails
cssstrCSS to be included in the light media query
+ +``` python +light_media('.body {color: green;}') +``` + +``` html + +``` + +------------------------------------------------------------------------ + +source + +### dark_media + +> dark_media (css:str) + +*Render dark media for night mode views* + + + + + + + + + + + + + + + + +
TypeDetails
cssstrCSS to be included in the dark media query
+ +``` python +dark_media('.body {color: white;}') +``` + +``` html + +``` + +------------------------------------------------------------------------ + +source + +### MarkdownJS + +> MarkdownJS (sel='.marked') + +*Implements browser-based markdown rendering.* + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstr.markedCSS selector for markdown elements
+ +Usage example +[here](../tutorials/quickstart_for_web_devs.html#rendering-markdown). + +``` python +__file__ = '../../fasthtml/katex.js' +``` + +------------------------------------------------------------------------ + +source + +### KatexMarkdownJS + +> KatexMarkdownJS (sel='.marked', inline_delim='$', display_delim='$$', +> math_envs=None) + + ++++++ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstr.markedCSS selector for markdown elements
inline_delimstr$Delimiter for inline math
display_delimstr$$Delimiter for long math
math_envsNoneTypeNoneList of environments to render as display math
+ +KatexMarkdown usage example: + +``` python +longexample = r""" +Long example: + +$$\begin{array}{c} + +\nabla \times \vec{\mathbf{B}} -\, \frac1c\, \frac{\partial\vec{\mathbf{E}}}{\partial t} & += \frac{4\pi}{c}\vec{\mathbf{j}} \nabla \cdot \vec{\mathbf{E}} & = 4 \pi \rho \\ + +\nabla \times \vec{\mathbf{E}}\, +\, \frac1c\, \frac{\partial\vec{\mathbf{B}}}{\partial t} & = \vec{\mathbf{0}} \\ + +\nabla \cdot \vec{\mathbf{B}} & = 0 + +\end{array}$$ +""" + +app, rt = fast_app(hdrs=[KatexMarkdownJS()]) + +@rt('/') +def get(): + return Titled("Katex Examples", + # Assigning 'marked' class to components renders content as markdown + P(cls='marked')("Inline example: $\sqrt{3x-1}+(1+x)^2$"), + Div(cls='marked')(longexample) + ) +``` + +------------------------------------------------------------------------ + +source + +### HighlightJS + +> HighlightJS (sel='pre code:not([data-highlighted="yes"])', +> langs:str|list|tuple='python', light='atom-one-light', +> dark='atom-one-dark') + +*Implements browser-based syntax highlighting. Usage example +[here](../tutorials/quickstart_for_web_devs.html#code-highlighting).* + + ++++++ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstrpre code:not([data-highlighted=“yes”])CSS selector for code elements. Default is industry standard, be +careful before adjusting it
langsstr | list | tuplepythonLanguage(s) to highlight
lightstratom-one-lightLight theme
darkstratom-one-darkDark theme
+ +------------------------------------------------------------------------ + +source + +### SortableJS + +> SortableJS (sel='.sortable', ghost_class='blue-background-class') + + ++++++ + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstr.sortableCSS selector for sortable elements
ghost_classstrblue-background-classWhen an element is being dragged, this is the class used to +distinguish it from the rest
+ +------------------------------------------------------------------------ + +source + +### MermaidJS + +> MermaidJS (sel='.language-mermaid', theme='base') + +*Implements browser-based Mermaid diagram rendering.* + + + + + + + + + + + + + + + + + + + + + + + + +
TypeDefaultDetails
selstr.language-mermaidCSS selector for mermaid elements
themestrbaseMermaid theme to use
+ +``` python +app, rt = fast_app(hdrs=[MermaidJS()]) +@rt('/') +def get(): + return Titled("Mermaid Examples", + # Assigning 'marked' class to components renders content as markdown + Pre(Code(cls ="language-mermaid")('''flowchart TD + A[main] --> B["fact(5)"] --> C["fact(4)"] --> D["fact(3)"] --> E["fact(2)"] --> F["fact(1)"] --> G["fact(0)"] + '''))) +``` + +In a markdown file, just like a code cell you can define + +\`\`\`mermaid + + graph TD + A --> B + B --> C + C --> E + +\`\`\` diff --git a/api/jupyter.html b/api/jupyter.html new file mode 100644 index 00000000..c6a0a506 --- /dev/null +++ b/api/jupyter.html @@ -0,0 +1,1075 @@ + + + + + + + + + + +Jupyter compatibility – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Jupyter compatibility

+
+ +
+
+ Use FastHTML in Jupyter notebooks +
+
+ + +
+ + + + +
+ + + +
+ + + +
+
from httpx import get, AsyncClient
+
+
+

Helper functions

+
+

source

+
+

nb_serve

+
+
 nb_serve (app, log_level='error', port=8000, host='0.0.0.0', **kwargs)
+
+

Start a Jupyter compatible uvicorn server with ASGI app on port with log_level

+
+

source

+
+
+

nb_serve_async

+
+
 nb_serve_async (app, log_level='error', port=8000, host='0.0.0.0',
+                 **kwargs)
+
+

Async version of nb_serve

+
+

source

+
+
+

is_port_free

+
+
 is_port_free (port, host='localhost')
+
+

Check if port is free on host

+
+

source

+
+
+

wait_port_free

+
+
 wait_port_free (port, host='localhost', max_wait=3)
+
+

Wait for port to be free on host

+
+
+
+

Using FastHTML in Jupyter

+
+

source

+
+

show

+
+
 show (*s)
+
+

Same as fasthtml.components.show, but also adds htmx.process()

+
+

source

+
+
+

render_ft

+
+
 render_ft ()
+
+
+

source

+
+
+

htmx_config_port

+
+
 htmx_config_port (port=8000)
+
+
+

source

+
+
+

JupyUvi

+
+
 JupyUvi (app, log_level='error', host='0.0.0.0', port=8000, start=True,
+          **kwargs)
+
+

Start and stop a Jupyter compatible uvicorn server with ASGI app on port with log_level

+

Creating an object of this class also starts the Uvicorn server. It runs in a separate thread, so you can use normal HTTP client functions in a notebook.

+
+
app = FastHTML()
+rt = app.route
+
+@app.route
+def index(): return 'hi'
+
+port = 8000
+server = JupyUvi(app, port=port)
+
+ + +
+
+
+
get(f'http://localhost:{port}').text
+
+
'hi'
+
+
+

You can stop the server, modify routes, and start the server again without restarting the notebook or recreating the server or application.

+
+
+

Using a notebook as a web app

+

You can also run an HTMX web app directly in a notebook. To make this work, you have to add the default FastHTML headers to the DOM of the notebook with show(*def_hdrs()). Additionally, you might find it convenient to use auto_id mode, in which the ID of an FT object is automatically generated if not provided.

+
+
fh_cfg['auto_id' ]=True
+
+

After importing fasthtml.jupyter and calling render_ft(), FT components render directly in the notebook.

+
+
(c := Div('Cogito ergo sum'))
+
+
+
+Cogito ergo sum +
+ +
+
+
+

Handlers are written just like a regular web app:

+
+
@rt
+def hoho(): return P('loaded!'), Div('hee hee', id=c, hx_swap_oob='true')
+
+

All the usual hx_* attributes can be used:

+
+
P('not loaded', hx_get=hoho, hx_trigger='load')
+
+
+

+not loaded +

+ +
+
+
+

FT components can be used directly both as id values and as hx_target values.

+
+
(c := Div(''))
+
+
+
+ +
+ +
+
+
+
+
@rt
+def foo(): return Div('foo bar')
+P('hi', hx_get=foo, hx_trigger='load', hx_target=c)
+
+
+

+hi +

+ +
+
+
+
+
server.stop()
+
+
+
+

Running apps in an IFrame

+

Using an IFrame can be a good idea to get complete isolation of the styles and scripts in an app. The HTMX function creates an auto-sizing IFrame for a web app.

+
+

source

+
+
+

HTMX

+
+
 HTMX (path='', app=None, host='localhost', port=8000, height='auto',
+       link=False, iframe=True)
+
+

An iframe which displays the HTMX application in a notebook.

+
+
@rt
+def index():
+    return Div(
+        P(A('Click me', hx_get=update, hx_target='#result')),
+        P(A('No me!', hx_get=update, hx_target='#result')),
+        Div(id='result'))
+
+@rt
+def update(): return Div(P('Hi!'),P('There!'))
+
+
+
server.start()
+
+
+
# Run the notebook locally to see the HTMX iframe in action
+HTMX()
+
+ +
+
+
+
server.stop()
+
+
+

source

+
+
+

ws_client

+
+
 ws_client (app, nm='', host='localhost', port=8000, ws_connect='/ws',
+            frame=True, link=True, **kwargs)
+
+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/jupyter.html.md b/api/jupyter.html.md new file mode 100644 index 00000000..d66b0e5b --- /dev/null +++ b/api/jupyter.html.md @@ -0,0 +1,299 @@ +# Jupyter compatibility + + + + +``` python +from httpx import get, AsyncClient +``` + +## Helper functions + +------------------------------------------------------------------------ + +source + +### nb_serve + +> nb_serve (app, log_level='error', port=8000, host='0.0.0.0', **kwargs) + +*Start a Jupyter compatible uvicorn server with ASGI `app` on `port` +with `log_level`* + +------------------------------------------------------------------------ + +source + +### nb_serve_async + +> nb_serve_async (app, log_level='error', port=8000, host='0.0.0.0', +> **kwargs) + +*Async version of +[`nb_serve`](https://AnswerDotAI.github.io/fasthtml/api/jupyter.html#nb_serve)* + +------------------------------------------------------------------------ + +source + +### is_port_free + +> is_port_free (port, host='localhost') + +*Check if `port` is free on `host`* + +------------------------------------------------------------------------ + +source + +### wait_port_free + +> wait_port_free (port, host='localhost', max_wait=3) + +*Wait for `port` to be free on `host`* + +## Using FastHTML in Jupyter + +------------------------------------------------------------------------ + +source + +### show + +> show (*s) + +*Same as fasthtml.components.show, but also adds `htmx.process()`* + +------------------------------------------------------------------------ + +source + +### render_ft + +> render_ft () + +------------------------------------------------------------------------ + +source + +### htmx_config_port + +> htmx_config_port (port=8000) + +------------------------------------------------------------------------ + +source + +### JupyUvi + +> JupyUvi (app, log_level='error', host='0.0.0.0', port=8000, start=True, +> **kwargs) + +*Start and stop a Jupyter compatible uvicorn server with ASGI `app` on +`port` with `log_level`* + +Creating an object of this class also starts the Uvicorn server. It runs +in a separate thread, so you can use normal HTTP client functions in a +notebook. + +``` python +app = FastHTML() +rt = app.route + +@app.route +def index(): return 'hi' + +port = 8000 +server = JupyUvi(app, port=port) +``` + + + +``` python +get(f'http://localhost:{port}').text +``` + + 'hi' + +You can stop the server, modify routes, and start the server again +without restarting the notebook or recreating the server or application. + +### Using a notebook as a web app + +You can also run an HTMX web app directly in a notebook. To make this +work, you have to add the default FastHTML headers to the DOM of the +notebook with `show(*def_hdrs())`. Additionally, you might find it +convenient to use *auto_id* mode, in which the ID of an `FT` object is +automatically generated if not provided. + +``` python +fh_cfg['auto_id' ]=True +``` + +After importing `fasthtml.jupyter` and calling +[`render_ft()`](https://AnswerDotAI.github.io/fasthtml/api/jupyter.html#render_ft), +FT components render directly in the notebook. + +``` python +(c := Div('Cogito ergo sum')) +``` + +
+ +
+ +Cogito ergo sum + +
+ + + +
+ +Handlers are written just like a regular web app: + +``` python +@rt +def hoho(): return P('loaded!'), Div('hee hee', id=c, hx_swap_oob='true') +``` + +All the usual `hx_*` attributes can be used: + +``` python +P('not loaded', hx_get=hoho, hx_trigger='load') +``` + +
+ +

+ +not loaded +

+ + + +
+ +FT components can be used directly both as `id` values and as +`hx_target` values. + +``` python +(c := Div('')) +``` + +
+ +
+ +
+ + + +
+ +``` python +@rt +def foo(): return Div('foo bar') +P('hi', hx_get=foo, hx_trigger='load', hx_target=c) +``` + +
+ +

+ +hi +

+ + + +
+ +``` python +server.stop() +``` + +### Running apps in an IFrame + +Using an IFrame can be a good idea to get complete isolation of the +styles and scripts in an app. The +[`HTMX`](https://AnswerDotAI.github.io/fasthtml/api/jupyter.html#htmx) +function creates an auto-sizing IFrame for a web app. + +------------------------------------------------------------------------ + +source + +### HTMX + +> HTMX (path='', app=None, host='localhost', port=8000, height='auto', +> link=False, iframe=True) + +*An iframe which displays the HTMX application in a notebook.* + +``` python +@rt +def index(): + return Div( + P(A('Click me', hx_get=update, hx_target='#result')), + P(A('No me!', hx_get=update, hx_target='#result')), + Div(id='result')) + +@rt +def update(): return Div(P('Hi!'),P('There!')) +``` + +``` python +server.start() +``` + +``` python +# Run the notebook locally to see the HTMX iframe in action +HTMX() +``` + + + +``` python +server.stop() +``` + +------------------------------------------------------------------------ + +source + +### ws_client + +> ws_client (app, nm='', host='localhost', port=8000, ws_connect='/ws', +> frame=True, link=True, **kwargs) diff --git a/api/oauth.html b/api/oauth.html new file mode 100644 index 00000000..186cd7c2 --- /dev/null +++ b/api/oauth.html @@ -0,0 +1,1024 @@ + + + + + + + + + + +OAuth – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

OAuth

+
+ +
+
+ Basic scaffolding for handling OAuth +
+
+ + +
+ + + + +
+ + + +
+ + + +

See the docs page for an explanation of how to use this.

+
+
from IPython.display import Markdown
+
+
+

source

+
+

GoogleAppClient

+
+
 GoogleAppClient (client_id, client_secret, code=None, scope=None,
+                  **kwargs)
+
+

A WebApplicationClient for Google oauth2

+
+

source

+
+
+

GitHubAppClient

+
+
 GitHubAppClient (client_id, client_secret, code=None, scope=None,
+                  **kwargs)
+
+

A WebApplicationClient for GitHub oauth2

+
+

source

+
+
+

HuggingFaceClient

+
+
 HuggingFaceClient (client_id, client_secret, code=None, scope=None,
+                    state=None, **kwargs)
+
+

A WebApplicationClient for HuggingFace oauth2

+
+

source

+
+
+

DiscordAppClient

+
+
 DiscordAppClient (client_id, client_secret, is_user=False, perms=0,
+                   scope=None, **kwargs)
+
+

A WebApplicationClient for Discord oauth2

+
+

source

+
+
+

Auth0AppClient

+
+
 Auth0AppClient (domain, client_id, client_secret, code=None, scope=None,
+                 redirect_uri='', **kwargs)
+
+

A WebApplicationClient for Auth0 OAuth2

+
+
# cli = GoogleAppClient.from_file('/Users/jhoward/subs_aai/_nbs/oauth-test/client_secret.json')
+
+
+

source

+
+ +
+

redir_url

+
+
 redir_url (request, redir_path, scheme=None)
+
+

Get the redir url for the host in request

+
+
@rt
+def index(request):
+    redir = redir_url(request, redir_path)
+    return A('login', href=cli.login_link(redir), target='_blank')
+
+
+

source

+
+
+

_AppClient.parse_response

+
+
 _AppClient.parse_response (code, redirect_uri)
+
+

Get the token from the oauth2 server response

+
+

source

+
+
+

_AppClient.get_info

+
+
 _AppClient.get_info (token=None)
+
+

Get the info for authenticated user

+
+

source

+
+
+

_AppClient.retr_info

+
+
 _AppClient.retr_info (code, redirect_uri)
+
+

Combines parse_response and get_info

+
+
@rt(redir_path)
+def get(request, code:str):
+    redir = redir_url(request, redir_path)
+    info = cli.retr_info(code, redir)
+    return P(f'Login successful for {info["name"]}!')
+
+
+
# HTMX()
+
+
+
server.stop()
+
+
+

source

+
+
+

_AppClient.retr_id

+
+
 _AppClient.retr_id (code, redirect_uri)
+
+

Call retr_info and then return id/subscriber value

+

After logging in via the provider, the user will be redirected back to the supplied redirect URL. The request to this URL will contain a code parameter, which is used to get an access token and fetch the user’s profile information. See the explanation here for a worked example. You can either:

+
    +
  • Use client.retr_info(code) to get all the profile information, or
  • +
  • Use client.retr_id(code) to get just the user’s ID.
  • +
+

After either of these calls, you can also access the access token (used to revoke access, for example) with client.token["access_token"].

+
+

source

+
+
+

url_match

+
+
 url_match (url, patterns=('^(localhost|127\\.0\\.0\\.1)(:\\d+)?$',))
+
+
+

source

+
+
+

OAuth

+
+
 OAuth (app, cli, skip=None, redir_path='/redirect', error_path='/error',
+        logout_path='/logout', login_path='/login', https=True,
+        http_patterns=('^(localhost|127\\.0\\.0\\.1)(:\\d+)?$',))
+
+

Initialize self. See help(type(self)) for accurate signature.

+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/oauth.html.md b/api/oauth.html.md new file mode 100644 index 00000000..4af36da1 --- /dev/null +++ b/api/oauth.html.md @@ -0,0 +1,243 @@ +# OAuth + + + + +See the [docs page](https://docs.fastht.ml/explains/oauth.html) for an +explanation of how to use this. + +``` python +from IPython.display import Markdown +``` + +------------------------------------------------------------------------ + +source + +### GoogleAppClient + +> GoogleAppClient (client_id, client_secret, code=None, scope=None, +> **kwargs) + +*A `WebApplicationClient` for Google oauth2* + +------------------------------------------------------------------------ + +source + +### GitHubAppClient + +> GitHubAppClient (client_id, client_secret, code=None, scope=None, +> **kwargs) + +*A `WebApplicationClient` for GitHub oauth2* + +------------------------------------------------------------------------ + +source + +### HuggingFaceClient + +> HuggingFaceClient (client_id, client_secret, code=None, scope=None, +> state=None, **kwargs) + +*A `WebApplicationClient` for HuggingFace oauth2* + +------------------------------------------------------------------------ + +source + +### DiscordAppClient + +> DiscordAppClient (client_id, client_secret, is_user=False, perms=0, +> scope=None, **kwargs) + +*A `WebApplicationClient` for Discord oauth2* + +------------------------------------------------------------------------ + +source + +### Auth0AppClient + +> Auth0AppClient (domain, client_id, client_secret, code=None, scope=None, +> redirect_uri='', **kwargs) + +*A `WebApplicationClient` for Auth0 OAuth2* + +``` python +# cli = GoogleAppClient.from_file('/Users/jhoward/subs_aai/_nbs/oauth-test/client_secret.json') +``` + +------------------------------------------------------------------------ + +source + +### WebApplicationClient.login_link + +> WebApplicationClient.login_link (redirect_uri, scope=None, state=None) + +*Get a login link for this client* + +Generating a login link that sends the user to the OAuth provider is +done with `client.login_link()`. + +It can sometimes be useful to pass state to the OAuth provider, so that +when the user returns you can pick up where they left off. This can be +done by passing the `state` parameter. + +``` python +from fasthtml.common import * +from fasthtml.jupyter import * +``` + +``` python +redir_path = '/redirect' +port = 8000 +code_stor = None +``` + +``` python +app,rt = fast_app() +server = JupyUvi(app, port=port) +``` + + + +------------------------------------------------------------------------ + +source + +### redir_url + +> redir_url (request, redir_path, scheme=None) + +*Get the redir url for the host in `request`* + +``` python +@rt +def index(request): + redir = redir_url(request, redir_path) + return A('login', href=cli.login_link(redir), target='_blank') +``` + +------------------------------------------------------------------------ + +source + +### \_AppClient.parse_response + +> _AppClient.parse_response (code, redirect_uri) + +*Get the token from the oauth2 server response* + +------------------------------------------------------------------------ + +source + +### \_AppClient.get_info + +> _AppClient.get_info (token=None) + +*Get the info for authenticated user* + +------------------------------------------------------------------------ + +source + +### \_AppClient.retr_info + +> _AppClient.retr_info (code, redirect_uri) + +*Combines `parse_response` and `get_info`* + +``` python +@rt(redir_path) +def get(request, code:str): + redir = redir_url(request, redir_path) + info = cli.retr_info(code, redir) + return P(f'Login successful for {info["name"]}!') +``` + +``` python +# HTMX() +``` + +``` python +server.stop() +``` + +------------------------------------------------------------------------ + +source + +### \_AppClient.retr_id + +> _AppClient.retr_id (code, redirect_uri) + +*Call `retr_info` and then return id/subscriber value* + +After logging in via the provider, the user will be redirected back to +the supplied redirect URL. The request to this URL will contain a `code` +parameter, which is used to get an access token and fetch the user’s +profile information. See [the explanation +here](https://docs.fastht.ml/explains/oauth.html) for a worked example. +You can either: + +- Use client.retr_info(code) to get all the profile information, or +- Use client.retr_id(code) to get just the user’s ID. + +After either of these calls, you can also access the access token (used +to revoke access, for example) with `client.token["access_token"]`. + +------------------------------------------------------------------------ + +source + +### url_match + +> url_match (url, patterns=('^(localhost|127\\.0\\.0\\.1)(:\\d+)?$',)) + +------------------------------------------------------------------------ + +source + +### OAuth + +> OAuth (app, cli, skip=None, redir_path='/redirect', error_path='/error', +> logout_path='/logout', login_path='/login', https=True, +> http_patterns=('^(localhost|127\\.0\\.0\\.1)(:\\d+)?$',)) + +*Initialize self. See help(type(self)) for accurate signature.* diff --git a/api/pico.html b/api/pico.html new file mode 100644 index 00000000..97102366 --- /dev/null +++ b/api/pico.html @@ -0,0 +1,1066 @@ + + + + + + + + + + +Pico.css components – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Pico.css components

+
+ +
+
+ Basic components for generating Pico CSS tags +
+
+ + +
+ + + + +
+ + + +
+ + + +

picocondlink is the class-conditional css link tag, and picolink is the regular tag.

+
+
show(picocondlink)
+
+ + +
+
+
+

source

+
+

set_pico_cls

+
+
 set_pico_cls ()
+
+

Run this to make jupyter outputs styled with pico:

+
+
set_pico_cls()
+
+ +
+
+
+

source

+
+
+

Card

+
+
 Card (*c, header=None, footer=None, target_id=None, hx_vals=None,
+       hx_target=None, id=None, cls=None, title=None, style=None,
+       accesskey=None, contenteditable=None, dir=None, draggable=None,
+       enterkeyhint=None, hidden=None, inert=None, inputmode=None,
+       lang=None, popover=None, spellcheck=None, tabindex=None,
+       translate=None, hx_get=None, hx_post=None, hx_put=None,
+       hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,
+       hx_swap_oob=None, hx_include=None, hx_select=None,
+       hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+       hx_confirm=None, hx_disable=None, hx_replace_url=None,
+       hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+       hx_history=None, hx_history_elt=None, hx_inherit=None,
+       hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,
+       hx_sync=None, hx_validate=None, **kwargs)
+
+

A PicoCSS Card, implemented as an Article with optional Header and Footer

+
+
show(Card('body', header=P('head'), footer=P('foot')))
+
+
+

head

+
+body +

foot

+
+
+
+
+
+

source

+
+
+

Group

+
+
 Group (*c, target_id=None, hx_vals=None, hx_target=None, id=None,
+        cls=None, title=None, style=None, accesskey=None,
+        contenteditable=None, dir=None, draggable=None, enterkeyhint=None,
+        hidden=None, inert=None, inputmode=None, lang=None, popover=None,
+        spellcheck=None, tabindex=None, translate=None, hx_get=None,
+        hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,
+        hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None,
+        hx_select=None, hx_select_oob=None, hx_indicator=None,
+        hx_push_url=None, hx_confirm=None, hx_disable=None,
+        hx_replace_url=None, hx_disabled_elt=None, hx_ext=None,
+        hx_headers=None, hx_history=None, hx_history_elt=None,
+        hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None,
+        hx_request=None, hx_sync=None, hx_validate=None, **kwargs)
+
+

A PicoCSS Group, implemented as a Fieldset with role ‘group’

+
+
show(Group(Input(), Button("Save")))
+
+
+ + +
+
+
+
+

source

+
+ +
+

Grid

+
+
 Grid (*c, cls='grid', target_id=None, hx_vals=None, hx_target=None,
+       id=None, title=None, style=None, accesskey=None,
+       contenteditable=None, dir=None, draggable=None, enterkeyhint=None,
+       hidden=None, inert=None, inputmode=None, lang=None, popover=None,
+       spellcheck=None, tabindex=None, translate=None, hx_get=None,
+       hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,
+       hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None,
+       hx_select=None, hx_select_oob=None, hx_indicator=None,
+       hx_push_url=None, hx_confirm=None, hx_disable=None,
+       hx_replace_url=None, hx_disabled_elt=None, hx_ext=None,
+       hx_headers=None, hx_history=None, hx_history_elt=None,
+       hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None,
+       hx_request=None, hx_sync=None, hx_validate=None, **kwargs)
+
+

A PicoCSS Grid, implemented as child Divs in a Div with class ‘grid’

+
+
colors = [Input(type="color", value=o) for o in ('#e66465', '#53d2c5', '#f6b73c')]
+show(Grid(*colors))
+
+
+
+
+
+
+
+
+
+
+
+
+

source

+
+
+

DialogX

+
+
 DialogX (*c, open=None, header=None, footer=None, id=None,
+          target_id=None, hx_vals=None, hx_target=None, cls=None,
+          title=None, style=None, accesskey=None, contenteditable=None,
+          dir=None, draggable=None, enterkeyhint=None, hidden=None,
+          inert=None, inputmode=None, lang=None, popover=None,
+          spellcheck=None, tabindex=None, translate=None, hx_get=None,
+          hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,
+          hx_trigger=None, hx_swap=None, hx_swap_oob=None,
+          hx_include=None, hx_select=None, hx_select_oob=None,
+          hx_indicator=None, hx_push_url=None, hx_confirm=None,
+          hx_disable=None, hx_replace_url=None, hx_disabled_elt=None,
+          hx_ext=None, hx_headers=None, hx_history=None,
+          hx_history_elt=None, hx_inherit=None, hx_params=None,
+          hx_preserve=None, hx_prompt=None, hx_request=None, hx_sync=None,
+          hx_validate=None, **kwargs)
+
+

A PicoCSS Dialog, with children inside a Card

+
+
hdr = Div(Button(aria_label="Close", rel="prev"), P('confirm'))
+ftr = Div(Button('Cancel', cls="secondary"), Button('Confirm'))
+d = DialogX('thank you!', header=hdr, footer=ftr, open=None, id='dlgtest')
+# use js or htmx to display modal
+
+
+

source

+
+
+

Container

+
+
 Container (*args, target_id=None, hx_vals=None, hx_target=None, id=None,
+            cls=None, title=None, style=None, accesskey=None,
+            contenteditable=None, dir=None, draggable=None,
+            enterkeyhint=None, hidden=None, inert=None, inputmode=None,
+            lang=None, popover=None, spellcheck=None, tabindex=None,
+            translate=None, hx_get=None, hx_post=None, hx_put=None,
+            hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,
+            hx_swap_oob=None, hx_include=None, hx_select=None,
+            hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+            hx_confirm=None, hx_disable=None, hx_replace_url=None,
+            hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+            hx_history=None, hx_history_elt=None, hx_inherit=None,
+            hx_params=None, hx_preserve=None, hx_prompt=None,
+            hx_request=None, hx_sync=None, hx_validate=None, **kwargs)
+
+

A PicoCSS Container, implemented as a Main with class ‘container’

+
+

source

+
+
+

PicoBusy

+
+
 PicoBusy ()
+
+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/pico.html.md b/api/pico.html.md new file mode 100644 index 00000000..dcf80ab0 --- /dev/null +++ b/api/pico.html.md @@ -0,0 +1,243 @@ +# Pico.css components + + + + +`picocondlink` is the class-conditional css `link` tag, and `picolink` +is the regular tag. + +``` python +show(picocondlink) +``` + + + + +------------------------------------------------------------------------ + +source + +### set_pico_cls + +> set_pico_cls () + +Run this to make jupyter outputs styled with pico: + +``` python +set_pico_cls() +``` + + + +------------------------------------------------------------------------ + +source + +### Card + +> Card (*c, header=None, footer=None, target_id=None, hx_vals=None, +> hx_target=None, id=None, cls=None, title=None, style=None, +> accesskey=None, contenteditable=None, dir=None, draggable=None, +> enterkeyhint=None, hidden=None, inert=None, inputmode=None, +> lang=None, popover=None, spellcheck=None, tabindex=None, +> translate=None, hx_get=None, hx_post=None, hx_put=None, +> hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None, +> hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None, +> hx_sync=None, hx_validate=None, **kwargs) + +*A PicoCSS Card, implemented as an Article with optional Header and +Footer* + +``` python +show(Card('body', header=P('head'), footer=P('foot'))) +``` + +
+

head

+
+body +

foot

+
+
+ +------------------------------------------------------------------------ + +source + +### Group + +> Group (*c, target_id=None, hx_vals=None, hx_target=None, id=None, +> cls=None, title=None, style=None, accesskey=None, +> contenteditable=None, dir=None, draggable=None, enterkeyhint=None, +> hidden=None, inert=None, inputmode=None, lang=None, popover=None, +> spellcheck=None, tabindex=None, translate=None, hx_get=None, +> hx_post=None, hx_put=None, hx_delete=None, hx_patch=None, +> hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None, +> hx_select=None, hx_select_oob=None, hx_indicator=None, +> hx_push_url=None, hx_confirm=None, hx_disable=None, +> hx_replace_url=None, hx_disabled_elt=None, hx_ext=None, +> hx_headers=None, hx_history=None, hx_history_elt=None, +> hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None, **kwargs) + +*A PicoCSS Group, implemented as a Fieldset with role ‘group’* + +``` python +show(Group(Input(), Button("Save"))) +``` + +
+ + +
+ +------------------------------------------------------------------------ + +source + +### Search + +> Search (*c, target_id=None, hx_vals=None, hx_target=None, id=None, +> cls=None, title=None, style=None, accesskey=None, +> contenteditable=None, dir=None, draggable=None, +> enterkeyhint=None, hidden=None, inert=None, inputmode=None, +> lang=None, popover=None, spellcheck=None, tabindex=None, +> translate=None, hx_get=None, hx_post=None, hx_put=None, +> hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None, +> hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None, **kwargs) + +*A PicoCSS Search, implemented as a Form with role ‘search’* + +``` python +show(Search(Input(type="search"), Button("Search"))) +``` + +
+ + +
+ +------------------------------------------------------------------------ + +source + +### Grid + +> Grid (*c, cls='grid', target_id=None, hx_vals=None, hx_target=None, +> id=None, title=None, style=None, accesskey=None, +> contenteditable=None, dir=None, draggable=None, enterkeyhint=None, +> hidden=None, inert=None, inputmode=None, lang=None, popover=None, +> spellcheck=None, tabindex=None, translate=None, hx_get=None, +> hx_post=None, hx_put=None, hx_delete=None, hx_patch=None, +> hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None, +> hx_select=None, hx_select_oob=None, hx_indicator=None, +> hx_push_url=None, hx_confirm=None, hx_disable=None, +> hx_replace_url=None, hx_disabled_elt=None, hx_ext=None, +> hx_headers=None, hx_history=None, hx_history_elt=None, +> hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None, **kwargs) + +*A PicoCSS Grid, implemented as child Divs in a Div with class ‘grid’* + +``` python +colors = [Input(type="color", value=o) for o in ('#e66465', '#53d2c5', '#f6b73c')] +show(Grid(*colors)) +``` + +
+
+
+
+
+
+
+
+ +------------------------------------------------------------------------ + +source + +### DialogX + +> DialogX (*c, open=None, header=None, footer=None, id=None, +> target_id=None, hx_vals=None, hx_target=None, cls=None, +> title=None, style=None, accesskey=None, contenteditable=None, +> dir=None, draggable=None, enterkeyhint=None, hidden=None, +> inert=None, inputmode=None, lang=None, popover=None, +> spellcheck=None, tabindex=None, translate=None, hx_get=None, +> hx_post=None, hx_put=None, hx_delete=None, hx_patch=None, +> hx_trigger=None, hx_swap=None, hx_swap_oob=None, +> hx_include=None, hx_select=None, hx_select_oob=None, +> hx_indicator=None, hx_push_url=None, hx_confirm=None, +> hx_disable=None, hx_replace_url=None, hx_disabled_elt=None, +> hx_ext=None, hx_headers=None, hx_history=None, +> hx_history_elt=None, hx_inherit=None, hx_params=None, +> hx_preserve=None, hx_prompt=None, hx_request=None, hx_sync=None, +> hx_validate=None, **kwargs) + +*A PicoCSS Dialog, with children inside a Card* + +``` python +hdr = Div(Button(aria_label="Close", rel="prev"), P('confirm')) +ftr = Div(Button('Cancel', cls="secondary"), Button('Confirm')) +d = DialogX('thank you!', header=hdr, footer=ftr, open=None, id='dlgtest') +# use js or htmx to display modal +``` + +------------------------------------------------------------------------ + +source + +### Container + +> Container (*args, target_id=None, hx_vals=None, hx_target=None, id=None, +> cls=None, title=None, style=None, accesskey=None, +> contenteditable=None, dir=None, draggable=None, +> enterkeyhint=None, hidden=None, inert=None, inputmode=None, +> lang=None, popover=None, spellcheck=None, tabindex=None, +> translate=None, hx_get=None, hx_post=None, hx_put=None, +> hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None, +> hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None, **kwargs) + +*A PicoCSS Container, implemented as a Main with class ‘container’* + +------------------------------------------------------------------------ + +source + +### PicoBusy + +> PicoBusy () diff --git a/api/svg.html b/api/svg.html new file mode 100644 index 00000000..138c6459 --- /dev/null +++ b/api/svg.html @@ -0,0 +1,1380 @@ + + + + + + + + + + +SVG – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

SVG

+
+ +
+
+ Simple SVG FT elements +
+
+ + +
+ + + + +
+ + + +
+ + + +
+
from nbdev.showdoc import show_doc
+
+

You can create SVGs directly from strings, for instance (as always, use NotStr or Safe to tell FastHTML to not escape the text):

+
+
svg = '<svg width="50" height="50"><circle cx="20" cy="20" r="15" fill="red"></circle></svg>'
+show(NotStr(svg))
+
+ +
+
+

You can also use libraries such as fa6-icons.

+

To create and modify SVGs using a Python API, use the FT elements in fasthtml.svg, discussed below.

+

Note: fasthtml.common does NOT automatically export SVG elements. To get access to them, you need to import fasthtml.svg like so

+
from fasthtml.svg import *
+
+

source

+
+

Svg

+
+
 Svg (*args, viewBox=None, h=None, w=None, height=None, width=None,
+      xmlns='http://www.w3.org/2000/svg', **kwargs)
+
+

An SVG tag; xmlns is added automatically, and viewBox defaults to height and width if not provided

+

To create your own SVGs, use SVG. It will automatically set the viewBox from height and width if not provided.

+

All of our shapes will have some convenient kwargs added by using ft_svg:

+
+

source

+
+
+

ft_svg

+
+
 ft_svg (tag:str, *c, transform=None, opacity=None, clip=None, mask=None,
+         filter=None, vector_effect=None, pointer_events=None,
+         target_id=None, hx_vals=None, hx_target=None, id=None, cls=None,
+         title=None, style=None, accesskey=None, contenteditable=None,
+         dir=None, draggable=None, enterkeyhint=None, hidden=None,
+         inert=None, inputmode=None, lang=None, popover=None,
+         spellcheck=None, tabindex=None, translate=None, hx_get=None,
+         hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,
+         hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None,
+         hx_select=None, hx_select_oob=None, hx_indicator=None,
+         hx_push_url=None, hx_confirm=None, hx_disable=None,
+         hx_replace_url=None, hx_disabled_elt=None, hx_ext=None,
+         hx_headers=None, hx_history=None, hx_history_elt=None,
+         hx_inherit=None, hx_params=None, hx_preserve=None,
+         hx_prompt=None, hx_request=None, hx_sync=None, hx_validate=None)
+
+

Create a standard FT element with some SVG-specific attrs

+
+
+

Basic shapes

+

We’ll define a simple function to display SVG shapes in this notebook:

+
+
def demo(el, h=50, w=50): return show(Svg(h=h,w=w)(el))
+
+
+

source

+
+

Rect

+
+
 Rect (width, height, x=0, y=0, fill=None, stroke=None, stroke_width=None,
+       rx=None, ry=None, transform=None, opacity=None, clip=None,
+       mask=None, filter=None, vector_effect=None, pointer_events=None,
+       target_id=None, hx_vals=None, hx_target=None, id=None, cls=None,
+       title=None, style=None, accesskey=None, contenteditable=None,
+       dir=None, draggable=None, enterkeyhint=None, hidden=None,
+       inert=None, inputmode=None, lang=None, popover=None,
+       spellcheck=None, tabindex=None, translate=None, hx_get=None,
+       hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,
+       hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None,
+       hx_select=None, hx_select_oob=None, hx_indicator=None,
+       hx_push_url=None, hx_confirm=None, hx_disable=None,
+       hx_replace_url=None, hx_disabled_elt=None, hx_ext=None,
+       hx_headers=None, hx_history=None, hx_history_elt=None,
+       hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None,
+       hx_request=None, hx_sync=None, hx_validate=None)
+
+

A standard SVG rect element

+

All our shapes just create regular FT elements. The only extra functionality provided by most of them is to add additional defined kwargs to improve auto-complete in IDEs and notebooks, and re-order parameters so that positional args can also be used to save a bit of typing, e.g:

+
+
demo(Rect(30, 30, fill='blue', rx=8, ry=8))
+
+ +
+
+
+

source

+
+
+

Circle

+
+
 Circle (r, cx=0, cy=0, fill=None, stroke=None, stroke_width=None,
+         transform=None, opacity=None, clip=None, mask=None, filter=None,
+         vector_effect=None, pointer_events=None, target_id=None,
+         hx_vals=None, hx_target=None, id=None, cls=None, title=None,
+         style=None, accesskey=None, contenteditable=None, dir=None,
+         draggable=None, enterkeyhint=None, hidden=None, inert=None,
+         inputmode=None, lang=None, popover=None, spellcheck=None,
+         tabindex=None, translate=None, hx_get=None, hx_post=None,
+         hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,
+         hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,
+         hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+         hx_confirm=None, hx_disable=None, hx_replace_url=None,
+         hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+         hx_history=None, hx_history_elt=None, hx_inherit=None,
+         hx_params=None, hx_preserve=None, hx_prompt=None,
+         hx_request=None, hx_sync=None, hx_validate=None)
+
+

A standard SVG circle element

+
+
demo(Circle(20, 25, 25, stroke='red', stroke_width=3))
+
+ +
+
+
+

source

+
+
+

Ellipse

+
+
 Ellipse (rx, ry, cx=0, cy=0, fill=None, stroke=None, stroke_width=None,
+          transform=None, opacity=None, clip=None, mask=None, filter=None,
+          vector_effect=None, pointer_events=None, target_id=None,
+          hx_vals=None, hx_target=None, id=None, cls=None, title=None,
+          style=None, accesskey=None, contenteditable=None, dir=None,
+          draggable=None, enterkeyhint=None, hidden=None, inert=None,
+          inputmode=None, lang=None, popover=None, spellcheck=None,
+          tabindex=None, translate=None, hx_get=None, hx_post=None,
+          hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,
+          hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,
+          hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+          hx_confirm=None, hx_disable=None, hx_replace_url=None,
+          hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+          hx_history=None, hx_history_elt=None, hx_inherit=None,
+          hx_params=None, hx_preserve=None, hx_prompt=None,
+          hx_request=None, hx_sync=None, hx_validate=None)
+
+

A standard SVG ellipse element

+
+
demo(Ellipse(20, 10, 25, 25))
+
+ +
+
+
+

source

+
+
+

transformd

+
+
 transformd (translate=None, scale=None, rotate=None, skewX=None,
+             skewY=None, matrix=None)
+
+

Create an SVG transform kwarg dict

+
+
rot = transformd(rotate=(45, 25, 25))
+rot
+
+
{'transform': 'rotate(45,25,25)'}
+
+
+
+
demo(Ellipse(20, 10, 25, 25, **rot))
+
+ +
+
+
+

source

+
+
+

Line

+
+
 Line (x1, y1, x2=0, y2=0, stroke='black', w=None, stroke_width=1,
+       transform=None, opacity=None, clip=None, mask=None, filter=None,
+       vector_effect=None, pointer_events=None, target_id=None,
+       hx_vals=None, hx_target=None, id=None, cls=None, title=None,
+       style=None, accesskey=None, contenteditable=None, dir=None,
+       draggable=None, enterkeyhint=None, hidden=None, inert=None,
+       inputmode=None, lang=None, popover=None, spellcheck=None,
+       tabindex=None, translate=None, hx_get=None, hx_post=None,
+       hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,
+       hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,
+       hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+       hx_confirm=None, hx_disable=None, hx_replace_url=None,
+       hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+       hx_history=None, hx_history_elt=None, hx_inherit=None,
+       hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,
+       hx_sync=None, hx_validate=None)
+
+

A standard SVG line element

+
+
demo(Line(20, 30, w=3))
+
+ +
+
+
+

source

+
+
+

Polyline

+
+
 Polyline (*args, points=None, fill=None, stroke=None, stroke_width=None,
+           transform=None, opacity=None, clip=None, mask=None,
+           filter=None, vector_effect=None, pointer_events=None,
+           target_id=None, hx_vals=None, hx_target=None, id=None,
+           cls=None, title=None, style=None, accesskey=None,
+           contenteditable=None, dir=None, draggable=None,
+           enterkeyhint=None, hidden=None, inert=None, inputmode=None,
+           lang=None, popover=None, spellcheck=None, tabindex=None,
+           translate=None, hx_get=None, hx_post=None, hx_put=None,
+           hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,
+           hx_swap_oob=None, hx_include=None, hx_select=None,
+           hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+           hx_confirm=None, hx_disable=None, hx_replace_url=None,
+           hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+           hx_history=None, hx_history_elt=None, hx_inherit=None,
+           hx_params=None, hx_preserve=None, hx_prompt=None,
+           hx_request=None, hx_sync=None, hx_validate=None)
+
+

A standard SVG polyline element

+
+
demo(Polyline((0,0), (10,10), (20,0), (30,10), (40,0),
+              fill='yellow', stroke='blue', stroke_width=2))
+
+ +
+
+
+
demo(Polyline(points='0,0 10,10 20,0 30,10 40,0', fill='purple', stroke_width=2))
+
+ +
+
+
+

source

+
+
+

Polygon

+
+
 Polygon (*args, points=None, fill=None, stroke=None, stroke_width=None,
+          transform=None, opacity=None, clip=None, mask=None, filter=None,
+          vector_effect=None, pointer_events=None, target_id=None,
+          hx_vals=None, hx_target=None, id=None, cls=None, title=None,
+          style=None, accesskey=None, contenteditable=None, dir=None,
+          draggable=None, enterkeyhint=None, hidden=None, inert=None,
+          inputmode=None, lang=None, popover=None, spellcheck=None,
+          tabindex=None, translate=None, hx_get=None, hx_post=None,
+          hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,
+          hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,
+          hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+          hx_confirm=None, hx_disable=None, hx_replace_url=None,
+          hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+          hx_history=None, hx_history_elt=None, hx_inherit=None,
+          hx_params=None, hx_preserve=None, hx_prompt=None,
+          hx_request=None, hx_sync=None, hx_validate=None)
+
+

A standard SVG polygon element

+
+
demo(Polygon((25,5), (43.3,15), (43.3,35), (25,45), (6.7,35), (6.7,15), 
+             fill='lightblue', stroke='navy', stroke_width=2))
+
+ +
+
+
+
demo(Polygon(points='25,5 43.3,15 43.3,35 25,45 6.7,35 6.7,15',
+             fill='lightgreen', stroke='darkgreen', stroke_width=2))
+
+ +
+
+
+

source

+
+
+

Text

+
+
 Text (*args, x=0, y=0, font_family=None, font_size=None, fill=None,
+       text_anchor=None, dominant_baseline=None, font_weight=None,
+       font_style=None, text_decoration=None, transform=None,
+       opacity=None, clip=None, mask=None, filter=None,
+       vector_effect=None, pointer_events=None, target_id=None,
+       hx_vals=None, hx_target=None, id=None, cls=None, title=None,
+       style=None, accesskey=None, contenteditable=None, dir=None,
+       draggable=None, enterkeyhint=None, hidden=None, inert=None,
+       inputmode=None, lang=None, popover=None, spellcheck=None,
+       tabindex=None, translate=None, hx_get=None, hx_post=None,
+       hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,
+       hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,
+       hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+       hx_confirm=None, hx_disable=None, hx_replace_url=None,
+       hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+       hx_history=None, hx_history_elt=None, hx_inherit=None,
+       hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,
+       hx_sync=None, hx_validate=None)
+
+

A standard SVG text element

+
+
demo(Text("Hello!", x=10, y=30))
+
+Hello! +
+
+
+
+
+

Paths

+

Paths in SVGs are more complex, so we add a small (optional) fluent interface for constructing them:

+
+

source

+
+

PathFT

+
+
 PathFT (tag:str, cs:tuple, attrs:dict=None, void_=False, **kwargs)
+
+

A ‘Fast Tag’ structure, containing tag,children,and attrs

+
+

source

+
+
+

Path

+
+
 Path (d='', fill=None, stroke=None, stroke_width=None, transform=None,
+       opacity=None, clip=None, mask=None, filter=None,
+       vector_effect=None, pointer_events=None, target_id=None,
+       hx_vals=None, hx_target=None, id=None, cls=None, title=None,
+       style=None, accesskey=None, contenteditable=None, dir=None,
+       draggable=None, enterkeyhint=None, hidden=None, inert=None,
+       inputmode=None, lang=None, popover=None, spellcheck=None,
+       tabindex=None, translate=None, hx_get=None, hx_post=None,
+       hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,
+       hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,
+       hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+       hx_confirm=None, hx_disable=None, hx_replace_url=None,
+       hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+       hx_history=None, hx_history_elt=None, hx_inherit=None,
+       hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,
+       hx_sync=None, hx_validate=None)
+
+

Create a standard path SVG element. This is a special object

+

Let’s create a square shape, but using Path instead of Rect:

+
    +
  • M(10, 10): Move to starting point (10, 10)
  • +
  • L(40, 10): Line to (40, 10) - top edge
  • +
  • L(40, 40): Line to (40, 40) - right edge
  • +
  • L(10, 40): Line to (10, 40) - bottom edge
  • +
  • Z(): Close path - connects back to start
  • +
+

M = Move to, L = Line to, Z = Close path

+
+
demo(Path(fill='none', stroke='purple', stroke_width=2
+         ).M(10, 10).L(40, 10).L(40, 40).L(10, 40).Z())
+
+ +
+
+

Using curves we can create a spiral:

+
+
p = (Path(fill='none', stroke='purple', stroke_width=2)
+     .M(25, 25)
+     .C(25, 25, 20, 20, 30, 20)
+     .C(40, 20, 40, 30, 30, 30)
+     .C(20, 30, 20, 15, 35, 15)
+     .C(50, 15, 50, 35, 25, 35)
+     .C(0, 35, 0, 10, 40, 10)
+     .C(80, 10, 80, 40, 25, 40))
+demo(p, 50, 100)
+
+ +
+
+

Using arcs and curves we can create a map marker icon:

+
+
p = (Path(fill='red')
+     .M(25,45)
+     .C(25,45,10,35,10,25)
+     .A(15,15,0,1,1,40,25)
+     .C(40,35,25,45,25,45)
+     .Z())
+demo(p)
+
+ +
+
+

Behind the scenes it’s just creating regular SVG path d attr – you can pass d in directly if you prefer.

+
+
print(p.d)
+
+
 M25 45 C25 45 10 35 10 25 A15 15 0 1 1 40 25 C40 35 25 45 25 45 Z
+
+
+
+
demo(Path(d='M25 45 C25 45 10 35 10 25 A15 15 0 1 1 40 25 C40 35 25 45 25 45 Z'))
+
+ +
+
+
+

source

+
+
+

PathFT.M

+
+
 PathFT.M (x, y)
+
+

Move to.

+
+

source

+
+
+

PathFT.L

+
+
 PathFT.L (x, y)
+
+

Line to.

+
+

source

+
+
+

PathFT.H

+
+
 PathFT.H (x)
+
+

Horizontal line to.

+
+

source

+
+
+

PathFT.V

+
+
 PathFT.V (y)
+
+

Vertical line to.

+
+

source

+
+
+

PathFT.Z

+
+
 PathFT.Z ()
+
+

Close path.

+
+

source

+
+
+

PathFT.C

+
+
 PathFT.C (x1, y1, x2, y2, x, y)
+
+

Cubic Bézier curve.

+
+

source

+
+
+

PathFT.S

+
+
 PathFT.S (x2, y2, x, y)
+
+

Smooth cubic Bézier curve.

+
+

source

+
+
+

PathFT.Q

+
+
 PathFT.Q (x1, y1, x, y)
+
+

Quadratic Bézier curve.

+
+

source

+
+
+

PathFT.T

+
+
 PathFT.T (x, y)
+
+

Smooth quadratic Bézier curve.

+
+

source

+
+
+

PathFT.A

+
+
 PathFT.A (rx, ry, x_axis_rotation, large_arc_flag, sweep_flag, x, y)
+
+

Elliptical Arc.

+
+
+
+

HTMX helpers

+
+

source

+
+

SvgOob

+
+
 SvgOob (*args, **kwargs)
+
+

Wraps an SVG shape as required for an HTMX OOB swap

+

When returning an SVG shape out-of-band (OOB) in HTMX, you need to wrap it with SvgOob to have it appear correctly. (SvgOob is just a shortcut for Template(Svg(...)), which is the trick that makes SVG OOB swaps work.)

+
+

source

+
+
+

SvgInb

+
+
 SvgInb (*args, **kwargs)
+
+

Wraps an SVG shape as required for an HTMX inband swap

+

When returning an SVG shape in-band in HTMX, either have the calling element include hx_select='svg>*', or **svg_inb (which are two ways of saying the same thing), or wrap the response with SvgInb to have it appear correctly. (SvgInb is just a shortcut for the tuple (Svg(...), HtmxResponseHeaders(hx_reselect='svg>*')), which is the trick that makes SVG in-band swaps work.)

+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/svg.html.md b/api/svg.html.md new file mode 100644 index 00000000..615aea43 --- /dev/null +++ b/api/svg.html.md @@ -0,0 +1,639 @@ +# SVG + + + + +``` python +from nbdev.showdoc import show_doc +``` + +You can create SVGs directly from strings, for instance (as always, use +`NotStr` or `Safe` to tell FastHTML to not escape the text): + +``` python +svg = '' +show(NotStr(svg)) +``` + + + +You can also use libraries such as +[fa6-icons](https://answerdotai.github.io/fa6-icons/). + +To create and modify SVGs using a Python API, use the FT elements in +`fasthtml.svg`, discussed below. + +**Note**: `fasthtml.common` does NOT automatically export SVG elements. +To get access to them, you need to import `fasthtml.svg` like so + +``` python +from fasthtml.svg import * +``` + +------------------------------------------------------------------------ + +source + +### Svg + +> Svg (*args, viewBox=None, h=None, w=None, height=None, width=None, +> xmlns='http://www.w3.org/2000/svg', **kwargs) + +*An SVG tag; xmlns is added automatically, and viewBox defaults to +height and width if not provided* + +To create your own SVGs, use `SVG`. It will automatically set the +`viewBox` from height and width if not provided. + +All of our shapes will have some convenient kwargs added by using +[`ft_svg`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#ft_svg): + +------------------------------------------------------------------------ + +source + +### ft_svg + +> ft_svg (tag:str, *c, transform=None, opacity=None, clip=None, mask=None, +> filter=None, vector_effect=None, pointer_events=None, +> target_id=None, hx_vals=None, hx_target=None, id=None, cls=None, +> title=None, style=None, accesskey=None, contenteditable=None, +> dir=None, draggable=None, enterkeyhint=None, hidden=None, +> inert=None, inputmode=None, lang=None, popover=None, +> spellcheck=None, tabindex=None, translate=None, hx_get=None, +> hx_post=None, hx_put=None, hx_delete=None, hx_patch=None, +> hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None, +> hx_select=None, hx_select_oob=None, hx_indicator=None, +> hx_push_url=None, hx_confirm=None, hx_disable=None, +> hx_replace_url=None, hx_disabled_elt=None, hx_ext=None, +> hx_headers=None, hx_history=None, hx_history_elt=None, +> hx_inherit=None, hx_params=None, hx_preserve=None, +> hx_prompt=None, hx_request=None, hx_sync=None, hx_validate=None) + +*Create a standard `FT` element with some SVG-specific attrs* + +## Basic shapes + +We’ll define a simple function to display SVG shapes in this notebook: + +``` python +def demo(el, h=50, w=50): return show(Svg(h=h,w=w)(el)) +``` + +------------------------------------------------------------------------ + +source + +### Rect + +> Rect (width, height, x=0, y=0, fill=None, stroke=None, stroke_width=None, +> rx=None, ry=None, transform=None, opacity=None, clip=None, +> mask=None, filter=None, vector_effect=None, pointer_events=None, +> target_id=None, hx_vals=None, hx_target=None, id=None, cls=None, +> title=None, style=None, accesskey=None, contenteditable=None, +> dir=None, draggable=None, enterkeyhint=None, hidden=None, +> inert=None, inputmode=None, lang=None, popover=None, +> spellcheck=None, tabindex=None, translate=None, hx_get=None, +> hx_post=None, hx_put=None, hx_delete=None, hx_patch=None, +> hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None, +> hx_select=None, hx_select_oob=None, hx_indicator=None, +> hx_push_url=None, hx_confirm=None, hx_disable=None, +> hx_replace_url=None, hx_disabled_elt=None, hx_ext=None, +> hx_headers=None, hx_history=None, hx_history_elt=None, +> hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None) + +*A standard SVG `rect` element* + +All our shapes just create regular `FT` elements. The only extra +functionality provided by most of them is to add additional defined +kwargs to improve auto-complete in IDEs and notebooks, and re-order +parameters so that positional args can also be used to save a bit of +typing, e.g: + +``` python +demo(Rect(30, 30, fill='blue', rx=8, ry=8)) +``` + + + +------------------------------------------------------------------------ + +source + +### Circle + +> Circle (r, cx=0, cy=0, fill=None, stroke=None, stroke_width=None, +> transform=None, opacity=None, clip=None, mask=None, filter=None, +> vector_effect=None, pointer_events=None, target_id=None, +> hx_vals=None, hx_target=None, id=None, cls=None, title=None, +> style=None, accesskey=None, contenteditable=None, dir=None, +> draggable=None, enterkeyhint=None, hidden=None, inert=None, +> inputmode=None, lang=None, popover=None, spellcheck=None, +> tabindex=None, translate=None, hx_get=None, hx_post=None, +> hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None, +> hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None) + +*A standard SVG `circle` element* + +``` python +demo(Circle(20, 25, 25, stroke='red', stroke_width=3)) +``` + + + +------------------------------------------------------------------------ + +source + +### Ellipse + +> Ellipse (rx, ry, cx=0, cy=0, fill=None, stroke=None, stroke_width=None, +> transform=None, opacity=None, clip=None, mask=None, filter=None, +> vector_effect=None, pointer_events=None, target_id=None, +> hx_vals=None, hx_target=None, id=None, cls=None, title=None, +> style=None, accesskey=None, contenteditable=None, dir=None, +> draggable=None, enterkeyhint=None, hidden=None, inert=None, +> inputmode=None, lang=None, popover=None, spellcheck=None, +> tabindex=None, translate=None, hx_get=None, hx_post=None, +> hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None, +> hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None) + +*A standard SVG `ellipse` element* + +``` python +demo(Ellipse(20, 10, 25, 25)) +``` + + + +------------------------------------------------------------------------ + +source + +### transformd + +> transformd (translate=None, scale=None, rotate=None, skewX=None, +> skewY=None, matrix=None) + +*Create an SVG `transform` kwarg dict* + +``` python +rot = transformd(rotate=(45, 25, 25)) +rot +``` + + {'transform': 'rotate(45,25,25)'} + +``` python +demo(Ellipse(20, 10, 25, 25, **rot)) +``` + + + +------------------------------------------------------------------------ + +source + +### Line + +> Line (x1, y1, x2=0, y2=0, stroke='black', w=None, stroke_width=1, +> transform=None, opacity=None, clip=None, mask=None, filter=None, +> vector_effect=None, pointer_events=None, target_id=None, +> hx_vals=None, hx_target=None, id=None, cls=None, title=None, +> style=None, accesskey=None, contenteditable=None, dir=None, +> draggable=None, enterkeyhint=None, hidden=None, inert=None, +> inputmode=None, lang=None, popover=None, spellcheck=None, +> tabindex=None, translate=None, hx_get=None, hx_post=None, +> hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None, +> hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None, +> hx_sync=None, hx_validate=None) + +*A standard SVG `line` element* + +``` python +demo(Line(20, 30, w=3)) +``` + + + +------------------------------------------------------------------------ + +source + +### Polyline + +> Polyline (*args, points=None, fill=None, stroke=None, stroke_width=None, +> transform=None, opacity=None, clip=None, mask=None, +> filter=None, vector_effect=None, pointer_events=None, +> target_id=None, hx_vals=None, hx_target=None, id=None, +> cls=None, title=None, style=None, accesskey=None, +> contenteditable=None, dir=None, draggable=None, +> enterkeyhint=None, hidden=None, inert=None, inputmode=None, +> lang=None, popover=None, spellcheck=None, tabindex=None, +> translate=None, hx_get=None, hx_post=None, hx_put=None, +> hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None, +> hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None) + +*A standard SVG `polyline` element* + +``` python +demo(Polyline((0,0), (10,10), (20,0), (30,10), (40,0), + fill='yellow', stroke='blue', stroke_width=2)) +``` + + + +``` python +demo(Polyline(points='0,0 10,10 20,0 30,10 40,0', fill='purple', stroke_width=2)) +``` + + + +------------------------------------------------------------------------ + +source + +### Polygon + +> Polygon (*args, points=None, fill=None, stroke=None, stroke_width=None, +> transform=None, opacity=None, clip=None, mask=None, filter=None, +> vector_effect=None, pointer_events=None, target_id=None, +> hx_vals=None, hx_target=None, id=None, cls=None, title=None, +> style=None, accesskey=None, contenteditable=None, dir=None, +> draggable=None, enterkeyhint=None, hidden=None, inert=None, +> inputmode=None, lang=None, popover=None, spellcheck=None, +> tabindex=None, translate=None, hx_get=None, hx_post=None, +> hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None, +> hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None) + +*A standard SVG `polygon` element* + +``` python +demo(Polygon((25,5), (43.3,15), (43.3,35), (25,45), (6.7,35), (6.7,15), + fill='lightblue', stroke='navy', stroke_width=2)) +``` + + + +``` python +demo(Polygon(points='25,5 43.3,15 43.3,35 25,45 6.7,35 6.7,15', + fill='lightgreen', stroke='darkgreen', stroke_width=2)) +``` + + + +------------------------------------------------------------------------ + +source + +### Text + +> Text (*args, x=0, y=0, font_family=None, font_size=None, fill=None, +> text_anchor=None, dominant_baseline=None, font_weight=None, +> font_style=None, text_decoration=None, transform=None, +> opacity=None, clip=None, mask=None, filter=None, +> vector_effect=None, pointer_events=None, target_id=None, +> hx_vals=None, hx_target=None, id=None, cls=None, title=None, +> style=None, accesskey=None, contenteditable=None, dir=None, +> draggable=None, enterkeyhint=None, hidden=None, inert=None, +> inputmode=None, lang=None, popover=None, spellcheck=None, +> tabindex=None, translate=None, hx_get=None, hx_post=None, +> hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None, +> hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None, +> hx_sync=None, hx_validate=None) + +*A standard SVG `text` element* + +``` python +demo(Text("Hello!", x=10, y=30)) +``` + +Hello! + +## Paths + +Paths in SVGs are more complex, so we add a small (optional) fluent +interface for constructing them: + +------------------------------------------------------------------------ + +source + +### PathFT + +> PathFT (tag:str, cs:tuple, attrs:dict=None, void_=False, **kwargs) + +*A ‘Fast Tag’ structure, containing `tag`,`children`,and `attrs`* + +------------------------------------------------------------------------ + +source + +### Path + +> Path (d='', fill=None, stroke=None, stroke_width=None, transform=None, +> opacity=None, clip=None, mask=None, filter=None, +> vector_effect=None, pointer_events=None, target_id=None, +> hx_vals=None, hx_target=None, id=None, cls=None, title=None, +> style=None, accesskey=None, contenteditable=None, dir=None, +> draggable=None, enterkeyhint=None, hidden=None, inert=None, +> inputmode=None, lang=None, popover=None, spellcheck=None, +> tabindex=None, translate=None, hx_get=None, hx_post=None, +> hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None, +> hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None, +> hx_sync=None, hx_validate=None) + +*Create a standard `path` SVG element. This is a special object* + +Let’s create a square shape, but using +[`Path`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#path) +instead of +[`Rect`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#rect): + +- M(10, 10): Move to starting point (10, 10) +- L(40, 10): Line to (40, 10) - top edge +- L(40, 40): Line to (40, 40) - right edge +- L(10, 40): Line to (10, 40) - bottom edge +- Z(): Close path - connects back to start + +M = Move to, L = Line to, Z = Close path + +``` python +demo(Path(fill='none', stroke='purple', stroke_width=2 + ).M(10, 10).L(40, 10).L(40, 40).L(10, 40).Z()) +``` + + + +Using curves we can create a spiral: + +``` python +p = (Path(fill='none', stroke='purple', stroke_width=2) + .M(25, 25) + .C(25, 25, 20, 20, 30, 20) + .C(40, 20, 40, 30, 30, 30) + .C(20, 30, 20, 15, 35, 15) + .C(50, 15, 50, 35, 25, 35) + .C(0, 35, 0, 10, 40, 10) + .C(80, 10, 80, 40, 25, 40)) +demo(p, 50, 100) +``` + + + +Using arcs and curves we can create a map marker icon: + +``` python +p = (Path(fill='red') + .M(25,45) + .C(25,45,10,35,10,25) + .A(15,15,0,1,1,40,25) + .C(40,35,25,45,25,45) + .Z()) +demo(p) +``` + + + +Behind the scenes it’s just creating regular SVG path `d` attr – you can +pass `d` in directly if you prefer. + +``` python +print(p.d) +``` + + M25 45 C25 45 10 35 10 25 A15 15 0 1 1 40 25 C40 35 25 45 25 45 Z + +``` python +demo(Path(d='M25 45 C25 45 10 35 10 25 A15 15 0 1 1 40 25 C40 35 25 45 25 45 Z')) +``` + + + +------------------------------------------------------------------------ + +source + +### PathFT.M + +> PathFT.M (x, y) + +*Move to.* + +------------------------------------------------------------------------ + +source + +### PathFT.L + +> PathFT.L (x, y) + +*Line to.* + +------------------------------------------------------------------------ + +source + +### PathFT.H + +> PathFT.H (x) + +*Horizontal line to.* + +------------------------------------------------------------------------ + +source + +### PathFT.V + +> PathFT.V (y) + +*Vertical line to.* + +------------------------------------------------------------------------ + +source + +### PathFT.Z + +> PathFT.Z () + +*Close path.* + +------------------------------------------------------------------------ + +source + +### PathFT.C + +> PathFT.C (x1, y1, x2, y2, x, y) + +*Cubic Bézier curve.* + +------------------------------------------------------------------------ + +source + +### PathFT.S + +> PathFT.S (x2, y2, x, y) + +*Smooth cubic Bézier curve.* + +------------------------------------------------------------------------ + +source + +### PathFT.Q + +> PathFT.Q (x1, y1, x, y) + +*Quadratic Bézier curve.* + +------------------------------------------------------------------------ + +source + +### PathFT.T + +> PathFT.T (x, y) + +*Smooth quadratic Bézier curve.* + +------------------------------------------------------------------------ + +source + +### PathFT.A + +> PathFT.A (rx, ry, x_axis_rotation, large_arc_flag, sweep_flag, x, y) + +*Elliptical Arc.* + +## HTMX helpers + +------------------------------------------------------------------------ + +source + +### SvgOob + +> SvgOob (*args, **kwargs) + +*Wraps an SVG shape as required for an HTMX OOB swap* + +When returning an SVG shape out-of-band (OOB) in HTMX, you need to wrap +it with +[`SvgOob`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#svgoob) +to have it appear correctly. +([`SvgOob`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#svgoob) +is just a shortcut for `Template(Svg(...))`, which is the trick that +makes SVG OOB swaps work.) + +------------------------------------------------------------------------ + +source + +### SvgInb + +> SvgInb (*args, **kwargs) + +*Wraps an SVG shape as required for an HTMX inband swap* + +When returning an SVG shape in-band in HTMX, either have the calling +element include `hx_select='svg>*'`, or `**svg_inb` (which are two ways +of saying the same thing), or wrap the response with +[`SvgInb`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#svginb) +to have it appear correctly. +([`SvgInb`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#svginb) +is just a shortcut for the tuple +`(Svg(...), HtmxResponseHeaders(hx_reselect='svg>*'))`, which is the +trick that makes SVG in-band swaps work.) diff --git a/api/xtend.html b/api/xtend.html new file mode 100644 index 00000000..a9ee5e4a --- /dev/null +++ b/api/xtend.html @@ -0,0 +1,1228 @@ + + + + + + + + + + +Component extensions – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Component extensions

+
+ +
+
+ Simple extensions to standard HTML components, such as adding sensible defaults +
+
+ + +
+ + + + +
+ + + +
+ + + +
+
from pprint import pprint
+
+
+

source

+
+

A

+
+
 A (*c, hx_get=None, target_id=None, hx_swap=None, href='#', hx_vals=None,
+    hx_target=None, id=None, cls=None, title=None, style=None,
+    accesskey=None, contenteditable=None, dir=None, draggable=None,
+    enterkeyhint=None, hidden=None, inert=None, inputmode=None, lang=None,
+    popover=None, spellcheck=None, tabindex=None, translate=None,
+    hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,
+    hx_trigger=None, hx_swap_oob=None, hx_include=None, hx_select=None,
+    hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+    hx_confirm=None, hx_disable=None, hx_replace_url=None,
+    hx_disabled_elt=None, hx_ext=None, hx_headers=None, hx_history=None,
+    hx_history_elt=None, hx_inherit=None, hx_params=None,
+    hx_preserve=None, hx_prompt=None, hx_request=None, hx_sync=None,
+    hx_validate=None, **kwargs)
+
+

An A tag; href defaults to ‘#’ for more concise use with HTMX

+
+
A('text', ht_get='/get', target_id='id')
+
+
<a href="#" ht-get="/get" hx-target="#id">text</a>
+
+
+
+

source

+
+
+

AX

+
+
 AX (txt, hx_get=None, target_id=None, hx_swap=None, href='#',
+     hx_vals=None, hx_target=None, id=None, cls=None, title=None,
+     style=None, accesskey=None, contenteditable=None, dir=None,
+     draggable=None, enterkeyhint=None, hidden=None, inert=None,
+     inputmode=None, lang=None, popover=None, spellcheck=None,
+     tabindex=None, translate=None, hx_post=None, hx_put=None,
+     hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap_oob=None,
+     hx_include=None, hx_select=None, hx_select_oob=None,
+     hx_indicator=None, hx_push_url=None, hx_confirm=None,
+     hx_disable=None, hx_replace_url=None, hx_disabled_elt=None,
+     hx_ext=None, hx_headers=None, hx_history=None, hx_history_elt=None,
+     hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None,
+     hx_request=None, hx_sync=None, hx_validate=None, **kwargs)
+
+

An A tag with just one text child, allowing hx_get, target_id, and hx_swap to be positional params

+
+
AX('text', '/get', 'id')
+
+
<a href="#" hx-get="/get" hx-target="#id">text</a>
+
+
+
+
+

Forms

+
+

source

+
+

Form

+
+
 Form (*c, enctype='multipart/form-data', target_id=None, hx_vals=None,
+       hx_target=None, id=None, cls=None, title=None, style=None,
+       accesskey=None, contenteditable=None, dir=None, draggable=None,
+       enterkeyhint=None, hidden=None, inert=None, inputmode=None,
+       lang=None, popover=None, spellcheck=None, tabindex=None,
+       translate=None, hx_get=None, hx_post=None, hx_put=None,
+       hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,
+       hx_swap_oob=None, hx_include=None, hx_select=None,
+       hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+       hx_confirm=None, hx_disable=None, hx_replace_url=None,
+       hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+       hx_history=None, hx_history_elt=None, hx_inherit=None,
+       hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,
+       hx_sync=None, hx_validate=None, **kwargs)
+
+

A Form tag; identical to plain ft_hx version except default enctype='multipart/form-data'

+
+

source

+
+
+

Hidden

+
+
 Hidden (value:Any='', id:Any=None, target_id=None, hx_vals=None,
+         hx_target=None, cls=None, title=None, style=None, accesskey=None,
+         contenteditable=None, dir=None, draggable=None,
+         enterkeyhint=None, hidden=None, inert=None, inputmode=None,
+         lang=None, popover=None, spellcheck=None, tabindex=None,
+         translate=None, hx_get=None, hx_post=None, hx_put=None,
+         hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,
+         hx_swap_oob=None, hx_include=None, hx_select=None,
+         hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+         hx_confirm=None, hx_disable=None, hx_replace_url=None,
+         hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+         hx_history=None, hx_history_elt=None, hx_inherit=None,
+         hx_params=None, hx_preserve=None, hx_prompt=None,
+         hx_request=None, hx_sync=None, hx_validate=None, **kwargs)
+
+

An Input of type ‘hidden’

+
+

source

+
+
+

CheckboxX

+
+
 CheckboxX (checked:bool=False, label=None, value='1', id=None, name=None,
+            target_id=None, hx_vals=None, hx_target=None, cls=None,
+            title=None, style=None, accesskey=None, contenteditable=None,
+            dir=None, draggable=None, enterkeyhint=None, hidden=None,
+            inert=None, inputmode=None, lang=None, popover=None,
+            spellcheck=None, tabindex=None, translate=None, hx_get=None,
+            hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,
+            hx_trigger=None, hx_swap=None, hx_swap_oob=None,
+            hx_include=None, hx_select=None, hx_select_oob=None,
+            hx_indicator=None, hx_push_url=None, hx_confirm=None,
+            hx_disable=None, hx_replace_url=None, hx_disabled_elt=None,
+            hx_ext=None, hx_headers=None, hx_history=None,
+            hx_history_elt=None, hx_inherit=None, hx_params=None,
+            hx_preserve=None, hx_prompt=None, hx_request=None,
+            hx_sync=None, hx_validate=None, **kwargs)
+
+

A Checkbox optionally inside a Label, preceded by a Hidden with matching name

+
+
show(CheckboxX(True, 'Check me out!'))
+
+ + +
+
+
+

source

+
+
+

Script

+
+
 Script (code:str='', id=None, cls=None, title=None, style=None,
+         attrmap=None, valmap=None, ft_cls=None, **kwargs)
+
+

A Script tag that doesn’t escape its code

+
+

source

+
+
+

Style

+
+
 Style (*c, id=None, cls=None, title=None, style=None, attrmap=None,
+        valmap=None, ft_cls=None, **kwargs)
+
+

A Style tag that doesn’t escape its code

+
+
+
+

Style and script templates

+
+

source

+
+

double_braces

+
+
 double_braces (s)
+
+

Convert single braces to double braces if next to special chars or newline

+
+

source

+
+
+

undouble_braces

+
+
 undouble_braces (s)
+
+

Convert double braces to single braces if next to special chars or newline

+
+

source

+
+
+

loose_format

+
+
 loose_format (s, **kw)
+
+

String format s using kw, without being strict about braces outside of template params

+
+

source

+
+
+

ScriptX

+
+
 ScriptX (fname, src=None, nomodule=None, type=None, _async=None,
+          defer=None, charset=None, crossorigin=None, integrity=None,
+          **kw)
+
+

A script element with contents read from fname

+
+

source

+
+
+

replace_css_vars

+
+
 replace_css_vars (css, pre='tpl', **kwargs)
+
+

Replace var(--) CSS variables with kwargs if name prefix matches pre

+
+

source

+
+
+

StyleX

+
+
 StyleX (fname, **kw)
+
+

A style element with contents read from fname and variables replaced from kw

+
+

source

+
+
+

Nbsp

+
+
 Nbsp ()
+
+

A non-breaking space

+
+
+
+

Surreal and JS

+
+

source

+
+

Surreal

+
+
 Surreal (code:str)
+
+

Wrap code in domReadyExecute and set m=me() and p=me('-')

+
+

source

+
+
+

On

+
+
 On (code:str, event:str='click', sel:str='', me=True)
+
+

An async surreal.js script block event handler for event on selector sel,p, making available parent p, event ev, and target e

+
+

source

+
+
+

Prev

+
+
 Prev (code:str, event:str='click')
+
+

An async surreal.js script block event handler for event on previous sibling, with same vars as On

+
+

source

+
+
+

Now

+
+
 Now (code:str, sel:str='')
+
+

An async surreal.js script block on selector me(sel)

+
+

source

+
+
+

AnyNow

+
+
 AnyNow (sel:str, code:str)
+
+

An async surreal.js script block on selector any(sel)

+
+

source

+
+
+

run_js

+
+
 run_js (js, id=None, **kw)
+
+

Run js script, auto-generating id based on name of caller if needed, and js-escaping any kw params

+
+

source

+
+
+

HtmxOn

+
+
 HtmxOn (eventname:str, code:str)
+
+
+

source

+
+
+

jsd

+
+
 jsd (org, repo, root, path, prov='gh', typ='script', ver=None, esm=False,
+      **kwargs)
+
+

jsdelivr Script or CSS Link tag, or URL

+
+
+
+

Other helpers

+
+

source

+
+

Titled

+
+
 Titled (title:str='FastHTML app', *args, cls='container', target_id=None,
+         hx_vals=None, hx_target=None, id=None, style=None,
+         accesskey=None, contenteditable=None, dir=None, draggable=None,
+         enterkeyhint=None, hidden=None, inert=None, inputmode=None,
+         lang=None, popover=None, spellcheck=None, tabindex=None,
+         translate=None, hx_get=None, hx_post=None, hx_put=None,
+         hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,
+         hx_swap_oob=None, hx_include=None, hx_select=None,
+         hx_select_oob=None, hx_indicator=None, hx_push_url=None,
+         hx_confirm=None, hx_disable=None, hx_replace_url=None,
+         hx_disabled_elt=None, hx_ext=None, hx_headers=None,
+         hx_history=None, hx_history_elt=None, hx_inherit=None,
+         hx_params=None, hx_preserve=None, hx_prompt=None,
+         hx_request=None, hx_sync=None, hx_validate=None, **kwargs)
+
+

An HTML partial containing a Title, and H1, and any provided children

+
+

source

+
+
+

Socials

+
+
 Socials (title, site_name, description, image, url=None, w=1200, h=630,
+          twitter_site=None, creator=None, card='summary')
+
+

OG and Twitter social card headers

+
+

source

+
+
+

Favicon

+
+
 Favicon (light_icon, dark_icon)
+
+

Light and dark favicon headers

+
+

source

+
+
+

clear

+
+
 clear (id)
+
+
+

source

+
+
+

with_sid

+
+
 with_sid (app, dest, path='/')
+
+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/api/xtend.html.md b/api/xtend.html.md new file mode 100644 index 00000000..1911cafb --- /dev/null +++ b/api/xtend.html.md @@ -0,0 +1,458 @@ +# Component extensions + + + + +``` python +from pprint import pprint +``` + +------------------------------------------------------------------------ + +source + +### A + +> A (*c, hx_get=None, target_id=None, hx_swap=None, href='#', hx_vals=None, +> hx_target=None, id=None, cls=None, title=None, style=None, +> accesskey=None, contenteditable=None, dir=None, draggable=None, +> enterkeyhint=None, hidden=None, inert=None, inputmode=None, lang=None, +> popover=None, spellcheck=None, tabindex=None, translate=None, +> hx_post=None, hx_put=None, hx_delete=None, hx_patch=None, +> hx_trigger=None, hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, hx_history=None, +> hx_history_elt=None, hx_inherit=None, hx_params=None, +> hx_preserve=None, hx_prompt=None, hx_request=None, hx_sync=None, +> hx_validate=None, **kwargs) + +*An A tag; `href` defaults to ‘\#’ for more concise use with HTMX* + +``` python +A('text', ht_get='/get', target_id='id') +``` + +``` html +text +``` + +------------------------------------------------------------------------ + +source + +### AX + +> AX (txt, hx_get=None, target_id=None, hx_swap=None, href='#', +> hx_vals=None, hx_target=None, id=None, cls=None, title=None, +> style=None, accesskey=None, contenteditable=None, dir=None, +> draggable=None, enterkeyhint=None, hidden=None, inert=None, +> inputmode=None, lang=None, popover=None, spellcheck=None, +> tabindex=None, translate=None, hx_post=None, hx_put=None, +> hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap_oob=None, +> hx_include=None, hx_select=None, hx_select_oob=None, +> hx_indicator=None, hx_push_url=None, hx_confirm=None, +> hx_disable=None, hx_replace_url=None, hx_disabled_elt=None, +> hx_ext=None, hx_headers=None, hx_history=None, hx_history_elt=None, +> hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None, **kwargs) + +*An A tag with just one text child, allowing hx_get, target_id, and +hx_swap to be positional params* + +``` python +AX('text', '/get', 'id') +``` + +``` html +text +``` + +## Forms + +------------------------------------------------------------------------ + +source + +### Form + +> Form (*c, enctype='multipart/form-data', target_id=None, hx_vals=None, +> hx_target=None, id=None, cls=None, title=None, style=None, +> accesskey=None, contenteditable=None, dir=None, draggable=None, +> enterkeyhint=None, hidden=None, inert=None, inputmode=None, +> lang=None, popover=None, spellcheck=None, tabindex=None, +> translate=None, hx_get=None, hx_post=None, hx_put=None, +> hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None, +> hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None, +> hx_sync=None, hx_validate=None, **kwargs) + +*A Form tag; identical to plain +[`ft_hx`](https://AnswerDotAI.github.io/fasthtml/api/components.html#ft_hx) +version except default `enctype='multipart/form-data'`* + +------------------------------------------------------------------------ + +source + +### Hidden + +> Hidden (value:Any='', id:Any=None, target_id=None, hx_vals=None, +> hx_target=None, cls=None, title=None, style=None, accesskey=None, +> contenteditable=None, dir=None, draggable=None, +> enterkeyhint=None, hidden=None, inert=None, inputmode=None, +> lang=None, popover=None, spellcheck=None, tabindex=None, +> translate=None, hx_get=None, hx_post=None, hx_put=None, +> hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None, +> hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None, **kwargs) + +*An Input of type ‘hidden’* + +------------------------------------------------------------------------ + +source + +### CheckboxX + +> CheckboxX (checked:bool=False, label=None, value='1', id=None, name=None, +> target_id=None, hx_vals=None, hx_target=None, cls=None, +> title=None, style=None, accesskey=None, contenteditable=None, +> dir=None, draggable=None, enterkeyhint=None, hidden=None, +> inert=None, inputmode=None, lang=None, popover=None, +> spellcheck=None, tabindex=None, translate=None, hx_get=None, +> hx_post=None, hx_put=None, hx_delete=None, hx_patch=None, +> hx_trigger=None, hx_swap=None, hx_swap_oob=None, +> hx_include=None, hx_select=None, hx_select_oob=None, +> hx_indicator=None, hx_push_url=None, hx_confirm=None, +> hx_disable=None, hx_replace_url=None, hx_disabled_elt=None, +> hx_ext=None, hx_headers=None, hx_history=None, +> hx_history_elt=None, hx_inherit=None, hx_params=None, +> hx_preserve=None, hx_prompt=None, hx_request=None, +> hx_sync=None, hx_validate=None, **kwargs) + +*A Checkbox optionally inside a Label, preceded by a +[`Hidden`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#hidden) +with matching name* + +``` python +show(CheckboxX(True, 'Check me out!')) +``` + + + + +------------------------------------------------------------------------ + +source + +### Script + +> Script (code:str='', id=None, cls=None, title=None, style=None, +> attrmap=None, valmap=None, ft_cls=None, **kwargs) + +*A Script tag that doesn’t escape its code* + +------------------------------------------------------------------------ + +source + +### Style + +> Style (*c, id=None, cls=None, title=None, style=None, attrmap=None, +> valmap=None, ft_cls=None, **kwargs) + +*A Style tag that doesn’t escape its code* + +## Style and script templates + +------------------------------------------------------------------------ + +source + +### double_braces + +> double_braces (s) + +*Convert single braces to double braces if next to special chars or +newline* + +------------------------------------------------------------------------ + +source + +### undouble_braces + +> undouble_braces (s) + +*Convert double braces to single braces if next to special chars or +newline* + +------------------------------------------------------------------------ + +source + +### loose_format + +> loose_format (s, **kw) + +*String format `s` using `kw`, without being strict about braces outside +of template params* + +------------------------------------------------------------------------ + +source + +### ScriptX + +> ScriptX (fname, src=None, nomodule=None, type=None, _async=None, +> defer=None, charset=None, crossorigin=None, integrity=None, +> **kw) + +*A `script` element with contents read from `fname`* + +------------------------------------------------------------------------ + +source + +### replace_css_vars + +> replace_css_vars (css, pre='tpl', **kwargs) + +*Replace `var(--)` CSS variables with `kwargs` if name prefix matches +`pre`* + +------------------------------------------------------------------------ + +source + +### StyleX + +> StyleX (fname, **kw) + +*A `style` element with contents read from `fname` and variables +replaced from `kw`* + +------------------------------------------------------------------------ + +source + +### Nbsp + +> Nbsp () + +*A non-breaking space* + +## Surreal and JS + +------------------------------------------------------------------------ + +source + +### Surreal + +> Surreal (code:str) + +*Wrap `code` in `domReadyExecute` and set `m=me()` and `p=me('-')`* + +------------------------------------------------------------------------ + +source + +### On + +> On (code:str, event:str='click', sel:str='', me=True) + +*An async surreal.js script block event handler for `event` on selector +`sel,p`, making available parent `p`, event `ev`, and target `e`* + +------------------------------------------------------------------------ + +source + +### Prev + +> Prev (code:str, event:str='click') + +*An async surreal.js script block event handler for `event` on previous +sibling, with same vars as +[`On`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#on)* + +------------------------------------------------------------------------ + +source + +### Now + +> Now (code:str, sel:str='') + +*An async surreal.js script block on selector `me(sel)`* + +------------------------------------------------------------------------ + +source + +### AnyNow + +> AnyNow (sel:str, code:str) + +*An async surreal.js script block on selector `any(sel)`* + +------------------------------------------------------------------------ + +source + +### run_js + +> run_js (js, id=None, **kw) + +*Run `js` script, auto-generating `id` based on name of caller if +needed, and js-escaping any `kw` params* + +------------------------------------------------------------------------ + +source + +### HtmxOn + +> HtmxOn (eventname:str, code:str) + +------------------------------------------------------------------------ + +source + +### jsd + +> jsd (org, repo, root, path, prov='gh', typ='script', ver=None, esm=False, +> **kwargs) + +*jsdelivr +[`Script`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#script) +or CSS `Link` tag, or URL* + +## Other helpers + +------------------------------------------------------------------------ + +source + +### Titled + +> Titled (title:str='FastHTML app', *args, cls='container', target_id=None, +> hx_vals=None, hx_target=None, id=None, style=None, +> accesskey=None, contenteditable=None, dir=None, draggable=None, +> enterkeyhint=None, hidden=None, inert=None, inputmode=None, +> lang=None, popover=None, spellcheck=None, tabindex=None, +> translate=None, hx_get=None, hx_post=None, hx_put=None, +> hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None, +> hx_swap_oob=None, hx_include=None, hx_select=None, +> hx_select_oob=None, hx_indicator=None, hx_push_url=None, +> hx_confirm=None, hx_disable=None, hx_replace_url=None, +> hx_disabled_elt=None, hx_ext=None, hx_headers=None, +> hx_history=None, hx_history_elt=None, hx_inherit=None, +> hx_params=None, hx_preserve=None, hx_prompt=None, +> hx_request=None, hx_sync=None, hx_validate=None, **kwargs) + +*An HTML partial containing a `Title`, and `H1`, and any provided +children* + +------------------------------------------------------------------------ + +source + +### Socials + +> Socials (title, site_name, description, image, url=None, w=1200, h=630, +> twitter_site=None, creator=None, card='summary') + +*OG and Twitter social card headers* + +------------------------------------------------------------------------ + +source + +### Favicon + +> Favicon (light_icon, dark_icon) + +*Light and dark favicon headers* + +------------------------------------------------------------------------ + +source + +### clear + +> clear (id) + +------------------------------------------------------------------------ + +source + +### with_sid + +> with_sid (app, dest, path='/') diff --git a/apilist.txt b/apilist.txt new file mode 100644 index 00000000..801d1a88 --- /dev/null +++ b/apilist.txt @@ -0,0 +1,467 @@ +# fasthtml Module Documentation + +## fasthtml.authmw + +- `class BasicAuthMiddleware` + - `def __init__(self, app, cb, skip)` + - `def __call__(self, scope, receive, send)` + - `def authenticate(self, conn)` + +## fasthtml.cli + +- `@call_parse def railway_link()` + Link the current directory to the current project's Railway service + +- `@call_parse def railway_deploy(name, mount)` + Deploy a FastHTML app to Railway + +## fasthtml.components + +> `ft_html` and `ft_hx` functions to add some conveniences to `ft`, along with a full set of basic HTML components, and functions to work with forms and `FT` conversion + +- `def show(ft, *rest)` + Renders FT Components into HTML within a Jupyter notebook. + +- `def File(fname)` + Use the unescaped text in file `fname` directly + +- `def fill_form(form, obj)` + Fills named items in `form` using attributes in `obj` + +- `def fill_dataclass(src, dest)` + Modifies dataclass in-place and returns it + +- `def find_inputs(e, tags, **kw)` + Recursively find all elements in `e` with `tags` and attrs matching `kw` + +- `def html2ft(html, attr1st)` + Convert HTML to an `ft` expression + +- `def sse_message(elm, event)` + Convert element `elm` into a format suitable for SSE streaming + +## fasthtml.core + +> The `FastHTML` subclass of `Starlette`, along with the `RouterX` and `RouteX` classes it automatically uses. + +- `def parsed_date(s)` + Convert `s` to a datetime + +- `def snake2hyphens(s)` + Convert `s` from snake case to hyphenated and capitalised + +- `@dataclass class HtmxHeaders` + - `def __bool__(self)` + - `def __init__(self, boosted, current_url, history_restore_request, prompt, request, target, trigger_name, trigger)` + +- `@dataclass class HttpHeader` + - `def __init__(self, k, v)` + +- `@use_kwargs_dict(**htmx_resps) def HtmxResponseHeaders(**kwargs)` + HTMX response headers + +- `def form2dict(form)` + Convert starlette form data to a dict + +- `def parse_form(req)` + Starlette errors on empty multipart forms, so this checks for that situation + +- `def flat_xt(lst)` + Flatten lists + +- `class Beforeware` + - `def __init__(self, f, skip)` + +- `def EventStream(s)` + Create a text/event-stream response from `s` + +- `def flat_tuple(o)` + Flatten lists + +- `def noop_body(c, req)` + Default Body wrap function which just returns the content + +- `def respond(req, heads, bdy)` + Default FT response creation function + +- `class Redirect` + Use HTMX or Starlette RedirectResponse as required to redirect to `loc` + + - `def __init__(self, loc)` + - `def __response__(self, req)` + +- `def qp(p, **kw)` + Add query parameters to path p + +- `def def_hdrs(htmx, surreal)` + Default headers for a FastHTML app + +- `class FastHTML` + - `def __init__(self, debug, routes, middleware, title, exception_handlers, on_startup, on_shutdown, lifespan, hdrs, ftrs, exts, before, after, surreal, htmx, default_hdrs, sess_cls, secret_key, session_cookie, max_age, sess_path, same_site, sess_https_only, sess_domain, key_fname, body_wrap, htmlkw, nb_hdrs, **bodykw)` + - `def add_route(self, route)` + +- `@patch def ws(self, path, conn, disconn, name, middleware)` + Add a websocket route at `path` + +- `def nested_name(f)` + Get name of function `f` using '_' to join nested function names + +- `@patch def route(self, path, methods, name, include_in_schema, body_wrap)` + Add a route at `path` + +- `def serve(appname, app, host, port, reload, reload_includes, reload_excludes)` + Run the app in an async server, with live reload set as the default. + +- `class Client` + A simple httpx ASGI client that doesn't require `async` + + - `def __init__(self, app, url)` + +- `class RouteFuncs` + - `def __init__(self)` + - `def __setattr__(self, name, value)` + - `def __getattr__(self, name)` + - `def __dir__(self)` + +- `class APIRouter` + Add routes to an app + + - `def __init__(self, prefix, body_wrap)` + - `def __call__(self, path, methods, name, include_in_schema, body_wrap)` + Add a route at `path` + + - `def __getattr__(self, name)` + - `def to_app(self, app)` + Add routes to `app` + + - `def ws(self, path, conn, disconn, name, middleware)` + Add a websocket route at `path` + + +- `def cookie(key, value, max_age, expires, path, domain, secure, httponly, samesite)` + Create a 'set-cookie' `HttpHeader` + +- `@patch def static_route_exts(self, prefix, static_path, exts)` + Add a static route at URL path `prefix` with files from `static_path` and `exts` defined by `reg_re_param()` + +- `@patch def static_route(self, ext, prefix, static_path)` + Add a static route at URL path `prefix` with files from `static_path` and single `ext` (including the '.') + +- `class MiddlewareBase` + - `def __call__(self, scope, receive, send)` + +- `class FtResponse` + Wrap an FT response with any Starlette `Response` + + - `def __init__(self, content, status_code, headers, cls, media_type)` + - `def __response__(self, req)` + +## fasthtml.fastapp + +> The `fast_app` convenience wrapper + +- `def fast_app(db_file, render, hdrs, ftrs, tbls, before, middleware, live, debug, routes, exception_handlers, on_startup, on_shutdown, lifespan, default_hdrs, pico, surreal, htmx, exts, secret_key, key_fname, session_cookie, max_age, sess_path, same_site, sess_https_only, sess_domain, htmlkw, bodykw, reload_attempts, reload_interval, static_path, body_wrap, nb_hdrs, **kwargs)` + Create a FastHTML or FastHTMLWithLiveReload app. + +## fasthtml.js + +> Basic external Javascript lib wrappers + +- `def light_media(css)` + Render light media for day mode views + +- `def dark_media(css)` + Render dark media for night mode views + +- `def MarkdownJS(sel)` + Implements browser-based markdown rendering. + +- `def HighlightJS(sel, langs, light, dark)` + Implements browser-based syntax highlighting. Usage example [here](/tutorials/quickstart_for_web_devs.html#code-highlighting). + +- `def MermaidJS(sel, theme)` + Implements browser-based Mermaid diagram rendering. + +## fasthtml.jupyter + +> Use FastHTML in Jupyter notebooks + +- `def nb_serve(app, log_level, port, host, **kwargs)` + Start a Jupyter compatible uvicorn server with ASGI `app` on `port` with `log_level` + +- `def nb_serve_async(app, log_level, port, host, **kwargs)` + Async version of `nb_serve` + +- `def is_port_free(port, host)` + Check if `port` is free on `host` + +- `def wait_port_free(port, host, max_wait)` + Wait for `port` to be free on `host` + +- `class JupyUvi` + Start and stop a Jupyter compatible uvicorn server with ASGI `app` on `port` with `log_level` + + - `def __init__(self, app, log_level, host, port, start, **kwargs)` + - `def start(self)` + - `def stop(self)` + +- `def HTMX(path, app, host, port, height, link, iframe)` + An iframe which displays the HTMX application in a notebook. + +## fasthtml.live_reload + +- `class FastHTMLWithLiveReload` + `FastHTMLWithLiveReload` enables live reloading. + This means that any code changes saved on the server will automatically + trigger a reload of both the server and browser window. + + How does it work? + - a websocket is created at `/live-reload` + - a small js snippet `LIVE_RELOAD_SCRIPT` is injected into each webpage + - this snippet connects to the websocket at `/live-reload` and listens for an `onclose` event + - when the `onclose` event is detected the browser is reloaded + + Why do we listen for an `onclose` event? + When code changes are saved the server automatically reloads if the --reload flag is set. + The server reload kills the websocket connection. The `onclose` event serves as a proxy + for "developer has saved some changes". + + Usage + >>> from fasthtml.common import * + >>> app = FastHTMLWithLiveReload() + + Run: + serve() + + - `def __init__(self, *args, **kwargs)` + +## fasthtml.oauth + +> Basic scaffolding for handling OAuth + +- `class GoogleAppClient` + A `WebApplicationClient` for Google oauth2 + + - `def __init__(self, client_id, client_secret, code, scope, **kwargs)` + - `@classmethod def from_file(cls, fname, code, scope, **kwargs)` + +- `class GitHubAppClient` + A `WebApplicationClient` for GitHub oauth2 + + - `def __init__(self, client_id, client_secret, code, scope, **kwargs)` + +- `class HuggingFaceClient` + A `WebApplicationClient` for HuggingFace oauth2 + + - `def __init__(self, client_id, client_secret, code, scope, state, **kwargs)` + +- `class DiscordAppClient` + A `WebApplicationClient` for Discord oauth2 + + - `def __init__(self, client_id, client_secret, is_user, perms, scope, **kwargs)` + - `def login_link(self)` + - `def parse_response(self, code)` + +- `class Auth0AppClient` + A `WebApplicationClient` for Auth0 OAuth2 + + - `def __init__(self, domain, client_id, client_secret, code, scope, redirect_uri, **kwargs)` + - `def login_link(self, req)` + +- `@patch def login_link(self, redirect_uri, scope, state)` + Get a login link for this client + +- `def redir_url(request, redir_path, scheme)` + Get the redir url for the host in `request` + +- `@patch def parse_response(self, code, redirect_uri)` + Get the token from the oauth2 server response + +- `@patch def get_info(self, token)` + Get the info for authenticated user + +- `@patch def retr_info(self, code, redirect_uri)` + Combines `parse_response` and `get_info` + +- `@patch def retr_id(self, code, redirect_uri)` + Call `retr_info` and then return id/subscriber value + +- `class OAuth` + - `def __init__(self, app, cli, skip, redir_path, error_path, logout_path, login_path, https, http_patterns)` + - `def redir_login(self, session)` + - `def redir_url(self, req)` + - `def login_link(self, req, scope, state)` + - `def check_invalid(self, req, session, auth)` + - `def logout(self, session)` + - `def get_auth(self, info, ident, session, state)` + +## fasthtml.pico + +> Basic components for generating Pico CSS tags + +- `@delegates(ft_hx, keep=True) def Card(*c, **kwargs)` + A PicoCSS Card, implemented as an Article with optional Header and Footer + +- `@delegates(ft_hx, keep=True) def Group(*c, **kwargs)` + A PicoCSS Group, implemented as a Fieldset with role 'group' + +- `@delegates(ft_hx, keep=True) def Search(*c, **kwargs)` + A PicoCSS Search, implemented as a Form with role 'search' + +- `@delegates(ft_hx, keep=True) def Grid(*c, **kwargs)` + A PicoCSS Grid, implemented as child Divs in a Div with class 'grid' + +- `@delegates(ft_hx, keep=True) def DialogX(*c, **kwargs)` + A PicoCSS Dialog, with children inside a Card + +- `@delegates(ft_hx, keep=True) def Container(*args, **kwargs)` + A PicoCSS Container, implemented as a Main with class 'container' + +## fasthtml.svg + +> Simple SVG FT elements + +- `def Svg(*args, **kwargs)` + An SVG tag; xmlns is added automatically, and viewBox defaults to height and width if not provided + +- `@delegates(ft_hx) def ft_svg(tag, *c, **kwargs)` + Create a standard `FT` element with some SVG-specific attrs + +- `@delegates(ft_svg) def Rect(width, height, x, y, fill, stroke, stroke_width, rx, ry, **kwargs)` + A standard SVG `rect` element + +- `@delegates(ft_svg) def Circle(r, cx, cy, fill, stroke, stroke_width, **kwargs)` + A standard SVG `circle` element + +- `@delegates(ft_svg) def Ellipse(rx, ry, cx, cy, fill, stroke, stroke_width, **kwargs)` + A standard SVG `ellipse` element + +- `def transformd(translate, scale, rotate, skewX, skewY, matrix)` + Create an SVG `transform` kwarg dict + +- `@delegates(ft_svg) def Line(x1, y1, x2, y2, stroke, w, stroke_width, **kwargs)` + A standard SVG `line` element + +- `@delegates(ft_svg) def Polyline(*args, **kwargs)` + A standard SVG `polyline` element + +- `@delegates(ft_svg) def Polygon(*args, **kwargs)` + A standard SVG `polygon` element + +- `@delegates(ft_svg) def Text(*args, **kwargs)` + A standard SVG `text` element + +- `class PathFT` + - `def M(self, x, y)` + Move to. + + - `def L(self, x, y)` + Line to. + + - `def H(self, x)` + Horizontal line to. + + - `def V(self, y)` + Vertical line to. + + - `def Z(self)` + Close path. + + - `def C(self, x1, y1, x2, y2, x, y)` + Cubic Bézier curve. + + - `def S(self, x2, y2, x, y)` + Smooth cubic Bézier curve. + + - `def Q(self, x1, y1, x, y)` + Quadratic Bézier curve. + + - `def T(self, x, y)` + Smooth quadratic Bézier curve. + + - `def A(self, rx, ry, x_axis_rotation, large_arc_flag, sweep_flag, x, y)` + Elliptical Arc. + + +- `def SvgOob(*args, **kwargs)` + Wraps an SVG shape as required for an HTMX OOB swap + +- `def SvgInb(*args, **kwargs)` + Wraps an SVG shape as required for an HTMX inband swap + +## fasthtml.xtend + +> Simple extensions to standard HTML components, such as adding sensible defaults + +- `@delegates(ft_hx, keep=True) def A(*c, **kwargs)` + An A tag; `href` defaults to '#' for more concise use with HTMX + +- `@delegates(ft_hx, keep=True) def AX(txt, hx_get, target_id, hx_swap, href, **kwargs)` + An A tag with just one text child, allowing hx_get, target_id, and hx_swap to be positional params + +- `@delegates(ft_hx, keep=True) def Form(*c, **kwargs)` + A Form tag; identical to plain `ft_hx` version except default `enctype='multipart/form-data'` + +- `@delegates(ft_hx, keep=True) def Hidden(value, id, **kwargs)` + An Input of type 'hidden' + +- `@delegates(ft_hx, keep=True) def CheckboxX(checked, label, value, id, name, **kwargs)` + A Checkbox optionally inside a Label, preceded by a `Hidden` with matching name + +- `@delegates(ft_html, keep=True) def Script(code, **kwargs)` + A Script tag that doesn't escape its code + +- `@delegates(ft_html, keep=True) def Style(*c, **kwargs)` + A Style tag that doesn't escape its code + +- `def double_braces(s)` + Convert single braces to double braces if next to special chars or newline + +- `def undouble_braces(s)` + Convert double braces to single braces if next to special chars or newline + +- `def loose_format(s, **kw)` + String format `s` using `kw`, without being strict about braces outside of template params + +- `def ScriptX(fname, src, nomodule, type, _async, defer, charset, crossorigin, integrity, **kw)` + A `script` element with contents read from `fname` + +- `def replace_css_vars(css, pre, **kwargs)` + Replace `var(--)` CSS variables with `kwargs` if name prefix matches `pre` + +- `def StyleX(fname, **kw)` + A `style` element with contents read from `fname` and variables replaced from `kw` + +- `def Nbsp()` + A non-breaking space + +- `def Surreal(code)` + Wrap `code` in `domReadyExecute` and set `m=me()` and `p=me('-')` + +- `def On(code, event, sel, me)` + An async surreal.js script block event handler for `event` on selector `sel,p`, making available parent `p`, event `ev`, and target `e` + +- `def Prev(code, event)` + An async surreal.js script block event handler for `event` on previous sibling, with same vars as `On` + +- `def Now(code, sel)` + An async surreal.js script block on selector `me(sel)` + +- `def AnyNow(sel, code)` + An async surreal.js script block on selector `any(sel)` + +- `def run_js(js, id, **kw)` + Run `js` script, auto-generating `id` based on name of caller if needed, and js-escaping any `kw` params + +- `def jsd(org, repo, root, path, prov, typ, ver, esm, **kwargs)` + jsdelivr `Script` or CSS `Link` tag, or URL + +- `@delegates(ft_hx, keep=True) def Titled(title, *args, **kwargs)` + An HTML partial containing a `Title`, and `H1`, and any provided children + +- `def Socials(title, site_name, description, image, url, w, h, twitter_site, creator, card)` + OG and Twitter social card headers + +- `def Favicon(light_icon, dark_icon)` + Light and dark favicon headers + diff --git a/explains/explaining_xt_components.html b/explains/explaining_xt_components.html new file mode 100644 index 00000000..1939cdc7 --- /dev/null +++ b/explains/explaining_xt_components.html @@ -0,0 +1,992 @@ + + + + + + + + + + +FT Components – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

FT Components

+
+ +
+
+ FT components turn Python objects into HTML. +
+
+ + +
+ + + + +
+ + + +
+ + + +

FT, or ‘FastTags’, are the display components of FastHTML. In fact, the word “components” in the context of FastHTML is often synonymous with FT.

+

For example, when we look at a FastHTML app, in particular the views, as well as various functions and other objects, we see something like the code snippet below. It’s the return statement that we want to pay attention to:

+
+
from fasthtml.common import *
+
+def example():
+    # The code below is a set of ft components
+    return Div(
+            H1("FastHTML APP"),
+            P("Let's do this"),
+            cls="go"
+    )
+
+

Let’s go ahead and call our function and print the result:

+
+
example()
+
+
<div class="go">
+  <h1>FastHTML APP</h1>
+  <p>Let&#x27;s do this</p>
+</div>
+
+
+

As you can see, when returned to the user from a Python callable, like a function, the ft components are transformed into their string representations of XML or XML-like content such as HTML. More concisely, ft turns Python objects into HTML.

+

Now that we know what ft components look and behave like we can begin to understand them. At their most fundamental level, ft components:

+
    +
  1. Are Python callables, specifically functions, classes, methods of classes, lambda functions, and anything else called with parenthesis that returns a value.
  2. +
  3. Return a sequence of values which has three elements: +
      +
    1. The tag to be generated
    2. +
    3. The content of the tag, which is a tuple of strings/tuples. If a tuple, it is the three-element structure of an ft component
    4. +
    5. A dictionary of XML attributes and their values
    6. +
  4. +
  5. FastHTML’s default ft components words begin with an uppercase letter. Examples include Title(), Ul(), and Div() Custom components have included things like BlogPost and CityMap.
  6. +
+
+

How FastHTML names ft components

+

When it comes to naming ft components, FastHTML appears to break from PEP8. Specifically, PEP8 specifies that when naming variables, functions and instantiated classes we use the snake_case_pattern. That is to say, lowercase with words separated by underscores. However, FastHTML uses PascalCase for ft components.

+

There’s a couple of reasons for this:

+
    +
  1. ft components can be made from any callable type, so adhering to any one pattern doesn’t make much sense
  2. +
  3. It makes for easier reading of FastHTML code, as anything that is PascalCase is probably an ft component
  4. +
+
+
+

Default FT components

+

FastHTML has over 150 FT components designed to accelerate web development. Most of these mirror HTML tags such as <div>, <p>, <a>, <title>, and more. However, there are some extra tags added, including:

+
    +
  • Titled, a combination of the Title() and H1() tags
  • +
  • Socials, renders popular social media tags
  • +
+
+
+

The fasthtml.ft Namespace

+

Some people prefer to write code using namespaces while adhering to PEP8. If that’s a preference, projects can be coded using the fasthtml.ft namespace.

+
+
from fasthtml import ft
+
+ft.Ul(
+    ft.Li("one"),
+    ft.Li("two"),
+    ft.Li("three")
+)
+
+
<ul>
+  <li>one</li>
+  <li>two</li>
+  <li>three</li>
+</ul>
+
+
+
+
+

Attributes

+

This example demonstrates many important things to know about how ft components handle attributes.

+
#| echo: False
+1Label(
+    "Choose an option", 
+    Select(
+2        Option("one", value="1", selected=True),
+3        Option("two", value="2", selected=False),
+4        Option("three", value=3),
+5        cls="selector",
+6        _id="counter",
+7        **{'@click':"alert('Clicked');"},
+    ),
+8    _for="counter",
+)
+
+
1
+
+Line 2 demonstrates that FastHTML appreciates Labels surrounding their fields. +
+
2
+
+On line 5, we can see that attributes set to the boolean value of True are rendered with just the name of the attribute. +
+
3
+
+On line 6, we demonstrate that attributes set to the boolean value of False do not appear in the rendered output. +
+
4
+
+Line 7 is an example of how integers and other non-string values in the rendered output are converted to strings. +
+
5
+
+Line 8 is where we set the HTML class using the cls argument. We use cls here as class is a reserved word in Python. During the rendering process this will be converted to the word “class”. +
+
6
+
+Line 9 demonstrates that any named argument passed into an ft component will have the leading underscore stripped away before rendering. Useful for handling reserved words in Python. +
+
7
+
+On line 10 we have an attribute name that cannot be represented as a python variable. In cases like these, we can use an unpacked dict to represent these values. +
+
8
+
+The use of _for on line 12 is another demonstration of an argument having the leading underscore stripped during render. We can also use fr as that will be expanded to for. +
+
+

This renders the following HTML snippet:

+
+
Label(
+    "Choose an option", 
+    Select(
+        Option("one", value="1", selected=True),
+        Option("two", value="2", selected=False),
+        Option("three", value=3),  # <4>,
+        cls="selector",
+        _id="counter",
+        **{'@click':"alert('Clicked');"},
+    ),
+    _for="counter",
+)
+
+
<label for="counter">
+Choose an option
+  <select id="counter" @click="alert(&#x27;Clicked&#x27;);" class="selector" name="counter">
+    <option value="1" selected>one</option>
+    <option value="2" >two</option>
+    <option value="3">three</option>
+  </select>
+</label>
+
+
+
+
+

Defining new ft components

+

It is possible and sometimes useful to create your own ft components that generate non-standard tags that are not in the FastHTML library. FastHTML supports created and defining those new tags flexibly.

+

For more information, see the Defining new ft components reference page.

+
+
+

FT components and type hints

+

If you use type hints, we strongly suggest that FT components be treated as the Any type.

+

The reason is that FastHTML leverages python’s dynamic features to a great degree. Especially when it comes to FT components, which can evaluate out to be FT|str|None|tuple as well as anything that supports the __ft__, __html__, and __str__ method. That’s enough of the Python stack that assigning anything but Any to be the FT type will prove an exercise in frustation.

+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/explains/explaining_xt_components.html.md b/explains/explaining_xt_components.html.md new file mode 100644 index 00000000..2005877a --- /dev/null +++ b/explains/explaining_xt_components.html.md @@ -0,0 +1,215 @@ +# **FT** Components + + + + +**FT**, or ‘FastTags’, are the display components of FastHTML. In fact, +the word “components” in the context of FastHTML is often synonymous +with **FT**. + +For example, when we look at a FastHTML app, in particular the views, as +well as various functions and other objects, we see something like the +code snippet below. It’s the `return` statement that we want to pay +attention to: + +``` python +from fasthtml.common import * + +def example(): + # The code below is a set of ft components + return Div( + H1("FastHTML APP"), + P("Let's do this"), + cls="go" + ) +``` + +Let’s go ahead and call our function and print the result: + +``` python +example() +``` + +``` xml +
+

FastHTML APP

+

Let's do this

+
+``` + +As you can see, when returned to the user from a Python callable, like a +function, the ft components are transformed into their string +representations of XML or XML-like content such as HTML. More concisely, +*ft turns Python objects into HTML*. + +Now that we know what ft components look and behave like we can begin to +understand them. At their most fundamental level, ft components: + +1. Are Python callables, specifically functions, classes, methods of + classes, lambda functions, and anything else called with parenthesis + that returns a value. +2. Return a sequence of values which has three elements: + 1. The tag to be generated + 2. The content of the tag, which is a tuple of strings/tuples. If a + tuple, it is the three-element structure of an ft component + 3. A dictionary of XML attributes and their values +3. FastHTML’s default ft components words begin with an uppercase + letter. Examples include `Title()`, `Ul()`, and `Div()` Custom + components have included things like `BlogPost` and `CityMap`. + +## How FastHTML names ft components + +When it comes to naming ft components, FastHTML appears to break from +PEP8. Specifically, PEP8 specifies that when naming variables, functions +and instantiated classes we use the `snake_case_pattern`. That is to +say, lowercase with words separated by underscores. However, FastHTML +uses `PascalCase` for ft components. + +There’s a couple of reasons for this: + +1. ft components can be made from any callable type, so adhering to any + one pattern doesn’t make much sense +2. It makes for easier reading of FastHTML code, as anything that is + PascalCase is probably an ft component + +## Default **FT** components + +FastHTML has over 150 **FT** components designed to accelerate web +development. Most of these mirror HTML tags such as `
`, `

`, +``, ``, and more. However, there are some extra tags added, +including: + +- [`Titled`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#titled), + a combination of the `Title()` and `H1()` tags +- [`Socials`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#socials), + renders popular social media tags + +## The `fasthtml.ft` Namespace + +Some people prefer to write code using namespaces while adhering to +PEP8. If that’s a preference, projects can be coded using the +`fasthtml.ft` namespace. + +``` python +from fasthtml import ft + +ft.Ul( + ft.Li("one"), + ft.Li("two"), + ft.Li("three") +) +``` + +``` xml +<ul> + <li>one</li> + <li>two</li> + <li>three</li> +</ul> +``` + +## Attributes + +This example demonstrates many important things to know about how ft +components handle attributes. + +``` python +#| echo: False +Label( + "Choose an option", + Select( + Option("one", value="1", selected=True), + Option("two", value="2", selected=False), + Option("three", value=3), + cls="selector", + _id="counter", + **{'@click':"alert('Clicked');"}, + ), + _for="counter", +) +``` + +Line 2 +Line 2 demonstrates that FastHTML appreciates `Label`s surrounding their +fields. + +Line 5 +On line 5, we can see that attributes set to the `boolean` value of +`True` are rendered with just the name of the attribute. + +Line 6 +On line 6, we demonstrate that attributes set to the `boolean` value of +`False` do not appear in the rendered output. + +Line 7 +Line 7 is an example of how integers and other non-string values in the +rendered output are converted to strings. + +Line 8 +Line 8 is where we set the HTML class using the `cls` argument. We use +`cls` here as `class` is a reserved word in Python. During the rendering +process this will be converted to the word “class”. + +Line 9 +Line 9 demonstrates that any named argument passed into an ft component +will have the leading underscore stripped away before rendering. Useful +for handling reserved words in Python. + +Line 10 +On line 10 we have an attribute name that cannot be represented as a +python variable. In cases like these, we can use an unpacked `dict` to +represent these values. + +Line 12 +The use of `_for` on line 12 is another demonstration of an argument +having the leading underscore stripped during render. We can also use +`fr` as that will be expanded to `for`. + +This renders the following HTML snippet: + +``` python +Label( + "Choose an option", + Select( + Option("one", value="1", selected=True), + Option("two", value="2", selected=False), + Option("three", value=3), # <4>, + cls="selector", + _id="counter", + **{'@click':"alert('Clicked');"}, + ), + _for="counter", +) +``` + +``` xml +<label for="counter"> +Choose an option + <select id="counter" @click="alert('Clicked');" class="selector" name="counter"> + <option value="1" selected>one</option> + <option value="2" >two</option> + <option value="3">three</option> + </select> +</label> +``` + +## Defining new ft components + +It is possible and sometimes useful to create your own ft components +that generate non-standard tags that are not in the FastHTML library. +FastHTML supports created and defining those new tags flexibly. + +For more information, see the [Defining new ft +components](../ref/defining_xt_component) reference page. + +## FT components and type hints + +If you use type hints, we strongly suggest that FT components be treated +as the `Any` type. + +The reason is that FastHTML leverages python’s dynamic features to a +great degree. Especially when it comes to `FT` components, which can +evaluate out to be `FT|str|None|tuple` as well as anything that supports +the `__ft__`, `__html__`, and `__str__` method. That’s enough of the +Python stack that assigning anything but `Any` to be the FT type will +prove an exercise in frustation. diff --git a/explains/faq.html b/explains/faq.html new file mode 100644 index 00000000..429f0c42 --- /dev/null +++ b/explains/faq.html @@ -0,0 +1,897 @@ +<!DOCTYPE html> +<html xmlns="http://www.w3.org/1999/xhtml" lang="en" xml:lang="en"><head> + +<meta charset="utf-8"> +<meta name="generator" content="quarto-1.6.40"> + +<meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes"> + +<meta name="description" content="Frequently Asked Questions"> + +<title>FAQ – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +

+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

FAQ

+
+ +
+
+ Frequently Asked Questions +
+
+ + +
+ + + + +
+ + + +
+ + + +
+

Why does my editor say that I have errors in my FastHTML code?

+

Many editors, including Visual Studio Code, use PyLance to provide error checking for Python. However, PyLance’s error checking is just a guess – it can’t actually know whether your code is correct or not. PyLance particularly struggles with FastHTML’s syntax, which leads to it often reporting false error messages in FastHTML projects.

+

To avoid these misleading error messages, it’s best to disable some PyLance error checking in your FastHTML projects. Here’s how to do it in Visual Studio Code (the same approach should also work in other editors based on vscode, such as Cursor and GitHub Codespaces):

+
    +
  1. Open your FastHTML project
  2. +
  3. Press Ctrl+Shift+P (or Cmd+Shift+P on Mac) to open the Command Palette
  4. +
  5. Type “Preferences: Open Workspace Settings (JSON)” and select it
  6. +
  7. In the JSON file that opens, add the following lines:
  8. +
+
{
+ "python.analysis.diagnosticSeverityOverrides": {
+      "reportGeneralTypeIssues": "none",
+      "reportOptionalMemberAccess": "none",
+      "reportWildcardImportFromLibrary": "none",
+      "reportRedeclaration": "none",
+      "reportAttributeAccessIssue": "none",
+      "reportInvalidTypeForm": "none",
+      "reportAssignmentType": "none",
+  }
+}
+
    +
  1. Save the file
  2. +
+

Even with PyLance diagnostics turned off, your FastHTML code will still run correctly. If you’re still seeing some false errors from PyLance, you can disable it entirely by adding this to your settings:

+
{
+  "python.analysis.ignore": [  "*"  ]
+}
+
+
+

Why the distinctive coding style?

+

FastHTML coding style is the fastai coding style.

+

If you are coming from a data science background the fastai coding style may already be your preferred style.

+

If you are coming from a PEP-8 background where the use of ruff is encouraged, there is a learning curve. However, once you get used to the fastai coding style you may discover yourself appreciating the concise nature of this style. It also encourages using more functional programming tooling, which is both productive and fun. Having said that, it’s entirely optional!

+
+
+

Why not JSX?

+

Many have asked! We think there’s no benefit… Python’s positional and kw args precisely 1:1 map already to html/xml children and attrs, so there’s no need for a new syntax.

+

We wrote some more thoughts on Why Python HTML components over Jinja2, Mako, or JSX here.

+
+
+

Why use import *

+

First, through the use of the __all__ attribute in our Python modules we control what actually gets imported. So there’s no risk of namespace pollution.

+

Second, our style lends itself to working in rather compact Jupyter notebooks and small Python modules. Hence we know about the source code whose libraries we import * from. This terseness means we can develop faster. We’re a small team, and any edge we can gain is important to us.

+

Third, for external libraries, be it core Python, SQLAlchemy, or other things we do tend to use explicit imports. In part to avoid namespace collisions, and also as reference to know where things are coming from.

+

We’ll finish by saying a lot of our users employ explicit imports. If that’s the path you want to take, we encourage the use of from fasthtml import common as fh. The acronym of fh makes it easy to recognize that a symbol is from the FastHTML library.

+
+
+

Can FastHTML be used for dashboards?

+

Yes it can. In fact, it excels at building dashboards. In addition to being great for building static dashboards, because of its foundation in ASGI and tech stack, FastHTML natively supports Websockets. That means using FastHTML we can create dashboards that autoupdate.

+
+
+

Why is FastHTML developed using notebooks?

+

Some people are under the impression that writing software in notebooks is bad.

+

Watch this video. We’ve used Jupyter notebooks exported via nbdev to write a wide range of “very serious” software projects over the last three years. This includes deep learning libraries, API clients, Python language extensions, terminal user interfaces, web frameworks, and more!

+

nbdev is a Jupyter-powered tool for writing software. Traditional programming environments throw away the result of your exploration in REPLs or notebooks. nbdev makes exploration an integral part of your workflow, all while promoting software engineering best practices.

+
+
+

Why not pyproject.toml for packaging?

+

FastHTML uses a setup.py module instead of a pyproject.toml file to configure itself for installation. The reason for this is pyproject.toml is not compatible with nbdev, which is what is used to write and build FastHTML.

+

The nbdev project spent around a year trying to move to pyproject.toml but there was insufficient functionality in the toml-based approach to complete the transition.

+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/explains/faq.html.md b/explains/faq.html.md new file mode 100644 index 00000000..bab46e9d --- /dev/null +++ b/explains/faq.html.md @@ -0,0 +1,133 @@ +# FAQ + + + + +## Why does my editor say that I have errors in my FastHTML code? + +Many editors, including Visual Studio Code, use PyLance to provide error +checking for Python. However, PyLance’s error checking is just a guess – +it can’t actually know whether your code is correct or not. PyLance +particularly struggles with FastHTML’s syntax, which leads to it often +reporting false error messages in FastHTML projects. + +To avoid these misleading error messages, it’s best to disable some +PyLance error checking in your FastHTML projects. Here’s how to do it in +Visual Studio Code (the same approach should also work in other editors +based on vscode, such as Cursor and GitHub Codespaces): + +1. Open your FastHTML project +2. Press `Ctrl+Shift+P` (or `Cmd+Shift+P` on Mac) to open the Command + Palette +3. Type “Preferences: Open Workspace Settings (JSON)” and select it +4. In the JSON file that opens, add the following lines: + +``` json +{ + "python.analysis.diagnosticSeverityOverrides": { + "reportGeneralTypeIssues": "none", + "reportOptionalMemberAccess": "none", + "reportWildcardImportFromLibrary": "none", + "reportRedeclaration": "none", + "reportAttributeAccessIssue": "none", + "reportInvalidTypeForm": "none", + "reportAssignmentType": "none", + } +} +``` + +5. Save the file + +Even with PyLance diagnostics turned off, your FastHTML code will still +run correctly. If you’re still seeing some false errors from PyLance, +you can disable it entirely by adding this to your settings: + +``` json +{ + "python.analysis.ignore": [ "*" ] +} +``` + +## Why the distinctive coding style? + +FastHTML coding style is the [fastai coding +style](https://docs.fast.ai/dev/style.html). + +If you are coming from a data science background the **fastai coding +style** may already be your preferred style. + +If you are coming from a PEP-8 background where the use of ruff is +encouraged, there is a learning curve. However, once you get used to the +**fastai coding style** you may discover yourself appreciating the +concise nature of this style. It also encourages using more functional +programming tooling, which is both productive and fun. Having said that, +it’s entirely optional! + +## Why not JSX? + +Many have asked! We think there’s no benefit… Python’s positional and kw +args precisely 1:1 map already to html/xml children and attrs, so +there’s no need for a new syntax. + +We wrote some more thoughts on Why Python HTML components over Jinja2, +Mako, or JSX +[here](https://www.answer.ai/posts/2024-08-03-fasthtml.html#why). + +## Why use `import *` + +First, through the use of the +[`__all__`](https://docs.python.org/3/tutorial/modules.html#importing-from-a-package) +attribute in our Python modules we control what actually gets imported. +So there’s no risk of namespace pollution. + +Second, our style lends itself to working in rather compact Jupyter +notebooks and small Python modules. Hence we know about the source code +whose libraries we `import *` from. This terseness means we can develop +faster. We’re a small team, and any edge we can gain is important to us. + +Third, for external libraries, be it core Python, SQLAlchemy, or other +things we do tend to use explicit imports. In part to avoid namespace +collisions, and also as reference to know where things are coming from. + +We’ll finish by saying a lot of our users employ explicit imports. If +that’s the path you want to take, we encourage the use of +`from fasthtml import common as fh`. The acronym of `fh` makes it easy +to recognize that a symbol is from the FastHTML library. + +## Can FastHTML be used for dashboards? + +Yes it can. In fact, it excels at building dashboards. In addition to +being great for building static dashboards, because of its +[foundation](https://about.fastht.ml/foundation) in ASGI and [tech +stack](https://about.fastht.ml/tech), FastHTML natively supports +Websockets. That means using FastHTML we can create dashboards that +autoupdate. + +## Why is FastHTML developed using notebooks? + +Some people are under the impression that writing software in notebooks +is bad. + +[Watch this +video](https://www.youtube.com/watch?v=9Q6sLbz37gk&ab_channel=JeremyHoward). +We’ve used Jupyter notebooks exported via `nbdev` to write a wide range +of “very serious” software projects over the last three years. This +includes deep learning libraries, API clients, Python language +extensions, terminal user interfaces, web frameworks, and more! + +[nbdev](https://nbdev.fast.ai/) is a Jupyter-powered tool for writing +software. Traditional programming environments throw away the result of +your exploration in REPLs or notebooks. `nbdev` makes exploration an +integral part of your workflow, all while promoting software engineering +best practices. + +## Why not pyproject.toml for packaging? + +FastHTML uses a `setup.py` module instead of a `pyproject.toml` file to +configure itself for installation. The reason for this is +`pyproject.toml` is not compatible with [nbdev](https://nbdev.fast.ai/), +which is what is used to write and build FastHTML. + +The nbdev project spent around a year trying to move to pyproject.toml +but there was insufficient functionality in the toml-based approach to +complete the transition. diff --git a/explains/imgs/gh-oauth.png b/explains/imgs/gh-oauth.png new file mode 100644 index 0000000000000000000000000000000000000000..6d3758ace0cac834c1058edc801814d6dfc56c75 GIT binary patch literal 122885 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
537G(j1J#Rps^P~$<8a*2usteO!Nsd^IfpWS0h zLvDh~S<%ZaUW0qXnl>H-Z!Z3*S;i$?Mt_Th!Ir$2^pRWfpVvS;%ZjMCu@rfjIuW?T zV^l>t^}-bM-9<8ba1t|Wu_p4fo`K0~;|ou9l3Hs?t9v234&?NBuDj!G9)=J504)@m z5y^yYadK;V^t<(RbZniUSCevXpUDzU-0Iw0b#Poq-fiMM@E4661XUhcZU+}~n>aYi znmFKj#dRWPc{P>XOR#A)LvB*BuJ?@7`26uxl%I)TfeJXO6K|jG{79N&j{}A&*+UZb zLnTtLlG$d?qay}4UVJ(2AFPyLo|!cmor0Q-iEtn>a<8VeGGM{SXu|IS%M`@n-zi@Q zYVoH*zA@bS5D_n@6s{~wqEo*|!4NsMf#}dtknagg1r()?V5ufEA157$y~^>^J2(nM zgFYL5GE08r0CIIPfBm4>W%^!hC081t$MU%LjAlprnKE%;elpi38MSr&24y9Pek5Tp@Xwb+Tm-Wx@o_4x&kn6ITjeZkedEdH^j9&SCJF3vgm7VKJ zEJlo4hUlbSU#uW%japxE$Z4g<`18{)2mdP)R3J5oxXs0|MrN(6t($Tgy57(YXcGN4L!KpRZf=c~_=+V@#6)3S6Ll!1sE>hGMR9)B8So8w!iTrJ zo7>z6S7%tr(iPrBcum5L@;#1LEri(J9n(X{ys>U7=ri69e$W1KjJk2EqEVN&AYzJ! zh=oG#*{9XIoyPhkl7z%W%&6HSmxQS(+wH?s49n-1xnkncb)|R{}wO87YxA9lRXv}^YfdCB;Bzr%**X_8hG&xS{O)~!uhhd)B7b6 z(#G7rBM8l*&$^zTd=^~x{9bpz{LC%Uit3)|cJ34(f46S#x!A(M^C{PHD$;iz-iTuB zS!?mdinaHbd(ME8IEzf2+Mg|04avHQn@rl`qR=n19OQDB91xgk?XE>o zkmFOu2*ju}Zk1_KEoMtpL(5Diau_8>l|7{&v1-dT3CfLV%W}GX35Oa>f>XgT$&duEdoV%j@OHk8(TIeK?EbxjX1&SIP?hSe$Y z^8ia1{1r~NqH5}vgqcdd0lXjNPYF>}gNbqhnwi<_D^+O<^K%$^PV^4ya3(^Xe* zDhP7}z0&}Cd5_npMkBi`JmDlwF^G7f5fC;1MwNlG|~+$wwCFL0eFN_UK*0; zH@Bgc?=VXd@61k~81KFQAT-aP1!o|*OR=I;|HV$zl)0j5Y87auC@a4{qdu9rg*BxL zLrDZpcnN6vg>e^U-^1}AN6xSLp=52t{$t{HDKv}EC&8b15o5_*B|Xxu>RA*0u#iDJ zKD`x!aI-a0Z%$Yriy5k8D9=3$ou^C2FjZMLTX0b8lW*?fv;)kvVOGt{cF%_vk3zX; znFc>W{^7DC$fzt&w!h&1Lqf}7{9gnUPX$%!6GY(?l;i&cS$_lYl0fm56SLuxhfEe+ z21%%Yjb}*0Hg^2RtpQp~1GWepCI5R7dzMA+%U>Hoxyb+bXx{%?^na5>{Qnq_eTZYu z_+J(+@lsS&Q5Pn?rYb{z82+Hg<%u#!ImFhM&qE4vrz@^C0W>hF&|io(kwZ`Kyr+?3MdoJ|Essn*n|`@o;e+a$P?)<16xPpGQ$;{aQrc z-fs&vCI*qQ-`(Gp&0IMX*#?Fovs##+sXMZh41&k+>{0pyp4#WT2WH=_3Jd(KI_dRlK@w+g)n z@-1`nJPw1q60~hX4OfabhVM#`HxQ|B{Goxg7iQz}zKM zY}d3M?g<&Gsxe4t@rU*ExY39{7JlQ4xE7%HlI&jx&g?{7@3?@oYqdjqyl%r7_vL~? ztYZn_#9v?YCtbU@WUuk(JEi0%Q}a0|<(mGOw0q$48pj}V;UoKLSnqa_ev5BD&-(7A z`RHPv(^9&u_DSKZf{m`H)w{f7mjzHPAI#ssf!grV-M&EA@gTS1*^HeL4f+_hOGJwR9(+ z(L-OgzLG&n>N=(CCQr;(i9cx6OcOeZ=Uwfz|KUq%v1$ zbYsbm1BVG*llsxx9j>F0(s#xZc4osQ&XLzjWLc5AXs(0D(N{NHj3J#XvXYf7yii-H=7o>^RqJT8mzYPktCqgcZtd# z%+h`_$FERz44>)oTR26f$9S5by3E|T9-Vu8Ff|AKlUF{ei;&DPM%%kG;)s{beRYD; zXjtidL)xC2anNbW>zS}i=HvT`WTL?d!t#@#ns zhrP|mgTp%kXINX99Zd!Te9fHX8yMg6i(TUB!?pA#br6aTh&;Rj_Oe>2&xJg6Fv~EvAwxZF7RkGb4g6FnobdAG3YhbL z6KnfOEN(F)eRotP-(DT1$tq;H*-3YArj79^a``^HwGj6+FCW?28r&ax%3k`!RCAZT zl3LOwvaS_+RMW*2>7WjaBmq4mJUkM5@9AgMmMq*ddByTNne9GNqKt_>;|&2F-N zP1`=g)6*Y;dp)O(%tpj*T^8_ZG(FZvYV>!H(hcllFPZu1W6Qa;Y z4T@w|>P%b+O|BuWha9j>O9BNgTL?;yj;(QG8s9nsWicKQC%w(>E`@ii%#Orch|m5S zSp1dw#h{i_MUF?n(*QFr1weN6sqtfTc%CdOJ6;8&RHH_cqADC?T%QgA@3<_93%wH? zFxf|Gd@2%4lUO-I_4Sw1;E!PWjg)&TA)eddM5juaY%anDWd?yp(&<-&kY_{;YSH-! zTF8{Cb5pl|rS3P`kz3xO%X8UL9dy}<@AIXbloW9TDm0v)n_^#**JozQk&zQPA_#Le zA>Hgl!~;(>J9w!P;yM&RKa=s$%>OI+wi-y5-TbN&+e%bj+ zzwM!q@{6Yahi!4IaBB;%WG0fv<%`i;CkIjAl~;X!q(5lscA`Lfihs zf=~wD*BE*C-y#o0Y#^hFWtBaTXOd=~Be+!DE_gT0H~*d{(1F|X&yPK<0yXTYvUFGH zOVd|OPqy2@U4gl3m>}kkVswJ52MR+t@{c$9vA^v?7j57&c`TnR8NQ0{g4RrLMJG=V`Q+u)DNVS0Pp28!;@Bs01=Jz(rcm;W^OAM&69>f+YKMGX2)fNH4| zsq>>4%jZr+F_;+aKgKtaSy`gcS+lh_WwucuKVvo)H{oMGRHrMY1XmA!zB3`WaX$P3oDyS@z^eAfrYVm~Tg!d!L_WCcH&%$g zx({BI4pflT)A+OaqmYZuGSY1j&F7>Ez50vco`ia`d$~bsK4=~0KB_D^E}Z#qFtUHy z4V;-}afeEFqydYb?WL2B4kQQhBZD7%ou3BvIy@1>X5%!gScg9J^Ph>u_2)?reOL;F zZ;RsGk8Hy}DF{7~5gTG&2~jilf7J#p5~+QIzN$4*vCj+Y|;wEYlED2De+LMceL;CZ{{cM zyCu+$W^#MiMR5_fBG*2iKk!ph`;Ph}IXg>DHS#J<)%RP|F6HX%`(okS^s7a^_J;o` zIOieU8j@@w`0UVAf5UP8r^@5GE$5>9EUHZs$c&2b@CIkvW&hG9W zi^cMciPHb;+pF1b&zFU_hvp8_#vZ&S;d|MhIL5E%?6-85rsp-Fh*f^Va?Jkj zhrX-#7`}LFln-QeiB1VW%ykAx``^_|3R`=8YJzsTqkXTO;a<6k>r(8yPd{7ooH?^o zP)@)*=6x+_a>>pE%uj_OiKiwO>ZNwe0p{PHJ35xRkdN$fc*|!VdlgT%r5Y0A!o_@~ z8fZVKQSSQHo#G3ba9Cj7UmiejdEKS{^%1cl5tYU~BR@<2Ik6f%JWC3Sw9Q3z5PKjy zHJwzju@27BbkHm~s;s7FI2C_-yz{YHONl)nLLd??p_KB;{Hy69lZyD?vTYLcY@VG3 zcg8+&({6VI_5EsELKceI@`o|>_B~U%sT{AaRGOtI*q z{TocT_SRTa{(bcP&n1O2U!F)zo0d&^(oMNOnKUf!@z>zC-EL1{9 z3-|BP95k&V7KG=>U>zZh)c48bfJz-rHb@tess9>f56dWrF*;>aY=(ZA2u)MB#mt;> zs_tr|V!CJgH@p?Ew=)iO@esye0iG?K-8RmJ@u>Ak85a%c#!iBsvWgK-j+_$ZrtS#P z4ctm+-5k|5LZJiJmr;r`Uq_(n6d^F&tjbK5uCQjw#(^-I0m=kH(>hl=%e0@Fxp9~; zZ|ca%LI8aLu;D>HzoKLTN!L;o5{Qe9oh4VEOzw2dVL<#dV!xMD!LT2ya&#?3|5pfv z(dsLKIQlW4YCH4`V*e&-gs;;zN>Ed3z}FUswJ!(;vtk><{XbK8`ajOhpjUviK@q+D zHwM%@g!=uq#F;L{Y4TC~p)}{tPE)-$jR8=fQZ;XidR@-dII`NGcDrZ3;zeZF9-GD`h`4Sb_-y=s@@+pBP zKMD#oPQJRk2p;qRVqfo8~o=KUPBDXyJz0$68&<2tjKvI{tQZB~C zn0*I7IaeWskxpS4*5TJj=yWo=M&y}TgFZT?8hCTU+;* zO~9g+DZe<>1qMoT33@&hI|qfWK17s+K^M{PB=FPI$2k2fIpjxyIscvu^!T4A`cNY5 zBJt7OVMFw@es%9$kqrI`- zaq*Yf>i*q~3x6|z>ca@+PdMvTgAI&4?KqVwH@^ z{;{MV6huSIBC^i=*U*8Ihh-MM0|)-5%=wPsW>Uv9TYfW-R5*e8MM2LvlPUg0$9L5v z{My1jU@&~E^{G#fzK2K4mR0U-OZO^x8+fnIC{}}!^W6HD_s7Sp#_i=-Nb%H$9MpKa zl&t$qDWO@yzX6`^Gs6K}>otsQN@LVSF1|=ryh5oA*9U}!PnphWx;%IL?}^CKp?;Td z{T6BxR-(v@?%L3++@o1^m0nTir<)02dRh}dDIdUOcs$r!iWKc|0wsG0e7x>21GHpW ziE?sa;YB`mT*a~;>d&qh`C^$uRill$vQ`&(DFB&g8OYB3h-P1?s)gEm%|o$E2gCe7J$m(^!k%0(ScoxxGiD zdN=f9%R8}yr(|&U4h3DS_qUX#-FVbsIA6(91bgC zm_F&g=50=7OcrKSqzmh8bX-I^E|aq7CkWR#zvL|B*#>4KKioeQgPTupsn09s_h){(WJ*@!0l;LCMcyML@&5PepzS(4%~ks*}v6f97;Y5R&wXyOeb{cNk_ z0g*Kyy2(m#Tx#V1`n8)IlvsP>i)OJyu^#w|+pR#|5*|gl=BO@d{%&Hwae?~?ljXy$ zL`XIALBGxyir;dtZFaJCChlJ)!K-ZV}N z3!S#yDj6dc0vWDm#P!AB-fR|$Z);P0j`X)63sI>kUkRcK;Xerktsc@EiVU_XT}{&J z$JPHL+eu2Bs2$gHv1YIXj106kl!NS~-}A5qtmy<)XdZpC=C~Vrubfp!u{daZe=tb) zuJH`GQBj!I)f2mbg?oEIYNDq^3Z*Fjw8*B>DfG3&|6z_2AY#U%kCHyQ zRO=(}A9lS!a0^}9J>(3q)Jf^}?mh_maH}%<`#;uqWG8cK?7y9AEVErDONw?nM`l!O zZ@lTpvJxNoUOK$M@MkfS=C`zqBTT(b@<6{BnesWU01;OzyEfZ5zaXhwCzNx&6XtL1 z+ITTOk?w62l$OWS1|_+FKp|jX3K%dyl{`Wc{XYL|AZ}WBJ3$_@*~&@N_O4p8_{d(y zcd_W#Xh6T_n>t;VuGB`w?Fs&|j1AONnP{7qx9l%(FZg(_K)VkGM%Vq>WSmSc~8K586BU<@Z z-N}W%Gpqj1hXh*LclfcE-8?o#rw2!o^&O+2^JW-=K)i)Y`eJQQ3*hf2|)Xy8H*U6j)?bCuKD1X!g>wX^lkH2=`v#P z;9^LNc=X^@a!tX5DYUxrL4rL zUo%KMgllX+L#+#Ex9H=zZ^Z;KuOOTm#(a854P@$zLxK+QTbeq+Ud$srJ)rU*8;Blv znyipk3n@TiCK#7C1;aEqSw}Ba4PFeVCf0BQSXz0mOV1zCjBYsYZ!wUZU^VB)d&ZC4 z-_j_2-~*9M*%5`mzzDr#%?dk9wa1q%E8vo|em12?>pDSzYgq!hKe$w^C-}~_UI78I z3C}2w8(o?!UIy&Dg>;oz?)fa|)jiAO{?6$}U$;DzwY5_s!Ig*t zI!BlVR@~3&{L2uBaX?ZZbe0+#v7u2u{DZp-yi_sXy_nL zdcoz}Zety*C`#pJl1ZIKVMJob4{jNn)5tQMO8@`@E+pt8A|e|$jA0%`J12`F&r3*v zs94kSPHBdED_J9<5c9n>+p&IDB34{;nK$a-#X(BWu|!9_=0q(ak1*3DRQEiwjN^vIrbB6gEhA8Zmcii(LhyCF_-`&tc z^Ou@l72dbj+eAwXyXEKJ6Jc1kQO&GB2dxJLa$cB`11X~)7CIqFW@bT-PeWqIE6zlG zmaG`$V-lurLXAJ$s5wE@B7MM_eb;jgWT_Y8Wuq$arEN|^LWplT4*-i?>sH22a({s# zx7V4T&(14fk=803{fL@gkhCya%kTKl+lnj>BRymqQJGo!iuEi; z_z45W_#cVRbH1~Dv=~(ngzLVq%5G$*dUa`FaxaSUR5|LVesvwo4OE#2qkGM!%QQg`9ljYzg|WM?aA znozb-NBdFP%2GGenk2mYkFTTUxZY37g}?)UWpL4#i{|&-ETMt|XzGMQZ8M>x5^bM}j_5`KIGS(c6P}1rx_nURlCDyhm6V zQHP3(3LoLTN*wZnd^$$_u$adITr_tJ*k99k3m15CZ1rWM#6Ouhy5egm0>?kjHu4h? z^j7e9=ZZJL%q{`B0|08ThKn4Him9Amk5E`d@&~(Les30OM`om(va$8%;U4K4J&`ua zZF5uu-SLup1V6s^vng>@Q3@0A-`ECA1e4Pj{4nyr3|6x9=Cv`)X(;G3tEyRH0wtDc z&Avul-H6~vXjxhAkApDV5c#WWPKv9R{`MkTZ=uuE+P{I{&7w*a52!Obb+VX!Rz*y4 z3`A>;SWqlZ@n-`bBQE$DrX|l7G9qjC8y(!rUbyeW!XlQUux;C}C6ZDF9_^cQYh+x_ zenrXgI8@f?RgR4FQd7|&J!UZuJ^J6CzOAXeG9taFMNn#F;jSZhSunM*LSbaxtzxcn zU*<20E!{3MSWcee3+3SE7Iy4*B;!s*`%b4s;8Xo2JWu1hTlv+KmoMQ5#C7L)S$j|8 zo?qRl=ctP8S*0~V#CgwEWI9Fj(sP-69Y{OF?`>v$N=>*?8h7yPSk<0cLSqk78q^PT29|NwyU(zglZ}L*V0grLeK#0 zB=0$R-sX{D-cpFY)g3{DgYTE5H3&5H79rxX#+floGdSf{Etmc*!vQ>rOcShRPcmB; z>aT5)ZI5r9rJZRd>{qcMx}ol5onV{W^I8@YP(}%c5RDwNM$8ZkTLLH?Bx0w6L5pK{ZkrNsY3g zsJ2JZ_{-LXV|HPQicOjyo?l74XJ^Q`-a$tCL)s*%p`dUw=5$*+H&$n> zYWPO)a@;fGm;fC*N3`FI%jseg$6RDvCIVzVQpy9)^=hq$h`~M@;pF?&OYSFH^o1{v za(H{SR|CIiVnuh*pES2YPK280a+fy}81x!&ULPt@5;~dDwD(u-v8J|r9o7wDK^BWN zJMa6XSuZQru%eL7*;j9G4@v3=-E1Hx>9Dwi?>0~9lr(-UO_sT`iL)Ex;uorz`42Tl zmg!D9vakuWWux_;Bc~Jbds3yae2Oiz?j;IxVVe8^QW&(YWAnD~=kNL`Mb=m`kfjbE8q4PLv5@ zgsANH$A=F*rVI+dwoeJIUybCdXX?FhzWe09UFDbRo^u*h*omQyP9V_VIw%8})UucT z;)Awy2-KK7HZ;)?Eo2Q_`N{oqx{Q`*Jc5ZwhExs&Qo342e_gAz91 zt#w)waco9cc@%V;lB073Mx!NpW5(=8i({iUlwCX69ZiiuK!RD{^~L@emaBI&+0bem zf%nmMCuC*m=(hreIJ$HjHxz*d4I-i?ft{LKrpV2Z4?5~y3^=Y-Ve5BfVbHliXwnkC zngcC*L63(>$^XUXruH8;H^2Wt9WHU9knq1pLe76g(VzZX6rKJ*GIm^@%;8_)B ze_3F9DjcK13xUsF@^6n_D5laJj60FZ{_f4| zbp%gz1CDz_(sWC^V+o^sPc}^RPd0w?l>+J&F3c{b_j3>f4Ie(jVt4id%)v0To1iA> z(4OGu^Hfq5>la|VaeGWkul|BFU*3i{djUX~?!SIBDzl~wHYN$M7UXy%N5*6u$jL+e z@w5#weOPXYE}-C&!SD5tq6?)TdukTxFqj_A1syNC6q#qWy7cATEr#f}C^w_-i9rGQw zh3_|WQhMPa8aJue7{L^z_HUzB%skKl+-JjvNg62*g+@fZk++%flIQJH^{=< z+~oKAK=t>g_x(il^Aw5B;{-Ck-B8{>1z8dQ73$%9>XWu@xxwG%Ljk*W4Ja21%Mpj* zt$Fy14qzxtr`dPA37`{uT!gddrmls^c0FMG)j;O~jUMrhd@ab{$!2iwfpp{Ryb*t# zo&;a2pkDM;={&itWe{cjl=XH-_IsH6c<<8Vwn872q$Z79k!K7R_j_QtzH}oqxX|_4 zeq|k8`jD6)D2&S~P1$osc@pTLk!w{h6f6F8Y@pZRz;2YbiXL;cXX;yu5$AuFdOvpT z=g^?D^_$Y?BV6fR>^JWv2dkdth2r7VP?*6pMjF0nd@sTUzbe}>tVBMG$ZcU^+EY%ivGnB#O_Suy z-?+P(fU8tS%SbAagEnO(yx>KS+VyT94DB6>rx~ne^#xzg}imOr319K;|k#Zy2$g znrLi=VAOfE9g(=mG(Y1f37&z-Z>)MvdFD%vBo&K4;)@c~4U#1>`6vlN1+me>O z)eG-NagpN=jS?dpmp->YuD*gX=HL9vob8)}`i+omgphpMdvC}2 zdPOd}u&i#{M9b!i*ZFX)6;jU568xTrU?7Sj$5^~YK}SW0r@?T^P{!Ih#(kGMd6-Qr z!^PC&@GfV=K>9{c@J{|7EMINr7pLrFF7+0VKfUlB*MaP4@7Q=G)DK<=ITaGrOgwPR zWFwfyMWY5vMX{)*QpzilWj%}Y6c59Zf<5vH{Gxy-$g^zB^`*}f{4((AL-18@%r+FP zJbMUMykITsput3{w?I;rhj5KAC<8kHQRRHi#xomyx#xSjb_ZRNid{j0_Qno#kp`H^ z3z!n&>45x8eHT<@!v6QdK$NlE$CY|mv>}wq4J~WtYtB_pD%PR9`eoBlX*lH@?KK$A z$b$$_Rriw0BGLH-M+v@WJ#VVnRgn)-FuFn{QfF(FA?vpNV9G5& zCgHVExsv&pDOu)K1x<5CG9SD^=occ_Z*cFjo1yoRr=*qGYCfQ}-fv!qAWfIrjq=)$ zo*rxeO6GB6Z@+;lZh+YG9!b>e>n2Ep&`_5sDWoFsgWkm>WrFTnNVnbg`S$A_(c=JV z9tLUH9tBBwR|mxY=}-pcG)oA-12Rx>_+_w!+V^Y_Ge<0@DYXh5=huJ;9$bytG>4dY zo7(O56-JOe!TH^4a*G)@!mK~6dC9;X#0$6}7_~C*myK4S9|c17g3LYEyb+be9}&es zo>-|1q;=U4Y6?%6w`lV|^9rW>oY?C`&lE<67n~Ts$YRCTipeV%?lyBkXZh5|ewJjN z+(Jly4RAgC?!PmlNq=y993}Xtf6Ms@rGGemIQ_<)XU)U!vyFp!QcnNr&uEtOn}0v# zk`ajp?}jL6Y&*NSkY#|_)?|znhuc8UZ#MZOSb;1l@JkHI-*qy^aDSs-+gtsolfkY4 zaABaLnOtQ*y!o+%&)tJOcIn=!%D-i36yK)&wH401&nt^p_6hwH?@;@ryVW|sfrp>0 zW}1{tq80kiH^K2ZP*oM^F-aII34AH{ku2l4%NBX_Jh008@((|JUAQUwzq)__)7TdT zxe|^5AB0RF1TCNV?m#)sCbJ}H5+schNN| z{&l6D?}L)=6GwNEX1?WWGy9HXW3Qqhl(JPB!WHX(1PkZq_8m3$D$d{KpvA2=AUgxc8>jSWW2pUH1~r+Iw#EZ3>d>tU-wIVi zU*eC-QgJa|El3qRvR2>m>U1$AS0o-H(n;a3fsTyQQLK`1!UtjhrUqJw0=U< zG)z-1zs+*3TUzpYRf1lh;;XO{0%zv84J1Rm(3I<+&6_7oL|JGQ7LPHYyRQzn)8l6t zu*65cDa2HV_P}+9Bn^Ma70S4zVY|k#)-){UX)*K>MhuO!s6&4Xc`qb+Vdy($yJ zVLiuwL%f2AKOXif2|}rGfK;Yu6bf(msYCDN8b+#tPoY8@6jrjYN*$qgjW=&?1Xq8h_P-z(u3{(CoGcZEP5$(m6pwx4;<=(Y)_ ztxjno7Nx?V9(|Tz1!r9(<*fUn9i`sTvbizL$>)61Yo?(gYWrT%M!yrM_>hfqB=E4k zK)_+ZW%24g*{WAJVf*K~_hid;58qvd41L+qTh%kX;UG@slJg$}`zfpBHG*L=>X6&q zEWkVH%}wSbgKpb;3CMOUVx&415_Y?+rzL+)(nJhR^opWX`}a44`;A%jYvZJ`XAPdb zskpfF9lZVKJBaDskKcNYb;G)&OaIh~G?@PXn0u?JIJ&4?Gz1OqG}b^9+@0VK!5snw zY200cyL)g8Gz53|1Zmve2^J){2R)td+%xXm9pn5@_pL^c?y9a;wbx#I?m6eeDZ1U~ zF}&?WQD437GrK}6vmjf|4atwqBmc9o8i;u28~JS6wWWYirm?)_+5X%0H?_nr>DiVh z%c<9f{8n8~gViO{TDGd2F-F=6KfJ!Bb44{f5u%obFOSXVfB%S21@oxBh!H~qN?{yy z_xcAO0|jGx>(;qv&qcgIT_Tw6E4p~WK)m}q;^=}TSvB3&z@9R`b_Z5n5=EzTD~hZs zKuK-57Io4(`!RJ0S$UaJm7mQV93X3u-|<7 zKr+{o_dUH0sl94=vB^pXvlPTmZywKesamT6i{Hm3^`Qh#Z!$vADPWNhgeo9x&W2VkD% z36dH8<#;8u>F*7wQocYnV(avgjPqAoLorH39n~LHV_L8^MW^K5emeL ze*=$!!T)LDO@mUD$c`*{$$Vb!j3#|_kz9J=`a+J=>KD5frO&3a*uDdxo;$}b+&s(x2?bmT$9KS=nOv*EV{K5w+ zjH9TIhvELG2}pHMY%S2#=`PZ2h-Tw(Zo5DJnO}2#d(|V;G~!`p#GIwK7;!LhitDYQ z3VyV?GiBwOo8lQMQd4ip%(3v2Cj^1E1ZAI>8W`Zs*sIA2!yR?@fY5jiY4(_YATBxD z$1udMVkMaUR-KLREJ^?L1dpTf-BvW+P@Jqmwn8g(k__dOhf!$E1prz$BWQBLr(V}XXBAfm)Y2e&g| zolUNyn*MO^wjLuUjc19`26fCJMKs~;nY&xrihAxR;_*PuYgaxywXCXXB!}dvzdQYM z88j-9l8zV`ZzjZ2{CBKx7H8Q77EUcVAYHivs`IfIYr9HDh$TfE7RT>GFV730SBJS1 z@8b6TFtt^}X^d>1F90hxu6N?EZK6hibqU6X!eg|&jUX$uf!Fxz$eo|&Ea*eRL=T>4 zoQzE=?g``wkjejMGSb}3dqFItp=Db)_wBnr+U+3c_34gR6kq#WH@6tg9!>4otN~uH z>=LI*0pIp260FFJPJ2#OonRG_<=Ef6>1YTkPY$Fk@ z|HD2jnENQ(=jO$7o4WMN#zm0po?gCZ*z*gjg$Uo9*<&7qGg)rtml~b%BqCxK zyLoY{9!2_WZEe}XE5hA3FmrN(+jP&y!ESRjr=PgfZ71kU&%-46+$}Noi?OqV@mr-d zm8QVE_@8AwXQrP$QcDK}Oy;GG=PHq*ws+vp?l-S`Uj`P_QZ4K2ynGM`HuwGP3B8-s zOdY&ddc9>@F{?&!-Su#k zTv8Ogzw<9g0atXEcZG?>miU&VJO`V6<2nW9bz1?U=!|i|pk~y{l}?JoOo*O`DPmcL zQ$hOyeo9>k2V>uXj||ZQd?7MP-J#fpc^^{#S*cqJaFS>qnTTYP5JJz1*Hx%hD$>bIw)zR>$K$@>l6Ny^s@6Vl zDE{FmQ$GUwD2cermrN(ZoZhx-0rV~5#N~DUpk%5}AKC{5o^3gYbsnO^i(@r1NA6$M z7nxHr+j3%V_~8C;wEVaXQ7+6rkM>U&eR2HuHsa>v`<^d}uEn_Pbn)W4`#TPOgS|_- zu`>%3dkKFcte+tT*st8QSb^%AeI(>Mhq)>gVRbw8`8QRXZ^NZWRkaP_ZS_B(D)40R z+^#`GWzciykjKMHre~Hz%v!|pl8#s+Y#rt~)>XTy}b2eYOSVo%fOIARjH*VYB!AWB0)wT#nM`Xnpy)jw;D{IEu^ zj(*YC1h$yVE9Zn3Sbm#e)WG&u1jgP^e6M7i4B@oEu2=6&hm=p*8&C|5#Heoi`UpJz ztIODsYiQa+tTj=G%yWsnhl>(B*1L~F+UnjxEz8-@Lb&$7de|wFHGaeEAV3Y{XYM56 z+~i)dOD9)sS)OLq=xr(OHdUfY?)VNdk~8W@X8EbFpiOl=ToyUq8{j~vmz71OW6LT; zmnvSR*$eKfr|69=5>IqAh${~_o6BW{Mw;A7P&X9jHzP*V?@3$I;zRsstcZHVKw1*+ zUH<%TJtzwXEHZ6V>b`)QNzq`c+2U=V+1{j%7rbLpqvd33F{U3@b>@lDw}PIBd5UFW z$4}$7oI}!JwV~?y@krz}LbSCYnNS1Xp(yfiChM$Bw0_TOYeXB7S~YCq9amQ)bX8V_ z^W*0+zv=K+RT&lzeJgC5y0Q0Yl18Z91!^`vhAj~uUCkzmGq^sUj=N^)8>dG4>H3=h zUA`kZEQUSHQaG7(j1b7z1$^Ff=w3idSxhN({%g79=O7qHV;oqp6LjY1GA(2V7T`?B zzYTxudp&F;$2G8Rx!FOaht0-9ib|iM3D#9J>&jz)}2~XO}PE#TC*x8Ev-nu z>|>NfIvzLf1n4t9?*hGK*&5?yN}PZESa`EM@v1U`@^l_I^ec_B{L)P^1G^L`4&qHhqB7n+qGDwLT z%ZvLZ`W`dDE>Pq0{BSFVz!iCW3l z7EagJiF1AVT1U&{h!K0Vsu82ITr2i6PBCMJs=I>+goKcSD0UD?0@rmpdQuz)dm;`L zNDS=~s`8FzWE=4q$n{Ak_{eDsHo82*?6s>m9kD2J%7Naq%E!XMF#7NCmY69q4Od8O z{0tJ_H{!2;q%d&<@cJVRB{hi19X#Ry&zK!RaSjR6hZ*7kAB5m3PSz8g{RD$GZB1E} z_x3!M)uS?60SKR6r~}x0LG%1>yX+~MALS+wLb~Y|jbBg^AIuR7+m5*F|MYVBNQ`L_ zs)cMk@6>%+w(c*&6f5d3L?YsKA;jG}N#6C*?h*g0W`#~ew1*~2$oP~gHWHqX@>B&C`CV%93kN4I zwZ3$Uk7#Bl6n>z?`R-YG-2bjo21{Ifq)EXvXdTH|ZbPQ>-i_tqRw1l#072SU2m+>d zD&9p<&T|N9+QrYxw%ZR$F+||fQ0AD(;i05Dow+692ccE)ISeX4s~UCkH=LnBkg+6l z#6!|1{}j9+As|4(2j_BQk7m@sj11PivcZB7+G}Di5~M0o6PCArj|X+Ocm4-tN{bMv5f{wNZhsEc%j?5|{lq2+8KOkggy`XsS4*pzAlk5SqLVBW;RsM~`(7oA70*zL zP@$5vX_E2VVB_F1VREwRE0$qecY&w%v;%R^I>*(~Gr8YA?=6q~SEM*2HSB&RTEGVWBBBW{+| zJtNLe<8Um1!{I#`J-UMTJELOI%(^8Gj~n+4HBa|mvRsp#GODzOG)HuWlGS`PHjg>G zgmmCnM()V?_~5Pr%|=Tm3Gds!p9hpu56uw+HVDRf}Bd61n z6h;(Ny?>874f>WRlALfK)b)#%P5S$C&~;BCN2O(B&0BDdNC@5nzgeiBn!1HV2i3bf z6>`dElffo4+$h4jD7C3Fv;CGB?J{XadE8wV=ui!N!KX?Md5O2lv$F-bChfH&Z%u8m zt`#F?r0mco++9k!dlLC=)RwJeRjrB~Ur}RYCCzG)!iT(yWIsATY1TRFBa~qFO4U8N zY(CK>+TO^;^X9EyrXN$J$3#Zq%xdS5Hktwl#H48Yshhjt=tjz+LMu|Fu{}e;T+g-n~2|j<;MF%_lnZ*AG%$#`q`UxlZ zdm{JGmkTJ7xD;P+*uZhq{c%3>oscjdPRl+c&ubLw+{`Pp)Y4o8JLvBSYoC6-&eo(X z=whb?*{~1~tsN)rp-}hw<)h*k2TaiO5U)K5TO1UjhoI$%6{9q`uoV}ADs(YqCIwTx zz_3bZkmSXT2!zzKcY}cZN6B2kSkbFtB;@tW4HUW7L?0s<0xxx7FQxPq$#XJi=pXPl z7Y5VBtg~Z)xx7I_Om7+ZHa8zms1{9!Vu`1ukw4#<+mBER5tMa74e>vgDfOn*K>H3X z-=HR(Mb()WPTZ+x5RinlVmrXN^N{LL@#uqaYwYx-=+}4Zh!9+G_1}6S_P0?%`T9k3 zNK}`!?ejV|adIQ<|5oc%t=s;aB(wak_2bu?)H|ZIhFe*{EP<*@4({32@#qk|cKA{oF63v78 z1%;wU2n9y-8#gceIr92MN{XquHsc11RphA~y1FNdtOkz%-SIj=c+daop|8;CSIjRl zzb`Z-dV&z8(d82_(~XiP|9X;^5rfo-D*gEU;osN74lP-={#%b;wj0_R7kf4irQ*`k zqv<5Sru3rhD+qUX=2ul6+j=OE9b}c9F){5#{kr3hF{E{yuKk!-UEZd#xg-9&emmiZ zgZL?fTKK^u2@dY@=o~*=)mJA`DP*~GX;k1n_lCbCFE|;`xgx&Wn|#xVBKnJ8dry9n zWHm+~pZ_xwFeQQ-#q*6HXWqjh#YHVAK2AHS(5p$sV9D7NJTg=G&iwRxK9Pzwg!5uY zXET_YlU%0z@!vL<0}0mmQ-nNiGTk=xiM+y#DQrRiCksr>G;U+<1l7&a>II~nu@Aw~ z3-sv3fZ$F^S+(UO89VQQ>oY>cwjur7->77hB7xz78-OhO!6@oMwb^1Z0~rc*{5}IE zAB_)}6LXRdC?RUAIGzTWT+zzAls_7Q9%UqD@Zk_0#e8~;Pl>ShV9^zdR_?<%*$ZfX zvpe}Iy3TG_n8*CSNbd6(OYI&6N9}h5`3@YV)nl@CQt^$uPV`-PL<9g0YY`V^qJoVh z7z>H4{pKKSh%(G^*ErJd}Au>Kw0}(lNp494>Pw z#=a@0`9P-XNY_<0S6-sQiIjZkcZoBRkVIUHw{4!uKq8XS3CcS+xN_zbcA1g_qhgeh zmH3vtwqig@{61u64fC@cb2$1byvq8gV?ig>ix64yRVSnp+f)Fj5aK){BD-}KHbw4U zhm_;y3w)yNo#YwHucfUW_76OANAnb!Q8FCnXty=G30;GzYD>6*^9pGY$Ia;jqAhWU zj||*t#-U;Tl6(5H$mN$R4-0kk43fHTxDR(H;cv`kv+>E-qx-_tVLr2H_R@5T`qnbF zx(PkcGX*IDbtXBmr+N@T5HgpW|MGsp&QaYMrAjRn}2eqICau+qWNKC3O8iseX)z9PHfKv)3E) zwbel^{qyxR$3DK-Yah1+0`ZYg5%U2r1TXDJZ)}t>53W(Ky^pmzAIDATm#$e2 z?{~mM=5S8>Hq)h1y{4Pb!iITPKa;d#YOMp>GK^W$x;0|9%MuDbee)f^h`0*J4MSG8 zG1M*jy5fwU%;A(ZvncUiKf1}7nJM+w_&+EWg<2a6bmMP6FT)85xnfToZ-gWZ;E|PE zDSkP4-P(WUBNkXTXQGstI=s0$|MMdtFywn2%KQN~?zhkZc_#F-+`iV_A;TA>m&)PPiC@$68f^sPL7|5(--YpcN9H0Z#15I%Sv zSN-k7^Qxg{+jp-o)F9V|EsD(m_`?K|ce4XD{8Ys82}B}sH?LnXeBdAREjO7FqE9Yf zHvMChmpV8Q2J23&c(Pbn4+=HTM{Li)HVL$LKKqh6WmU4-J`{vqNyxJ?T~i8r5iKt- z`Up4AAG|RL#Yd)Nb1Ayxb@r>o%AjyQRHTnXtqLG$@dixI#SWGy(GGyw z(c7nE+BBWX9v^%&oo^Oi>aYh|Nb8+182QKL9Nk{^EyV-Hzt#K*nisq^**^*Z`^urc zT#Rh0S-s>udaPvEp4NyC)}-x%2U=d^gk`aIexAnN4`1hXr_aN>1-u-c=>9OrSa2|m zXZSdNe_x1Esxi1-7SJ6+A;cW6kQpApsu!>G;}ykiM1k(CbS3C4VE^+Fl5jYW57v&; zgun0YXotzhBS-h^24;?tfGgL?x-f?)l<;`7jcoEp&A2a~fWErvbKTIwgT&LHZ?EAD z7Dx**)~B#y_tnb-Me~Fx+@5hQZu8j`Q62b~y!li&#$Ay2*B^%O_p57RrSxpQzo9Wn ztBH|6D%WC~vKLGbe*^?g*V%Av_&;sa{TO^_?b+F2gM|sR>U5`gChU@!-!JhyjwNOr z+YFTyd<{llpU-Y;YFua$_8{kvJc$X2@h|9UC+GV6uQZZ!Y_IpL&d%~JN)?zYbP?-h0aNvuziLc^NA?IuLGH9lz6 zT8LgJeQ~w_X7q!Je#&`ho1RVGo*3MkI4p&!A(yF#jEx1MRINw#g;k}alVY>u&kmZ2 zl_ZMc&Es(mLVwQbMIl}Sl$$i-n@Mnh&cfz{lvxzVwe2+*ZMB+n(~3}z^N-{ zAL1Q64?8+<+FA^^$RQjsoasgb9i5e%&)fCSdfJ!Iw*QSCT_fLEFgfn711U>s+)3-;(*HHTgdm;CN}!(9$UH$4`! z++j&t9dS!++m@OaiAjG>hDMU_D8O!HFhRel0V{vb1XHn18Jlc@Y{cOM+8$+@CQ@bw zmVm3gq%`Y_roi=aUJ$S#m#S1BNk%1J!o>?ec)v4ppX4=M30Y?cs&&P2pJ#b7C;4?+ zT`QDChaW^ zda?)?j~%;tFZ3x7p+C2c(PBaIT@p_nm$Bpm-^qHi&=#k^Fxy+*0bv%Ck%h#mkP(Rb zQGwQIbq5)$JEi6dC4WW5lpRqR)qL3Ek;|R&aN7EiuM3s`jVe;gt;|y!7|n$hOWnZ$ z)*}>~%o@RtZfb50)N2&Urq~ASLB7Z*mYW%9%#rfr$72yDy3G5m=_$7Ym&Jt@1O<&k z`b~EsOiUG=IQ6Jljs|u*rrVLI(&Y1-Z!Z{ksksBU-_L&SpLIq|^>eS;qY#YF1&(A{ zaHtMx?X#u&`u8l&AGOvBuaKx3fbqoDJoj$-`Nqt9LF$w>HR4sD-uF$p{;>-0bAV0i z@7$^&^P4;&ucrrRFaMx$(3O(1tM#mLPbWD~XOiQcFByI<2GFo={)$1*&J{&*Du9Z(A ztcez}?fzA`SZvYWa{EX#5~B?M)5F-hax16f&5k)_qtUml9V6OkJDWDyhW$(K}@8aanuR)NY9wf?9XC4{X)}yqNcAE{W<3M z)~p840f(2(;B8`5AcGF1`?l7qMV?!?rZb^x^4@1k)&u|bVbGV1ty&7Y{8jhwRtLlz z5J=30EIFwB7nV#TtE2a`J8IK*Nc)4D-fH+SVVR^~Pf$wV|LS0ux_GzXxSo(M6MG?C zv1s7lpdj*(hgGzIp8@+OKfX6kgbJx8NFUw&Hr^#$yKrBwi!?q)b@3!$+4hPWn^$HiLhZ{j9NegN+ zVq|1B6S@ZSA#3oy;Qa7^;jr_&H-CC*i%M4(spa#(S*W>qSWCF*qdS#uP>9PYQe1ni zI@|iA;k{#F6>IIaCAb-JWpuXl{jr!@sGYQUjfddkozL{#2Swu%tEYoSwyo!cw5{P! zvjwQ=5@t@^7Tx#4^t%qbXZPr@FJ_{4KtB9(u~@$s>oH(1rc&u!(Z^w|^V`mZi;kL2 ztO}{HNdK?r9h?4>HB&^%$p@>zgjWy|Pc=4r#{K*5*Co}>TWqGlu| zbxSo5F-kZYr|b++yuK1yUXME0xKuDlf{h2;hY_P$QPkqs*M-YgjG0S$-?)$tsZ&ab z1=e*}j|(ezl&0+#bVA`Iy~t)jJoKzWaRMs%_1**SDcbAG8B*rOmF0#AL?1wfxY)!v z=uMEqbDN4`Xx*hpQOqs7#|ECJQCds`E%D53@`R}8yL56bD3rd2PqGY4S{}?V58K1m z2QsBqUENjmvlkz^;A?T`e+J1$a*#P+OskHr)n?eJsxmK((uB#B$jZx;xa~XGT((Xb zXn`05<7(T?*<}VO1vv3WzIbJ-V zg{XegWvREBqFL=*e1A)uGtLxPh9P&nfV@?2u}hdDMxGqppFz1P?CR76n}toa3W-_E zNVbydBBR;GroS{~mR;jdXnvfsbn-MBx@y!{({>|Fjf2~6OUW+3{f9dErADE{%uF^m z4lmEk`Q&o*BKKBbD5`I-+V?{=cPHx(zhP0ox!&Ae2X0W#b6#WJ|JrY875+CeyQMZL zp+5hQA<9#~PZ5fG7~#9aOaN{M>K;#cQrWfKX=d1MDgThztNnh9tjqhBe_MOg0dQ<5 z|5=}aF5&;rQTx09snFp9`ODcGU;?szya#tCOnrIm)NZh0DK7On2?4bY9S+tS`fZ~s z!(O7?`+xgQt1;r`F|KXvrH#nvj(38uH)TqL`XGL=ockZ{26LT(eGY3k|9{W&dM5gy>TwKECgAI(0XFfb z?A+IJSv3nfckrbArrNh-=~he&m*3sR|9RxJrLbco?r{4~ldomVqRozFNW>StDUI#7 zXM~WlVEJU8n12(&E1rX7_TRo4)$l%9)TtSd$2(V8LV37kmz#D^rO=BL{ZaRpM#-pO z+>uGyaXrv}u>6;BkzJK&2+wG9!4?|-Paz*CH0@*lC^ik-KH^Hl)&J(wS+UEHbIZZO*M< z8-i{<)sAp$*R|&*3}?itvI$nL-^feM-H8 z1Uy2T;1{GxO%-6*ggN@S{b%^lb)NKp(uUA91t8demCE zHb*xK&Wf&L4nMlJvkd)v!d-{;z_?=FShR}4(>2Bn7W>FkVO-CI$xv;Nc_c~uQ@F~_ zyf~y1GB~g>Ug3)KFjW7A!b`LODIN~xd)10< z4EoB>un`~N*NQ;heG88s3zy@8Hlku~dNV=BPvbIVX47wR-M3^Sj9ql$AF2u7T3szr zo(!yf;=lYeR?9yJ5cESra*GK)s-NXy)#tip&W=c4TJ|zC!(U;!S*o|FS|a3?Gs6un zxgCtHgNG6I7y5xjV>bj#T@LVCJt>H zm+VpENBbkUo|8z9aCRJetMS~yuU61(M{TO_`;?00M1ffTuGQncTh4lF-rLyCr{l0< zh4F10!i6xh@!C{3%7Wa~hH7-v=%U!7=leCp5BwUo;qztvwZpUQim_+ri~`~If|vzG zP{j1vkfl6H4nG6h>!*Gi7&nzuHwjYVk}*>IV=1Z!9*>f*7^G!8aQFn`pVaxXW>v;C zSe+XYbT5*@qMpIdzE`NnjO_zm;PJ3~uVKuLxOL~VF$Tai7rHdv z1mtDy4^S}r*5PM7E<^o*;?LtUYGb>xf$b*bY0!-J-o?7EM=41rHSK*G9Tyd#yZE+o z>pt9E*RKf4`cvV>&&rP({O{pytYse;Qi!&H7R;i^pM(KHIU_WzMx=TIXQP+ZvKvu_ z{LA_MMH9DE+NT52Cv?VyC-Z)D20M%Ye*Csy{>5Y{N?2m>_ko!&&iBH9B1Xq^;Xh7r4t?dYHe4% zEEY(Wuy)QyJ&E8EdIrBAqlFS6b`GVBkNR`ZhGrB?O!{`E5atf8N>62!BsPo0+TgcJ zap`>v=T#DC+I{*<<{$u7c$?=8z1IjPxF_7Mbx*(ylOhf-N}PS5XeY0O)>QLV%9-~N z!h6+N_%x*L+xP^C?Fu(uh#k&INewO+uBh7*!>P=D46s!Y#yl2AxArws{3b*q9LvkB z*NB0UYp<=Nhpyx1IM!~I&1GVe?nK7kIH|)BE-7&lv8zLQ*;N2e3YSmG=xX3u^Kg&WR_8(h-3fm z8OYxsCU8Fbg-$JJ9zK%;huumuQ6C-$NWRx0TVA!djbYNeVgEOhgdcBqz%9aa*mS2b zUt@*cbJ;Lsps4pCw^BCl-r<=Y?1Z(+e!<~z{4nk@Hnla&hI;F}Ku`$nWGn_s<)|u=u!O3qP2Q0g_P4JaNI<@Bipu zblf~n|F9Ior6FGx>RhFd9XNe&wfzo;w5muf%cCRQ}I`gnipLUJJB`g|9ltc7H^Hd#PZYLolw+{lFeu9PVgHhP- z%P}5Y%Ezk_ISuAcZ8_i%{o&(`M~h2rJNXihE(@An`@zv>~TAnisaN&y&+!<8#bZ;Zd) zo@c^<+rrKOJp~-C*42=h*@8sTEsWLmcq4A{4Jd~?TToVvX%FHret#qoGV{GrfF&6k z^fGTKC`*hJqq_H(CrpSDReXUgqiW-{o1nFNK@VRqWhgpoAZ(mc*2?Prbsio)(U#V( z_B5qDLPS^&g$zVn(%e=uX59bza1{N33yh-7ycXHAF)5kga0n18#ge$uVb#(2$k3P+ zBFIK}Y>@m@)eQ}+KOQ2_?~XRM0-wuzTJQi+tp_ZeE(vTS0b3ASoU zjK|*7CuZOVSyg0t6m%~KONHEh#DH;iCol?C84FZ(rOg$DP@D^x`AB^di1K^3aTVsm z%d-eW`9AsnXYAsRfv)a`s4oepoZf*0Tn!&sasAP?i<64M=RrZ=pV%sri<}5_n^O{K zTQzBFiJHRkeF%WHL~BVD>o-GKKvE-BLenzcbe@yejo zxqWT_Ea?a(+a)%~pN;dHh$V+#PA57#QGz6)>~2!#H5z=pR)_U1hq-r_O^ANryg47< z;VhIur#i)eHbmb2dEnaZC6{#7l($eRwj^;_FfUcETCkh1vRRPrq%2-sF5Cl(cy}n^?`&- zLjot?VPN7e9fAv6ip-^+0~y+5`Lq4E@%2oRx3&g>eRwJ&Gk@>sf!RWD#zFllc^zs+ zRm7Gt<<6&xvkSTrYfmmx=I^7fZpcYUkJJqOUX-aC=TafH;t~CF&+h3G{;_+xrj4#~1vDlxW~ zxA@zXn@t*XcgDUac~R`mxwk9{RGhdrI>IiK-1ibI<5sK;gsFuSl)E1J zZm%C(dke8~wyTIo@-;`OZd$8X^rob#l`N%SG8<~$9Z(Wxe|KEZ;-$q}@i@exeA}2H zdrQ4D<=HQ_Ch7wi`%zu<=Bu87h@L# ze!6_M2ufZ32tQ=kt&ivucD~!vcxj-@eM&R(XM|D0DzU4n?#RUEUqeYDPEd^ge77@m zKl}3p+zUxrvD*Yh-TJY(i|4OXy56YEG=uVut`Du*iZ0{jp`t5j-$yLi!EjzgM)F*!?i$FGs6AHO9b2RuTPC zDZW);C8!?ywapTzdX|#+rdR0wkuf@HO4MtmVT5$F1-f#an8p`YkK?j|DIZ!&c}+ZF zj71|FcR3WH;+>_2KY>e&^FtFNfdU##!T+Fz&rxMc3$qY1HIlr_BQ2DJ zb9k0Htd{6ugriVRJOpXUcSl7*kqPrV_tmop9M(eI=CZYW2cKHw7&81e(cg!AOw76w zb}+iPI=W$zt+&BP$RFWs`Y;1?L;x~zcM_Ez&WU=+UldViz#+dHc_G^PHQtzGVu1#5 zlA5>={S`?8Amm4t-I(%dg!Oh$S+E`|e00S@QOv?-h~T?H87+S1N2*eS)GWUedCI7Rkav=%CiZDm4JxSk z5P3pS8k@9zZy8=2zf1F52T8s)E(Sc3HhfHdym)g8Xu(&@sme&`7!2`ckvi=9fTm_G zoJ3{}y)Yegrmq2vE32Ixd_|~Q4}JTU)Y|$GMMURqC--oh1L}_p0g2z0qvk+ijtthpc#*=lv^b$KsxXj{T{J7M;F~v%kkb}W9 z(j8&>{B)}=$*^C+hBsN0gL=0CInGws!sg`k5nERc^@FgujAr9(dZJsX5Wn%;H%~|{ ziO=UyZ1ig+&%PrQ_s=CSw44~&6@RzS2Aw4>G;z~5k!@bGgf~rMysf*u8Z8eqtaQ|t z!H#xuar&hxaDmxMVWaGskikd*Sl}RQN`~xmZM7_W1u4;j+C;vb?WuFQfDHynPDd6<$&3o7V zi5r-tDki_Wq_Q1Y&|@_B(M5ide*(AIx3di`Gdm3)>cv~5)L*n-2T8i#Qh(g>dXJ@ z(IS|gpKbOtge`!av+4Ir$osfH&mX5Ufk@Xz1y|2+oZ2;_1f^dOv8dMzx6S=v=edS? z_(HPZi|22-KG=O_FqsN^iNVi$6Z&g{Y90ZE4f}?akq}IP`|sd?0_Xh$p#En@kj?znBNS9)#3soZDniQ$2BPw zvee4IME;)~^8`vc|NWzwlqaSm%~2tiXdo0kfTto9bdUjn32(i8Yc8)~M#cvou<(7( z&;$jIp1|R_`|)P;N%=)CEWmn6?`m{oH+`A4*xmmYGfmT93bk$YgP zE&i^OV$o|vv2aBLdi{b!IB?T#bD@DzU?Zb1Q7=ABVSW!L`sWq3uKZ+i(t8a@qPf5m zE%tqT*$TpXWI!|G;Mk|3A{;hV%U)?v_f|HZ>_S*s`J;4I8SwxmIqd!pZcU_N`q{jo zh4K_DYwNJVK82^qb_kZ}x#5K6g%+e&DpK!!*>b7Hkqzj7Z&FlT+%q?~Em#Ig9R1&b zER*_Y11l%WBNF|oC2){T)nyn{X3F2p(h9HZI@zQES_-x4X{YwJR?pjKnDatpl!{ar ziW;LcCyj2+zvb*Y%*OO{jwYkHZY{8i7n!-4H$&eZBCv;~n-2q8K;UgPE!!h6$|U2V z$Z4wvLF}SgkXRwE%EgS~Kd3uS{0NhhfPmgFaK%$>&%>uf!Xg@UbY!NUc_xJlJG&nT zOkzs85yDuOa`zDyCD`>CH<%^=hp*Tic?C(w!kKIn@vD0=WJkC8^o>KrK z-#SI}!8(%@s>TmLVmEhP3b=iUc!A7!!u5H}LFU=Y5CS8flhSeFP)||aK&52@r^1*z zeB+yF`&Ya(hJQfu_CJv`#)s95fsQNarqBI4TR?(n=S$E%Y}Q!MWy7>);rDrBX4d&g zmiDldao3`vw!Yn2@Kkx`WQ8kz0hr`=tx z$eTy-he~(;FA#uzJkyStz5;o+O9Q{Y>MFvQkog-3<(*Es3y zD-M@qUGx)R5bK>r9c=Q(q`2!U(PO&0m?phaT{7z%x2moZYXaP@vAX%*_&!-G0UeIX zKf+>bg!Dq`NJN-RcoE(=g)FN z-kx72syc5Bs!^iFEZA7M?~lp_+Tc5~XPh?Q!|y&l?_B9+Gn(iCunBO&-_W#khlwDp zdwWKRSN$C4Z*zX9D16#C=TGA0Sx~!sdJt!v_xE2oDPV3p5m${Z1=T%1Ng4P8lI+}^ zj)|NQ{U2fo-V~9p+Y`K`B>I9Al}y}7fX4J<`0u>9U7S*h$qK%Gt67S*ZiFS*K?4;? zoas&PX!Qrh>`#cw1ulQ0T%Xbm9QxYv@DdGjBQIal)b||v-XR^C8+`!od(ZaL`aby=<95%dSvSoQqJY$XyFoNkz@%}2+IQ` zulGw<9UR0oED zqT&!}MO2JALVM3XJD&HBHCsI3rq$NkI&{26Nw7K7aBOgO;CHjf(uqUe4<0dMLKl-R zA>~Hh6~DebpNWnD-D!^MmJ>ZF=R@-jjs3n@OlcMplFejGfU7GLzRg+vh%v8zzK(al zK*t2zAawNz*0v81JM>Kx8Ezf3{Q|GooB%)c*1`32d-cXmtxr*6J6ZS1@p?_H%rn6w zOkgA~Q*%01K3TOZ{Rq)TcJ#3+IE zfA!fjK}aDV433k{N5PC3I}(#h00PKL`jf+$^4}>1%aAdzLhsGnEYM$9+|Z-Z+RQ|L zZ-?mCbs?=7(UB?6n*WInSkgfSez&+6f!W zdgYK{d7Bm{CEvS zKM;MBMBh`+6ERhcf^9MpHUqJ^Cg*ua{U~j8i)1GYjkjMi{u<)}cZ8-)<#uj2MnV!s zx@F(7xFH;6`Fc+V?o?S)y#F<>GJUd9!cIFDojSS*F(3$YN}?-bB_G(}!lf6C-m|?b zz{CjcYDS8bgRXD?6;p#g-6t@a9BiW1zrkp~QBj?wvl1c1=k5CA&)3BhtOcpx$ znQ6aT+fa%LMLFrdNI3aRhqt|dh0wC9w**e;L7boAaegl0zKg9 z%XQ2Z5+NF970gRCKgkOmOE^k&f<0MUfRzksugBpy>|^&c5I80}DT$)mpqviIpzudO zO!Q&rl8j2&@WT>MB36x>%9yU<^M=_5jhX8H@4~R%G>QyQ zl#{Ya1?|Q>H*}&C?@;te;hb08G52gL+i)-#GNqM~Ye}b~$OA)>EpGdu(#`+O0}Quci9`4^^-C`U=BYB9Ittk*+0BlBDGOHi5}oez!Kjp+u=8~ z#u2ASX4L1Bi6xGmQBtTI=R#oHSVhD5i(F?hN{k@=9XWSD!()D#p@FU&rp8LIB z4SD9F%%)#(lcho;r!(QeUHsdaLRd;}A%a;&jcq$ogSz=nNl9XESMV|(zg5v&148d( z!mY~$L%jR9E;0)XzDAA2?Rj@7k}~#-=X08posv@Qux$O@o--7PL}8`7qX=Yipac?c zDrkzXIIHqf{FSZA(eo2$J!w81gg+M|9v7FJWw!ef~}R^KsN#Mn}WBhVI6o- z3d^|dZIDA#&lhl{7Fl%zlx>-+HaHR)2uJMnsIB*B4}^z|66Ify?I~1>dELDL7BzP} z8XZ?!6^-Wq>|SEfPoWR;S>E1ixht03?QAOWQhhk&cn@F&8b$v!l)RABEUjVGbjv5m zcDbD5Js$|3yJ?)qH!)HIbYE4Kq2P?pF_!ae4Aq;?63Wenh=;;Z5vEY0Nb8NMhsLw~~46t}CJ~E^xhXc3XA@q%9BVkADnO)5N25L1n zIsmcX^ZFGQk%(U@-mwnKBZ%8qGoEYoLh$}(OAq6cR*8rC;kILuV9sVzv)=%M%0PBL z@T!N>v#TxkY?|w5y?)@=yYR)QT-#cIOH43z#9zajM)~vWgLmETf^8Snb^>nlIp~q? zjpP54avF!+dJn{Oo1Id4;m%Roiyo50J2$6~9iV1H$IeWSG+UzE2zQW|5h${bTg@fS z14`~F{4z{;-_M0>{gD`u;d1!JsboAD&%Ek@hpYGsmlF{T0b^$@`a%JN)Y2VAc0)& z-jcBaBK+Z?^mXB;@?eCK`j)e6f0ic&_csQ*RXSq8-syKs(rV%YCpZTRa^V5tET2wchBwV=RWtG z-?5p~)}<|-i|q@w6k6vb8T5#eS|Y|3)%6vK-&x{(uIoNv1dW=Vq(uQ19aH8qq!YPK zcxqAO=Dz@1J=l(_HS|c7W`cMCx%`V!BSUjg{OUA$-xFokK<~9Q8W|W}V?_V+z8*HY zae{G2*P8Fr%z5)nHWn*?n3pRGgb-n?xJ~ms`qaqc?!9zjMyuc{8heQMa^8ZcX>o} zVwE4M-cw)9I8*X^oJc7y{FCB1GJrdj7fX=iN7mfVWbWRvU^p0tV|wBlTUlq8jiL?a zATrPn>&MCD2Y^n!`LXzZQd2ea=jDoEC{R<_`$1^!MGagh^UklA$A8h@8dRCqqi9#~ zD7WuZ3wtageUEU@`$i~X|9RaR4==5tyY*djA|Bi^M~-Hq`)jEhl-baTjWl2B%;I=` zi|3^3q{O#JRROUUzTF|4Xse{`ciz?_!T-E?wAP;(G(yoySp3I>akc*ac=K-WPNI0l zpB!jaj)ZO_C2OuFD|fF}iKoUgmC6;uc~a;L1%|$_Vvq^{y35Y<*C_qxF57A(E?!=Q zd>b-6&1gQ8qokz9cc(1wsiqFm6#1^ZL}RNWTl~i8${ZG3CyBFeHh>m;-*iWJgq4h; z)sVl+9esPRoo5hDs(c}4>+8MG%a>Grb$FqAH$n_v;`4(=iv9+-A4RHKRHErAlz*Ww zi8q$}YURbru*fxGe!a9t&YL7Bcoc-FI6=o%-Aef!dEBfck#L99pAGZ z+a$FT(HwJx)#n7r8Gh#gqM|FWaZY(!#0m1E%?C!d`VeQ;(sUPR#zITbo>b=xw>hCS zhmlJxBJ9A!QAd_6=^JKd(J?RD0Pk0(l!ct^D4dxh=3RjXc)MmEv-?46B&-K092hlS zGUVFih07M025}O%h3k(C2H{eloM&GCJ}V_>6v>MTv1<&`3L{QmgDF)Cdnbd0llm^k+ODvp~G-C*eOFm1zJ&Lv>3J+7BWWB$=>QcN86a!arpx3PL6Rf*RnznyzFp zE2(SDV1G8&`FK*HhDIX#UiGzy$y%7TLhLuvpd6^v97$?nL!$d;QZY`-^;%nzhTX{u zTk-nWul^lhh=iU4hHcgFKSR$2wX(Ko$jxrC+L>da3yN zQ*rd_C3A2iMUi^CR#*CDB5p1+jqs6o`-QX&?P@8$WLU?Zj&_-@@l@-pGd6OXXEtPR z2-IwQB#V;@yP&uPsj*D^6dw1JK=574kiC@Vr;o8w7o4`K*6eJ9hY@L4#baMI0A!x6 z>{Cg<^xzPdoS7j;HQ7bBKF?I{b>$$WZWAV@P6v{TBva2T2EKo-1C7qcgrDPDp6@Cb zQwof0g8tI?P1J2q-IaNLApL!|65j>?DG@{yUp@)yu*&TSt4Y>~Da0Ky+b*@cc=T;U zItT00&C^0dCJN_Ayf#9Ub%=)Sm|es(1w_#my_11haM}-!oflvgb=Zd9565x7!#_*} z>*?RqMhtDwexNLnCS#Se{ZdrsY)(&-3kcF~vTzdT&O?F#*@e5rzBU=04F+`QDwHP&Z``X+g&#cnO1`1=<%ur^rb4e$~z#4wf1h z8~EBmn#RWz5l<2q_6}GR+lfM4^n@`m3xjxmNJ>csHE%HI641&i%jJu@=+)T?fC0E6 zNOdQIhv8m|eg^NG9rR=FrhaUk{Y~b`%A=V5iar>P-*GMr4gjvZoz}ZgNwz-LLbl=mIvEW)=l4en3qMV%+bzTX$=opvJyMqN? zE;t<<8<+fR%6Ct#ajc&GQ;9Nh3UyDLpCtL`z-Q@(_PPFHrf&D~XW7{XUA#<8?j$8; zJZ(ewg1{V49vdWiWu`Z;yz4Yh&~74dAtj4XxMFAeQ=CkBK9vNeyM z?uH!PWfb&T14H!!OsS=qLQ(+j>9#jCAd}TJjhVEFD`Sh6rYsq%gECSRz2@=T1iR+( z)QL~@_t8!`;WgtAL2kxDF{d*YH&o{&Yd<=wyn%13$6V)ago(sVVu2+aH_N|QOu+1e zU00eFL-oxo8w+r`kk{1}^oKV8?MqL=7E!QG#n5B>F7v@I0qe_@$Kg1-`syOj8>w4_ez3*AYa8Jn%<%1Mc8(;ro9|;w1~%1gOT5F*sbKem-xK^} zgiGPj+{rPHaq*1gmQAElwb)z039#C-Ctc9K)*G+;Z0nUT9y17@nEnv=;GHtyFft~km#=rj0Cb22z#*P!~iV(t(J7shU?o9Ge{8##*h2r^_$m27+1l4prWIsuNVD=sHBm)?S zz~DmUG|CXJ*P#JxcRHIqlFn1cp*3iXru6DEn#b;~3-}@2sRRayzW<42{I51UA*96r zq$vS9Gzc{M|1*glYTV|ZoOP5L%Ou;`S?G^n8Mkqe_3-AcQdte4Agqd;x3;eDKka)X zvy&o@KNSn|vD@EXo?Y@{rKGp(sF_sVBYTisCSzy$rMgG;G-$JA{LczgbPQ!^N&%N0 zmYVh~J8$=VmvpU%Pt9ro(Yk@R+yT$oY_%JFmyNVu16KC~HE$tt50kB@SyLM~>(B4s z?t#fjERN4L0?k|Kot*nFC#=W7PA9Mu@IWYVWV4>$H2dwOgJboT@2(!-63h_*%71YH zKVW(J)7wAq8&xg)dY{)*d0klD1BdqF4_ggf+)phg@CRPkp6-^9fybdTfDA)7^OWD_ zV1F->Bl1@BpFdpZ0$Xv?NDZL9acFZ~4x^n`?^t!~MfWYl-$R3p4O1@P^i!m)>-Kg> z)eWHA?ZW~--Nr8AutW^UH#m|eRF&2O%}N)$6Go#R0y`8t&h1UBBx@|y^2X^_BkM5P ziC+n_1>+gDhxs+dyAMRMXY1(9f7|xi0A12>Odbf2n3M@>ZTKS#pv1;XB9txy6qm>0 z)7s7az&8ey$AKrHt0cG9&vXkKF^qfjkb8;pL&S_@32ibL<;<%2y`DLh6S!{rLcl(~c+YCoQ zqf|JdP1BjQacc>Y(kvHHmVYk3ubBncd42a5)NRC7K6`QGD<;wM4%@=5Wus&LSs*R& z5n1;33C0JZPpsKhYto+*BgXi1xd9M-4@+{~kM;6vmNU%1jw}D)qY)!Mot8qgDs59JoHdkY(g|4_V(%JfrTsU9&kKW^p@&dA^-*n+Cu5KKp_2 zon9ZO^!!^92yRM&7QYRYfzKmA;V!pF3c{?ZlYU>Pz=+kfcN4%P_|;_mK=aok+uF0W zmf9^dimY51BDckXR;OQBM6iOMJtcwOX!-MKBTo5ay%qdo5q|wGT-_d+U;|&G?d~h9 z3DEcmN8S^vifOZfh}?SUK4lxCk0H~ogGp$(-ED!1ZSNd(ec)?_7QFB`1iz*yXi+t_ zKxZw4vg@A6XZ_{vJUP?A@A|5(Y=UT;RIKr;^=T&pvlh5(lEc-Jxa<$NC(K1^gS{o= zCG^|=g{Y>xgV-(LLX|OoVxTnJ6+2OsghR}YqX@mCvZZ4@zG8Jvuigyalc14`uN*P& z3G9``EGgyICsAe8PL`dJE=7TtP>v}a;VbujYuJ0*2>9-5{TV5e^gEIF9WKz%z|^z} zQHAuE>)lqv9RYReaL5vcxdd!t{*8Y0qheSjMvtPJVTW5~xll>-< z4wRjt-Xql+R}(Fq-7OclTE2ALaHY5XpjysO+V|~$kKR9{Yy3hAoAQ2_wpOX$Wy7{w zuMhGLz-lZ+Q-`jCu6H+}+jV(>4bf(Tm?;3)=RwBn36h90o85csL(!ZL{Gl)W#piPp zIgAd&ZHC5VllXvuYburb<#+n;aFVyMoYcgBpblj8g(jM{s!cMFuZM`eSiu{llISU6 zY^0`lhpWbWt@c?8Hc^QA-NQV8(Zmk1Oir{&F`rrQ4jt*f(M`z`I#a#;?Yo|*4cyMB zq@gTC!oUz$lEGFX8aqoB`($+XC`99qu^I7DjD-rgy2)~atV6m(YFDhCrBVR=@d`jwo5_bH5vv)43x0{ihJy)JgvaGbaPm9v6a7A zcwF@Xv8u-B<5=s@XCYC?vJHfPdAdyWM5796WVK6 z7YYCJos9mg&hzVfevt}!2-4hUT)tRB58Fh9F;Dv81cu-JMX5E~SnqlFT2jxGzwkvr zhja6G#D_VuXFSZwjL;rOR!KcSoc>01#_F!;VLC#-&Ih;W>*|{W(KBi^$d^us^a<}( z6`nkiVB;V3bfuV*_8Gt4qDFZWe;Jr98-z`ZJ)@k7T$wwhN+A6Er{q9{Zer$+MB}d9 z7J_j>F#b5$Wp#y9y`uNdGlF=A6yoh?2tNwrhj-*Cq*cHaDU2iAQR$SjLBRIv)<1Xk z!;F2QQ_pb48jXOiYgOP*f}P9#XJm?#HC&+u_Th^6Yyo$DHC$a@aCGqm_TG2t_d_G3 zwf6_b_Y0$C@!ne5iQG@?KnrQsIc`LZlOF5WWw_NR*Ae(|D4}}mq`B#U_vOO`Mbi5w zKszP(DHB?!7z(u@w5-4{IGxxAI6nfGOSZYL!KLb5K#pH?#2GKw{4;FQ%HRLG&kP!u z3Af@aHSceCJ9Wi4t*>JWwvjePrfo(MzD4jJW0;NdN90o)B=Zr70X+dJd+$468z39W z#vSSo+2JYUDAe(np+L93fUty|P_fQSIvgn;=jMfiq4bnFqPJ@nAQdk%b0%V%<{025 zH*jn8%&0LgelTttuEV(}W4QHNyj8ih;N|h-QoEbbS^gYp-Gx5zas?I(pW=wr(<~Nt zs>cp-?^^#Q&|geKzZijuqDSJJdi^;;=pBU>u;vW zb9?@o0rO?cHiuXXmHeKxrnhLt^~?E-Kr+^;gOHeD20)HR@^i1TS0AS>8Q$4fV2C44 zB6LZCy+-%Xz zN5c%*DkOABJJ)1(D8R@{JvG}uCqVJR+QCGEK4|^AxzgaghT42o4P;s_Uw77@UW;kt z2B8V2Pvp(k!{(cJRs%}{wUtc-zB)C%dBtsI9L$4B8XrRjRyNy{x+jet7=MpTe6b|? zp-TBKw*F_?kKnQ~Rmv$a`Z?&{8 z0;>-O!hR4mIuMkkhY1RrxvNJ~MTghx+M?ar+bt@pf`Km~faCw`T-+5O8KE*5#*TvraJG+Vz^l>5m`~k0;%Hyp`+$-U`)^5#Sc+x+=A<+G?V}0`wFt>Aw38D^H zX6~>!8P;UExWs%5+P;;K4cs7p-VjQl5nW!-4rAX7seI}0cY57u0d9@F1=p_-313g3 z0zH2N-G{Z8P6--?R;bjCgvh;96p4n0tsNYsLl1pkqQ|LlKN-({j+pgO=f7;$nTtl} zNQDRWpsJ31R~gO+=S6zGJs2v-RSj}sB>@G6kH?ox7w1Pa*(7yN_ILICaRxNSgbWi)XW(zE0eifSH`^ z(A-Ik?{s%T6337g>ENZaHS8^GnX}h{GCr?~#V>YY`m9o8eg0({VP?Z!cb3cF>p&>y z;#-%H^hiV22gaJ_-cy$W9WR{h=H`MxKi2*m<{ZIJ81xIN^Ceg>^XIbH>2dNh2$?D& z4h!?=!ZP7bQid3Re*36-2Pw_*@cg^^@X~F>z!%YeRNXDi8vih#apCoo}MPS61LNSHT`47W{JOX zrU2#fLcYvV>tCXryjZN$@JsP?f6c<@6XTO#qH!HQHiJ|`wEX!+ikYTAA%rN)Gm z8I|VBy2bXBE)%|Y>t(sl{Pee-Kx}P!3OMv8HOw7yQ5mjyaqb1!VMDPA#qDf27KlK10j~uLd;1GPprSILugpZ z-|i0Fg>QdZ-M>KWxVlYggwqtud_>}{WOM9riQl(R4!XJBZd$zo5d%L#!Xd($VwA*& z+Fvv8hXLy{3AIuFx5fN>Cf7n5Ho&djG0uO<@CSjYQ0M4^USb@M1IbskQ_S4GKT2)y zePTx@CRCF&wGbkG@PwQe5t}1QOKPNVjFb@6zbit+7dQL#aNEBspvKDBOdc(C!aj9`t%xN%_2yh9@5B z&)R%xEZz$aBV)u!u(F@xtKeIU+uNh@6kTiiTJU1bjbW4S4gcmz~n25rp-9GF% z|CK;7{G4u9+~ zRVJl*@vQn9270)vmC!oIXy1_^BZC5DdJwr-$KFMi;%60LtzVY0S1~q5U=0HEJC-}- zvEzMTA#zB{M)A9({N{Dq(Q{{^E(z-K8Jn1!40k#fLwWOh+ViNY6FIRom7ot2P|7YP}} z^0ep!Q6ew$HY7VLI{Nf{*E)*i@MB(I{+eu+Q4>r)n`p$+Js9X-Z;h(ZQDfY{6WslP zk@E`3tKr!`^`Nx$@PVVC$VJ%wGRDtt#}{ysEncPHgdoIMBb6;3Be6{}^bxnz;&c7$ zw`P<&8xAcTdj_D<423(WtoC=U8$$zlyj>i>%302s4p zWEpN+J4M2g7eaem2P&H{%U2QQ1TpBrg6P)f;h`x1rd*n)VfLYQS>FyvJXYaSsU!9>&k;h%u6eC&OyBu1_dbY&m>pEQor zZ{xdLsz;T|nbi(RXw(0qBW^Rk1*WdFZZanDB?- z88JOgy8v{+3>iO#gGOfcsI9mr=#AWG!~>(c3U~WPMu8j&{ylqV?vmwN|DHwm#7Cct zX+UM7?cV?HoJpySQsv}b_&}|{4}CuKDzfE`i|@W9KZ-9f$7_U`&B*j zGQgWt4Qg`F{jsefnoxr?KW7oFQ66Mci2$7_iJ|R0sgm)dYOu{`?DyyBh1C zpphujUMLBG%!AV)CbiO=aK@;xJ2~hV%%O2GKn@5h>cYUGd-2-tXKm;tJmcJ=D?(BM zE2hMfD`U+#{1eTc=Xq-~e(n~7UN-c?c{}C)HDdo;?ZHRNeQ_I4zUX7}R@$Z(Du^_@ zvd#?7f?$qlRwQ33BD)=Kd_JqBTQ+Wsl%_Y#Z=I%9{hrwq*Mgr1Jkcbl2kk~>Of57$ z3vt;=`A&UN2n>!>CbG@Lwq)QI$`MpMNDg09wt7x_=vt(!U%n+&!jJrKW@fZdqApN` z9TuVe0qZ1us%$Cv{j7~lPmr_Rh+#O@_(+ipZooHeP#WCataj7N?}P41%zz9x97hiQ zJDgj;6PqIWlJl|ZNB#NglvvR_yxXC+w#pu&^^wnMH8P^va}~owz_5;?<*Xgi3z{ur zxM(kHlYIJZzw7Dm!Zv<)piq^J-X++t)8*9(o#Z#VwL@=VHEEKjZ*kvHNHI&ADNprK zi^GanRU&MgFTdsfA&K9iGJzjtmc(xx_To+*9V}wob>zc~pLHI$OP`|W$c_|6j`uJ^ zQrmX0H!RjvLlARy+EO#wL!&%YeLplzzvL{WUu!k1v!l~oG~z0YNjOh`Qr3LYLe3&w1nc9+8?}Dsj*oi0}X=k`Dz5vh01X^oaa08#_eHratA3~JIpiU*8 zmbDEs^0MDT;h^r1ezQJ)dsJEuk60`T3!9ZEJB1|5{#`I1N5VI&M`+HMS!tJ{j*D{D z>}mJ0izZaxW}af2Qg~8SCt=R)TWzydMD!K`IKP`#`%~b{fCVc@TkurO9=+d{YAWNA zJ&$e&YMeu!;gfx94ih1w%ADosfeX>xM-o>*kG2zsGAG-P!XSTJhy>i=&A_=tm-*a= z(xOgUXwTaVE{}((r=H==Dowxr6S;OELlFlOGloYa!cWH>ZY2%EZ>xcA#l6IlHyT>y5L682u?>C=d;N`5o^MN!o&P= z6(UNtb?g`oR_Ky93-C*St)&y!=v)o>V<|I{Umm(f;b#vA?(O$&o)h0-iP!wK>{c0Zp+#T@6j5P*10x9Cf2?INaq1feJ z?*14*^@4;QRD}VB48mV=FFk+ux4ULK5xXw~S8t}}GgMCp_sC0Xql*l)<)Sx>>8WK* zqCOdV6&F;XT#4l*ybhF_mHl`}X&{V5_!w$${<%`YDXD8V|iPdZZC;XhSuOZ+i8RLgk>ZAejQ}4@=j# zG9}&KZE4LlWIt{Q79BMSV~WN#u_vJq)P@8c!q!Vc0T|(Yzaa}NANV6`b-b*e*oBK2 zeZ+3}5#HY0_hRp?D883fyY1&a&sEGxiTi{ku|o zEl>Ozhbd>0;5v^R7WSKE0Z+Qw?iQPTroY%@1V~DuPK?Z;1#UM=)7gt(U?HHM;OD6P zl3k_wff)R@uLs5wdk4v;1_K7UwT!zhR>Tcl z8ms+qa~4%>6c(iNNXIX*4*BiQf=r{6RwyTqif|_Wj+58ObnN#7lHH2vd^!7<~ zdc@!T`rPp=-abhJ_KSuvXCJ5v#3J^ZU14wz@zBAG`r)QPhvYQ;kj!5F$&+<$HOQk{ z1Gt{HX_4RK>&}S~5p&p@14yF{*~s%;y5_zxf$RRLAE~;>U$`>IIialf@T2aFH#~+6 z+@_F}`zV+RoIHsV7JauYaZw7=+O|{2=?v_rj!t)^D?Pf%t@By$%(9&Ty5$l-I5OFv zIx3HIZ}fE=jFOU`Hw>{~$0wO-<|?rkqiQmcT+}Z8xZ=;KE9*s8uq2Q7LhV)B4srHV zWdpO{_IB_nwvnx8e3t$`6x-sdSKG84oTXZEF)9yRf5PW?mx241hl|j4UnS^PMjSX* z*Tp*;UUgVvtozQDT5rTvQ3SQ1l!Jkb@Wm~FlUgd;>nCRE)BHIp&wSbg) zU9-N@E3TKHzB6lPQnw4*OPaKy){53V!3X^I0B+pa+MHDo{K!fY(T=2c_R^WAEcISd zi^WW^_L6Z)_|FoQ`Dy~ao3V_APm%r%s57=6)O3YHXL*8UmMi*|wcfL$JGzF0@p~as z*&+JL2s9FYuXyC(5b`!1xMJc->cb3yA7aZYkipB)EjKm42wF~TO<6Q?iN+rPm82K?16*vgK~mXX+d+VfndNrum_e)cHOnn^ zmo22`vk93F9azX=>O@{BSAi>XPTk|Axp!R{E`rxV9j!$$lD|K9FTKszb%s_Q<$9 zvywlaA(Yaz*Q@`IGfSBlCSPZuUE;meXv<;13*JqlW){Uwmiw3e0Lu>5)*eQ!)&p!9%$93F!{1T)ymyc<25lp5Tt|hG=SDSHdW!K|)$pV)!XL=GWT>q1 z%_}NS119`zg;N|n^e*;Fms~h;Fk`zLO?I03VmQ>G`FnLa)pm7Vd`JmmqNtl&qh)*? zl;9eE4{@6KMKu!-G)xge#2Jej!z-1-rnZvp%iIV} zz0M}QfgVqOgrRT{Y`$j39hjN%eNV821vcv!89@0-sFMgU;?a+fz?YY502Y3N3$b8X z6ZuWuR*nc@4ky?t(6UJ0xrve2re=wPZI?!xO&~09OWal~0y{npC1V*su@#o<{48>Y zO(T36*YJFJnU;tpWX@wdvJ2;?`L*@9Tw#C3{xr{7Sb&=Ns~21z(3S|zqIPygY+!FK zaz@MrPkE%pe1?mVBJ(5S#n#a9mLJGcDyL(hlEzm^=Y)MQ@vobsp!8o7#lheeSt1^d z^~+L~EXLoj(atu*p9DOdNmsupn&NAGJ&8&Z>D(L&GbU{*7O@fR=Hi24v+~>&8*hmT zE$~fbbfNORW}T3~jzGc%d*5j0Fe*sM+Y1q}6F7*yQlR0QPiO$&)pTWV+ZB&D>EsSp z&${t7)coo+D z4Ue(&n=mK05hfuRa6XVH8$dv_fFvW_M1*%f=y>I9%Y~k4ie!^BPf}$C>!aV&>>30F# z^hjdpw*^0zqang%QFL`H>6A7ysX%f|ik?x-eO-y+2+dT1y9-%5gw>JrE2^QajxraN9< z`5-KZNJQSI#gGF&(Cx*=o$%?QOp7hP`T?)f%lcw4R6kv7zcc;8AZwC5{WdB{OFt*F zc~BI%v8le*<%1Lm48(oBc}0j&&PAThw4b}n1O46BnAY)oXLp-sHhTTU#mR(}mbDGs z7dtLM_9Y8VHZEctH&-QN{qYe;aE2b=&;j7vPCWYtp&k@cx|r!ytp)SonZ7-eg>|r; zPX)5dsJQ0bv!VxmN@nl&9e(WgjrL?4;=Wi*ez=BP4NS*Jk5S#aD5bdWT4cL{;(~5t z_^?M;Y*EfA{vdsPXfFi&YWK-h-cgoq_(bnumu( zsr^vW8#&O%G`&>%Xb>A|LM-g|;7mRWMcU09Sq1L9Sv`_E7}mH%6iq7PTJ{j;BK*!f z`Dzpk%3YkUG(KUk5Q}hz{dHH}&tc+^?zXs7q!PyCwh!ZOBpZmP0zr~(FVnhCxTUJ8 zzmoYP91*8*<*&~&CwqN}AJrccZO0xV1y>>E{fS8MYX*n3ces^19Ojoq7>~%s4X40B zSV*-%$gd)1J?FJl9MjH@0y0h<*e$^{4)MnM19yBC(8td1m})fj46e?rZ}L(k8jYuA zWmR1}6fhp*wE)m)-_Ifh9)LS2q5HBwy$Dxs6Yco)>CbJorx_ZJREioLQ32oXR`sIk zKo22A!oizY^iG9p=IaZB>GD?wm9+k+Wm(k8XRE|8>S(Q!HdmD0pQ+btS z3$M@FSOG0yY`3ZtaY$6 zV{1Q=)J-(X;T-SgAzj!qBe(OPkc z9^f#JieO)sU7}PfLYR5vZ1v#JZKS^Af(wO)IRqbleHmzETFd-yu&K+hD?~;?&G<2$ z=3n5?vJK<-uVg+{ZC#zwU2V*nu=s=nz9Z!lv;?aijYUp}VCCmrdbu;P+tINp4>FXcLf;hR$JG3T^0W}kV!rD`)) zcKINdm$t%?%>vkzB4bFoDkE(@ zT|@3C76+-6ZKPu6_nyv>6ZrGq&K|7gM!Zy4Jf50d3W^T!BxhxF-h>xqzFPp-IZpsa z&H-gyw6c%vuv#P`iP(iW+ASPqmee$liWV|Ze{T{g_6Lljs+pOs*zW)+-{XOuvg4jp zZm%nL;OrDAfgHCd=tVWQgQRv`*!`=eqXnn#^UPP5X#pf?7WEnu5BXpvBo7-gp1fxK zOFD&~!Uxwf?Ofb%qXnJT%{Wnjj6X(CRrw>QLB_;EW~X0*e=4vJJZmT3k;Uwr=Ajt& z!92T2Z0@@spW<1KkS8MiVz5drKBLQ{BF zoQZ@6*x<`Dw>|a+7{y7JtkJ!!t@;8`plWRu$#LUJ=(QuFPJr~1;N_Rcl0Uu*ROQBHhez7fTvxt1? zj_=y&A>O*BO`YSGX-I~J4MH|~>e9Dx=1XdHnPt@3KxUzV?um)h&;GWGmD)1JWNVU< zKFHrydHeUz-tdJ^-EMqVUC$M%Qc{uJ8>ja^4|`VpJ!Q3%cTG-g`FHFZ)&`5~do0$B z!ogSG)C{Y~P7FlhTD>8@MFsZ9`V?_CzM9$H!)N=}R%5}1fh&RJ8xVZs$_j_~o8pl4 z_pOVz7?{t&vT#OLAICW!g}+7Lqyy=6^N8{DZ$7C&xQ#n$#|M39&`W2Z`l^CVD4hcy zPIXwaj-u-V1MT)76=!;!+WA=mF#d*ngQ!l|Mpo1F*|3i78&)#~z@ z=mq{oW-K{+_EGroEoR~Ay`dBrIKKydvmEwZv^|4P$QE&_Jl8zqzpe z2Yfu~LEZ9KM@g9&oMPD<=>niV6Iag^8h^n#~KiDOMju1!VDo_~pV_cXNHr8qyEVU`W)LudN1Z{XX$8WoKoQgy0! z>vUW&I{q#JT%-?2Lc0qX>YMBEA~-uG51QR#J>K1a2^}oAFl;*f!$2O#h-(#zi2h45 z15j+w#b4BRh(Kc0sBf&a@1dFNKtMr&VdxL5$fg*3))`n`(u|OL08sl#lET=toX{qZ zFHa7DY4CY>MOY)@F`in2nJF2Z!5LL!{M)XjKs;wYAr862HEZj{Yp<2sIjUiGB--hP z)lpv22~8!%Y^eDo5*mhN!BPTZYI#ZRb-yQYAW+hUVy75e1r>#GL zB%6H&k)@(Wu%>ntD?{00DJHA1`FAv@Wi5vqixxXiORK1tV7jIpqQ&<{^h($2Rv@Yn ziT(X77vb5|pJ-S;Qdd+upra&bfS?GR*v0E7^kLL4I)L_0h@_VN4hp7nE`*TFcW!?D zu`gFpG8oW+JK5|ui-R9lA1_Un&V&Vpi@2_YKk#Idmk}yZN%)z3haXeZ3wQvtL)GFg zv$n^=#MEH)hh6@<99i`^oYZ*V`Vhfo$!GsS4Ne5vdoPOGx~1SWHQukcf5fQ?wll`U zr`ByX2IYQDYs?4COwoZn%Ux-e;T+f;;}xqV3);zd;|PEH+T(9_$!Nh@Ubp(AAq*;1 zl~wl~xYbvY;V+wo!I^s7((*8qNl+&deSGMDvb`J?)vf;d_z8}+xGYV3&gj|=_LNU6 z8W(_FSlEis*MO|e9t!CgF)ZZQ)#<G z-41?Ch6<(P9^_AQQNo+hhaPlm=+Kbjws&C1cG!-rQc3(Xoqo~{H_l$vBoy(E*OV#s*akh)3^(tzx7_`ZkTAe>I zf9xM}QT(8yC1bg44@v(!-{GHz2xq=L*QX~CNMSIXaBchq25N7GyA~)v zng5h};>Kr1sIn_CXwOr|>P0R5;W?UFxg}sgeWeqlrh8vjr~UqyQ>B!(55b>pw56sm z^q$)6^ z7c!9bnl@CEIlWj zh?e&K<}*~pHoh)>Fl%<+hLMW5>+{4fKO`7(ZR-<;IpTR&|78Y+4_!%??yj=Rl4RjeLkYD|AHAl+Skp z-MnK#J?xbd^R-kUFQ!|{y6tlP&FN??QAc-19oC;CQ`zM5gb;|1`ev>uBOY!85AbUY z@QtU5cIj*oA&JzdPn9N7Oz?MzY1CA)=GvUvxU|Qw4 z3N=G}nE-LB03m6_wuildkxw{yp-joLn*$%N)E1p;AsS8n#R3N+{v@5ca^U6wrY|!Y zH)es)zBe>lbRVddEhWs&QR#)R-)+gHRv*;&?TRp%^`1d_W7cgDH>_rYQJN=n=^8^+ zL2^8KN?wAOMTg#FNal=^-@@POR7MQ4Xu0~qPUkBzy8_ftlKlI5zabQuN2o_bt07R+ z4m!1k?_$i7o8mQDw`s?x?O{|nXiMZ+4+926TBCUG36$W65M4f}_E=p;2ZVj0P24Dt}V;sr0_ zBuT=Ml=F9u9bJeu2A|PS5;6{uq{EL2rxB=mgwbTXj-u`Ex@M!Ixm57Go$n!W@fxus%WtPKp$dW|@^@a%*E{|AvGKAMZmmYHE_?SV5~B$0o5xT4@&a*RJcD>usK=B9j9LhFu3X-rF3VS^NA zCOZ}vb)UskI298uJhlNtl?7!QDD>g`H64+M!HfwuXc{QZI|l2aB{+PeB<}9VismzO zyA?REkq)V^3R=n|65~WK+_DFPUjK};q_QL?l5sTmjNJKtjNXv0%j?2oJl6Y3#g!l* zN`}Fe5OfX~OF36Pdy6&aeVk*sF0EtVhSo6wL`Xw`F`K?F4l(oIyXT2X+DrIjaRJ^LUI`^HD+ zFdC{MyCkYxbqH^Bgg89T0KS?J18ouwk22d2+6wa`S3>d%$y%-V*%^T?JY7hCxPdX^ z25}^JX!FJS83A%aqe&zT%gT*iDh#(Joa%-6$GJ2n$x&}UCFTC97%@vGFq^i*ADOWV z!N;auBBUgENQ9>+T`l7+6cVmPuEeb<$2w%`;i?>n!%3BUP# zF?X)fTaQ95*I;9oDg>d;nBLT5$CV9}NYW1OCoFpjeAjTgK6S$e-&Ja{-%1qU=fnoJ zG$@x)lZ+L8%Gq*Tq)nqPnG#1+Ic$|vz%9UqDAcW!c;u#Qn>-3Is)rRBCNRP=vUfax z@@pm;^yOQAYGdBB0)g-#=Cpp430p4CdSOnlVBg&9tp;ulKZJWxmR%SSHbf$~>=}4x zT*60YDR+Yl!Pnn^5|Qj&@5=Yc{j`^>Do<3@HUIHKzifsWi&cZuP(M;;0^JwD+qA z+6VeFM10Rb@Z`zM;7dE-ke1Sa79JH}4xfgbrY(R>^>_iQ|l zwoOTAW`>;-&OOmB#ej-Vp#JesXd2fa)Fr**z-4r<`0uaz12q3(g`s&Er2jJoFw>G1tb`KVxo1^fQl$2wH z^5ta%`@i+@7Z(?x=;-(VIdNlBs9@zQccx$Z_091jrcc*v`+%&R-1ZJMFS)9!Q!&xC zu$>Wes3}b;Z~mYtX@QNvfY@;id>0S`dG$o@?hqur`)uT&;SuPAHPwX=dRTdl-K>*1 z*Y$XX!OLbB)?u7SiUCOg^*HCES2TlJ#9Gmyk)+|o>#w4Z5o_QUoy{@W)~W|=B7)Eq z#t(e2r-b*Dzl5LNF6ckY{7c~gb;JD=G>vSxIpQC=g3Y_OjmDxs^aNrDE^GLYHRx*w z28q2LA`jY~i#fegQjFNGa0o9PqJtFVWm>w6&nMq}QOf&TG7!QmLbjJ$%@H3(AB?fT zJFft;SkCu2C`YX1*!=^Ju6hRYySd)~n?TpQP(&bU8el8nd3$7vN_s`TZ{{EryKGo@ zWO}6b_%0#peKzdme92OiCpE!OA40J*hvEFdr%!b7P#^=hRRC_xh;U4h+J|nWx`x0o zRa26{*E&HeEFW#s|NMC4(6A7G<(~`Ge0+O;iGronf&6LJld?UB4d&Jpea*1piQ%P| zM}wP4m`6wq4sl-q?Ov2B1PlFzD6(G{;45P3)>O6KxeMWPEw zH1anWcFa&z+3E#L0UTVTk2E-xGllVdhMLkT6|0>d{s(h!85CFWwfiQ7;O_1g+}#;0 zIKhIuyK4yUlAs~z;10pv2X`4tdZu+pvu@sqDbqdnQ7J% zClzmm-CV?TW;R?g;4uqEKDcbR++piTb z)Su)kn&2zv9~ZmK$X&}T5tNSNYt6zbUwoOIr>}DMn|%}i@Fkm$f+`s*c%<-CzXwT$ z=lPi;6f_c5=y&3s_I%*5F=ab2cbp^N2&H`GRuUxfh$hP1c@$U2c0jOydLvE&*PYcnS)t)&2I`o2jBA=Fd#rVpZ@|C=i6lJPdh{ z2i+4LqU{cEG6b!5ypKdq?AFi=ZxVe%VT0bG3JYxazUn;VLlk1`&@1EdG`Q0_0#CVL zo^Nv~G`${06v;HsUQxuJ*4nJ+0evOz#;?Ff9nXv352NU35(Rf~=RWFTCUYO|O&^}w z%wB_LspM$$Is}QU{r9bgY|zB(CBnjwCfS_^V8;SZBM`1|Hp#n;;_edPlo?Al23oSe zZ}_D^9b*u=tTHa8bBuKaExYm=g8F>1*H)Go6w&UwLe&mu%2${H{U9ob_Eu`X*28;O z8!DOdB`#E404dc!cA)&cM-1}B0^`kO z4hpHF1d>60p$3>Eds}Nc6QG`kb2B^T`IJ7R2r70M7eq3WN*gSTI-&|+H&(P#(AMO6 zK&01x%T8e#h1H{ z+Gg>XbLb;!g+Uh{(*j#xOujVkgc&HK3yNFR-*&ktOTM!~{w>%G@_?f2S<%}o@)4}R zkczE$c^^1&`rv0L8lTFu6)m*9O6UJD7)YVLOz!00iZZHIJ#h3GI8aamH}*Oxj3dy? z?(XY_qpd+%4Mdwrg|laCr0nox|BR%Pn^%b15cX(Y5#K)~I^EnL_m4R`=kexV!^dpd zt?}}(1do9=^`pzIP1^A4DR=8D0yx@WgfO%|)(}CbRZXL~ANjqw`Db*Tti>o&dS%m% z9%_EIrO1-MTKsIEMZTQ96%uX6BMt6MmNH@8FA(;Tzy&8x0m>BBWfsE zTd`yRy#XzrPN=Lb1CSWq6X!{~<+5U2g&o5nQv*&jv2x_FRXDyJbs|4b5AXOGmVy_W zS@qq33o{>7Y&_pHTf(uD+q7%y(1IOeg$@Wc3|BC zgRUY&zBWiRQi98z45joS^rM){m^W<`4ZvmHOf$MAv`ui+W z8p7v^R0`|<`m#Y!qq?HO?OZp@=NF) zTFlwrEKMR{W0MeB> z-m9R!kB5gxBERJ|;MZ?u$L_Ll)t3``G0={bDMhtdsE7jiZQ~rPM7xA`h^+fesT#nv z^@S($@8AlRit@v$Mo zX05ioxrzK&xe;%4p?k9MhT3;WV{JdkTvFbL^i3-*gpWMA@@G*YjnJgEFW`F)#kvsi zdcxe4t)%PpZciC3#6~a2w3an}Rhu=%I==(-hx!RYRVfQqsI;lg{? zp_YGl$z=7uvH!{5MpysOAO7d`g@=LhT_Oa|Wa*Cu0wQ9|p7Z)=4JoM)l2TGXQd2$0 zx9ow-KivN2mN>eTJe0%0{K!~9%^BQv6?e3ely^i^wk}t#`6fDn1EylXo7obD>^S8< zJ#o{NkLlcY6u#8M46RBHxp`%`Q#N`rK6`#Ju*I=NjSU%_ z54E+!p;J1DZzEUeW)pi7f)JJVio9-9)c&eD2T+0(+t6*)1XsCV-zf$b;6pl%Ss|%- z@js$$HLR+8h?&sxChMwA(<+zF)?w8xTw?wd`nEjtr z|1i7&FzC&XH;#}C4iOQN{r9+Y+nP2h$WOrNe7z%}?-LHVlf7#DLRa{)fB8DWFRqg~ z5Hfo%)-?DtGALnlX?p45fWmIuzM$xD0H( zpbV-hu+zQ(315AAk-sKe{|40*zE)uh-buQ=nS?N|ySpj|`mk0U1@6r9eT$}b_zVb@ zrf^%PPA_P=-N?;&Jz+vXdM0p!96~R$TSzf&Lf9dPIa{^xPuX2w%z4^Q{WNTV>_Yg~ zL0BoEGj&GsDAh+>6tty%qcs=r+8RiH!3sXo@Q~ zCc%C#Bx=nSFG+EgVf!g)*b(}n#Amzh&FXd~<4A(D$It`WKQV!glc zxb5lO)Rfm!i^&@y8;ToNf)XxCss7tbSGf&Wsfez=1rVMzjA4!vB*Q#8@A&1KMGOXm z*s^(>=iFb(w9_QZ;dIUL%=7v2aiqT$?jw9uV>`y@j;^Ko8Po73Itf612}oY3@yAzT zcn^jf(~p1ut3KeM3=sam*x(U-a+hBJf4JSXe&HV<64MLX_gmBmF}cXKMS5FX@Mjz;WqcL?F+HJ@Ti3FSRvnf_E6Pog06t`8HzA zvv{sDj}FH@SkGhXyb=rwZPYyuunSGo3LGiaJmgC=*#FtmYN=3SV61pizw3jd;WMMo zmpIgHYMmHey~jKMba@@%%7JdLk!y zVh7?~F0d?j!d3^ib|Ifrzg>(p|FJMi`0J6N>Cd0g@aK$fDilacbf(PRyoCSo^x1oC z@{&OH^GZ)}T(9glZ{BM3-pI!dm899_D0=l~pHk07{dVBx7vM;iI}nk+!bJE0`I|pYwHSWq|r#1P8uyB)c%H5~p(?R3j z&$ms^1uuIaR&UeZc;e_;B3-dY&3Aa_=Zgs=F|J?AmtgoD4fK{=n7_7vL_H|8=Wq4L zT(AOfJTRPjYV4B_mR{{$tN3hGS2p4JF00k}d|cOSY}*N-hWxod9h|XXa%p|qyPqxk zY46`?O|00IU8!14n8F(Kd0Yx4oP!LJD@1+b4tBN^HUbJWPtiM-9QK5dsd8h9Fb7u#N|J}N} zlYHbLDp1doch3ENwvDaLak8(L zI@Y0_33hrBmG$EB)Vz6ul9Hsx7LL~C;-OYXUl9TJ_9#Vl>XS*N_|;dVPkL!1Ifppr z!NKZvaK~4s`$gNms#2(EImhB@ebM*rmdu}YIqow_$LUpad-(c4@nfP_Ph5%#F0aLU z_M3jQ&G&j?#lNB|EbH@x!X*Qp&XGVO^)6+ zyF@NT!ENb?MBD9ppG<0*owRtV&bOd2{d9*sS_^zdYqIXPk}lmu)=#~Y(NH9)lum9Z0^&okgRlKnN?8*3>-TbI?=%_8&*B_yVGY4jUT z{g3^j!?u_U_7I;jgL;yfq_~=dyX~7&TOEICWR8U2Y5Nl8t;&7g71(|g+`YV~E!$jz zwx2mFl>2|)3@yF(tVWM`vb}j>rEi5wVUN{F6fk&TU5=&#~hYp7TEX+cU}ijO%1t9_sQ*~ra=upEs{vaN*1&aAG+NA0TcRk;GZb1Q=J8~4!k_3z*Y z|B~Fm9e8r(``BxjL;(0Zf_kS9Iev~>Jl-Xttpkph1M~Nu^_ZV-c;-=$>^4>#D|#MS z)Zz98@G7~D->@DPu5yzNS^_`Kn?6^u1k0anei+|ZU1yK9`Gx0$j87$^3(XVq_Fp;a zm|5~JTmYG#g8Ugs^MRjNVhbLTEfbVD$YqaZ5>vVm)Sd`y115Ib7td3Tgtgz!!ym(? zp3mHYu2ZTTVPh})u(C9iOggY38SBLDHZt6UuzhjOzV9WU>NqeoL{s>g;ykV6nf2pp;hw?;%6 zZs`gN2775*!Lu_nKL~hN2uRPYyIUq~XbS0jRy4!Y?&mrtRmISCM<=B+ej(>`yuVyl z<8Hjw_iwDef^F>3MvLIoU^XL}(vJn-=L{yVXb+V4bvAkBE(34b0+V%S_||>VskKpm zU5%UBW~9qrZXjPD5Zd*OR+9y560@xTMp(1{VCcKc09pf27*&_3kHlfR<#0rLtqYxX zrIwc$p;)kl;2RyXH*L->St<(PMt)Wfm(Q2hk1pW^MMza*6u&_r*X!CGJ#(VIY%fl9 zl$_S~nAL4QOn5o}={tnR)_dblIIP6~nHXMDT2zZWjZk0@YVly73B$rfp&#fpi;TbT zqNnHjII;z&;T|VJt8Loc2)i=i8=d)F5mMa|b#zu6bmj${pT> zBJ6W9dDXA*zP%nmApE-jEqf8F+K@bVZkIb7+T6t1l#*3~@$(rYz^aIfl8T_O*ekbP zCD{DZFH;^h;`XYi1WWF;k73Cu_(zL}=rlN?`(B4*18_E5D8yvEaCYhqs3xVRD|! zP?W~@O(&n4m~&mH>a)iD{U zR;1V@JJob~CM=1n`#F45Db9W8l&>j2(AQIBaM&Ro=l%S;I|2Mwy$(j#_ z12vc&nSaOCc+o!lx#ezIexv0mB`w}SSo=_>{B^$!u~y{SZZfxhrAfKH#&f|G9JIsW z#PvKW=3!wqE#iv|BFDm<&S8ct70O&&P53E6jdZ056v1OATHPWjEeG|(>+5gUDsAh# zd{XCI3*q@?GlVB*cgEW9?I9o|%=tIQhI`(#2mEDSV<{-}F^Yh=rdw1rJlv6fZ#P_@ zm**El_!uutU;Yk&nw_kN$cV22wy4>jnML>ei4St^r@`;Dv>rf?fT(l9vh^MgMJCDD z>l*)K-Tq?_?THEnZnRx>Ia=_Yg#sdOE}UrSis z<2Kd~W0~O43KC}B4Ej_Kbo4+U0hx&u;L7%IHa2sUPdaVBQGt*6mHH6{r80AOzpH-@ z&ONAx5?JnY=(7>M4yqUnT4A3VF)cV*qVcY^!To{M@h>=gEGANV4lq`XAPx3M5@G{& z{~jE67o5IUze_6jh*XJ}jzmbiEVYq~o0+VmZLV7UcH6P>;)%pQE%!>SPn zyHkh0v1=6nc#TwRTMPHB!flXgiP2kYE=~4xN6D1MAcJqCGb!s=tX$#8tiG*>Rcf!8 zW((=kl4u!bo$wU5oCaP{tXn1U$#oXLYKdctBlx0(punmiuf+M&Pr!UxaTNjAFQ!_H zQXQvjZR4*QXoba-rZap7YHEn1V}NV~<;x#3u{R!ld3cm`JfzanKp>Xa<2N>GXItK% z(hd!+KE33Bg!G6oMN~ATaI`1n`(N5>zh$GD;c=}}`MY~~8F)D4pyW-fMWQWGD^{Qf zI;X_E8UQ7)+8=f5{5=iv2TD{m*IVx@sY|g&?~24zp^g)qPNgo#WNE zkak#AhkKQ-=cF#QrOd#dd*gCpk)1~Q-ItOw4r;EE&>z{Ni$z6K%<;AL{p@eCxd0uK zWv>Y0LnldDBmBE)vgaK}LXsV+gCp%+y&K4$j3HFS5e)mp%t~X&hWb&R@fLVS?Hexr zEXpQ8#y=ZAJ?yQ4K2q!Gz!;C`)yLerUfgaAV01KX`Ze3ix~t7gevMUYuED4`<1>wViG}*2eieGQ zz<$Z%UcTITr~QX=lt|1Ru)ubVUs{CPMh)l-zxrAfgBlOEoSbpe zgQq=@8G*RrX)Oi~mqepxer0>V{ApOX+rr;0B|xBbdaGz!3wDHOCg29dMyVG{I+=|6 z`ugFZmPA~o??hB^-YN|Aw?FpW5udL?09D8T9PVWC9trZc9Uk+F{3Rz^1nVTE+Q5I~ z(>XB2O%U6%CPgB}*eql39Fx0b zxDmA1iS1pr8w+izWo3%}hz2#_)NJO6&^t3)7@n>pAUf&OtL{7_FWPTZ$yI_Kv8fF- zH!t3t3!+;zD5a{Xr!ax&K2Y86emoR)84F`we>bqZV(yR80Q~cpc_}2wb>jmWO8xI2 z>7H-gVWREp%Q@|;b{X4;?XC)ln-c*%I&T(1E;pz6s_H9D!O4fM=RYGy zf+gsK`rJx&cP=AA4T}bIz;azu>;#{YK7MB!I8P~@D*&5MX&d&fQrty&aWrU{rFA99 znjt*qsC(npnwIX_?sCQGm8^;_fY@$^jTlsn(*ePZn3|f2_u=@J*0B_oy3|L!1R}(7oxqq<1pFrYF;rY9Q;$*;=@e8I>4;i_E{kDKv^@+w317#uZBL%)UJ~xK*>s2!` z{r!oI-4N1C!T@Gn3>8cv9SxoxH~`L$ORNN#+XnHxJ_LZK2+N3F zGoa+v>Lj+mzsIGM^6~ib<>VP@sStpXnH*8vl>~n*M{W2xU{}M`_5Inc8s)Qz>C;dA zqbW5TP@*8ImxW9Z_@S$$3ucm(6lAvXinkXv&!H|BH`AKn%k+^nn7XgVe#x?8G(j?V z4HoU%r#_yjhJy>v)(v@kmKwn0@_6FhJ8JUDls$Jr5 zf^Y*TvKDXTqt-vJi$KmF1(>MlN;s+e(!h!;WRA-ypoE2ZswEC8sVAVYP~0Vp38bm} z5fm|KCDo&J)tY%==MZjC-*pQ@07R5;bxv52*@2nGjds;eN`D;}etSR2)&n(y7p zxYCY-F(EtPe z$}~Ur=F3DwI$c&{lPd@^g}&jXwyfj}wN2*PoyeSHrR0IsU(Ud?Kinn-X`&KMkQ937 zoN3XAFolpGl8Ays?Lv3)tiX1F_#A{aZx%0w$c4T6KXi_tB z%)g64)6~T;D+BoczQjVtSrF$Y*ZZt;+XRv>?_c#9GYt7$?!m#S zkeA{&J*x?Cwk<3-CY2Z&?x-7zg#BTy=6E|X8z(BV8K-~gxaL&{myywSlecIKGA<3A z*|K9E&PAWkZJP{{mcGUS&Qj=z1~noQA_2@em=Bg|3NdkiEr9BYHwvCIH<*djlVjcSd4Nikdn}@r+vSE zzT4EXARQ@RF`lI{J>DyfM8{+SL$3MlcuaSD-%@?k)|vUrOwP$cC7ejtY0N>dFC3XA z*4^I?!jcQDhoq@1CM+jZ;?1Z|Wf2XS7pg zCF=jNWFx0C84Oa737{|U_Xl^|gDI(nluwA$stIay=*HVj+bm@dDPrUX+F@xY&fdP!*ajdR@Mr92)>j#n->Ep zBrCkuRI$-w4pXCd59%W*#pvNinBubt713_;8XHJ?Jhq!lOQf4L!~D*=h_8S4zC1T7 zn)SZmpxQ<&xRm`N;;HPI>fe8Gs-nb{tzy-QZ){XE$7hk2`Fpad7%?CG;}fPQSCVnG zl1#KK!85?4`9+_{m&7DJEsNsrRag!lATH#ChD%kU@^OG#gc9efJlh z9{(I!(4bfoQbsr1&Oc%kFd=G4mE?c=sNyiJ+Xo+D<0N&VNaVF1i{fLY#f7y}Ff6Y9 z!&AAhm?Tm?!4B-P$euh4&#etTfv!0KaH5z1=+HM4cug^WPq%lb0ofg6#g?7ie6r4g+%YWsx92;n~jMm=x;rn#Cl@m zWvcKEBl*EP(^6>N+SAZioVBmW#}-8cZ98q;_GK?7@1c1<^K78H9QR?;NoG?e2D@S0 z@qlVhj&mI-3M9k_wK86w<#|$2!P%waQ+>M0KcmgT5o%?eFryR2mp37AN12dE{(N^P z5vLh7Fscn?=OW_t4;J=x=WcwLXCoIq_a9f+?&vvE$L2c%G)<45(-iSw#N@7LG&HRn zQ#kUjyekp@1?!&5IiGJhU7}_oZZcHgu*DYSUv`kCyxWF@(CtjS!*fm=BQ5u2trTeG;lpVJAm7Lfg}x$n+6^Zkdy^ahTzK`q!8Q!F~s zY#UuyUr2!dAhkeeHpW(`_F{DI)NS`BJe7c5HRfl4WHN4w(7JSt&zUx60b~5hz+t1F zOET6Q3YX~3QsJ^7^7BOhmBr&!mxopNG67d-T>wH1l=D3zxoHBL3g7k|lzm%k+6LF#z4U$B(Kj-b?~W5F zaDK&86${2v3-C^gllAaC<65Cs1c@crw}^+A+8eVwsS|N4A6ELgxm#7-zPC~!-M?Ho zIUc1<*f5zhQj4|}dPcKrqp!WdC+;zAGb(vs9FqUL1bFsR+|tW1evxEKli~#r2u~Y^ zeBcp*1Qz30NPla0T8xEHJ(I~SI|%KXz|M>um;NQ zQm}YYzL{+CX>!IXEEs!S(8SlbUF^Dl6z0W`*HYQ8Uqa4T3z}q^p&r`npSVV|H!uke zG@cqmIXb3iFPb+Anhwz>F1qah{6S;8w3LJRE_KOeoljDwxEnjh9TNWJJPQkH^J!eJ zwJ|}!lMQ?bK5Jfzb|@}>eb1z*AwZT~! z>m(OGIR<{%zjkHO%=hlFYUE4`m{$|wvBV!W`)xoLN1}prv>zO?>29{)f~XPWv0I*7 z$uvdjVM%1g%xOOkp6DlU>wzVkv++%lS+&_YhzJF20S->s&}s>lMUx8uMoWI;S?Tsz;}ZR3~ta9kwf{+ll$EJ2Z9W&25v8*{JZkb|jtN=}JoKTp8??bj!AJ7T{Fwg-^> zdXgu>9}H)ufm4W9pAL9I(d~ys)wm0{3!iQu7sC^|lmJOR)A0XGDZp5R(^eUbDkP4< zDE9zvnSVSQ?=)$rb7{vluqvsk-i?zMGB;~lA+MiZg1fZ^9P=htg0b%OnSv<2n}Z99>0G}%n(9Xn4x~XHggn>AoJlj z47lOJgpW z|8n+zMlLz=h@mhYP?e?sgC9MJ?6z4Z@O{mqvu0NsJbdb1?n)xC-C|J+mptES@ce_S zLYijRG|w6%jDFvEJX#-iT*TvQ$Wf8Vt%jKmw~v0krG32he*|x%*D8I`8Z6R1a(D*`= zOW%9vfS@&#Oz|s;w{m?^04m`Ah6eH|z(s_EG|dh+>0J1@Uw}N`K<)ihgPQZzwjW#G z@a3h0lH|M%@=m^dErhHNIcvMvY5R!M*033{b9L!}EjK>3g~*WrT6YnC1R>}nvHjIF z^VWtH-O#^7Q{MDi8RuU-z3i&4SLDtd6xK$o!=lXp86 z9f}7zL{GzJO$XUFhGc>y*>31{Tf>3+)6FdruSjYy80PPDc=Ip23!_Q$ z?*r+qcgHaO$~>D*lczoDfn0jdSCyyyiR=4C@E(> zlaO8M*sucQ@t^Lo3z?W z?CzZ#J<(59TR~|OrDK>FWn2ksK)>GW6z@Upu}%eTM=If)Y7Z|4B+60z+e$GL*?SA@PU^$qPU&q%AqUB;9e$T!#=frhHXZWtfoqWx}v>DU8O~ z_If?jsy){x3Lq}A?V?cY56iv zE>Kv^bJ9d)V5F*rqN<@~R@wwB*Z~&G;#5(y25V=07uRO3Pl-ZD1wy}Ah1WXn5ETsm z!bsvkPpBAcJu9}ol?grIfk4B0};BC|Te;roP$TukD28+u;l(sr$RVDP`1iIzns&Qoy4roPyq_KN=6G zKU^03kLhLUDLG(GY*u$e(pLG~ykI((;9dglP~{Lcw>oL_A@LGU1<JfkKVa7z)3*2wUWG@ zGKDJL@oV^My^hNZTR&$AF7EB=#fT=mb+T#{^r@)W^Zqg6+3$QKv?3V++AazyQq`lSTCYAwVt)Yd>%7 zqqpg+@X1C1hEykG9w4e6ca%enrAMq<4=G>GrxotY+jXh6fxrGdo?c2)`F*3~P_V!w zLL<3QaQG^~Jc55}LnJt8nEJt^&wlLJffrSgss<^YwDIyatFEPb?H>w*8;tv2&fV{B zrm&T~gm}knVd#a{0C*}o`!nB9rLdn&Q%dS)zY4*z%kCa6CkV@{7-B z_8n$!IGvTVlLW@ttK}Su&?R)8EOV_T%TdQdy*bNu?00p~jhhyC1}XtltwbtDa3sZW zAzbQc7~SQ13e%0YKJm7z!1R((NYX4-LCVCsxT*P6OeVvyhto19ZW-tvP}A~4R2p-_ z#n9mGtX;A_V@8Eg5XtKWZ6@BjZjfW~%5|_tsuV#I?$ip&A-d9*J@C!NOuimPPu6<*uF_D`J(S4NS!-`^TJcA;VQh8Zrw_ZfBF{f5Y9O8M+^(5gGJlthV;O7ZU z*V@WtZKF3Tfr43F@#RaDVS$w;m5_8oXVEvzdNJ?L%0A%F12w9LKWodzoV_-u!ShYN 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 literal 0 HcmV?d00001 diff --git a/explains/minidataapi.html b/explains/minidataapi.html new file mode 100644 index 00000000..c417661d --- /dev/null +++ b/explains/minidataapi.html @@ -0,0 +1,1367 @@ + + + + + + + + + +MiniDataAPI Spec – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ +
+ + +
+ + + +
+ +
+
+

MiniDataAPI Spec

+
+ + + +
+ + + + +
+ + + +
+ + + +

The MiniDataAPI is a persistence API specification that designed to be small and relatively easy to implement across a wide range of datastores. While early implementations have been SQL-based, the specification can be quickly implemented in key/value stores, document databases, and more.

+
+
+
+ +
+
+Work in Progress +
+
+
+

The MiniData API spec is a work in progress, subject to change. While the majority of design is complete, expect there could be breaking changes.

+
+
+
+

Why?

+

The MiniDataAPI specification allows us to use the same API for many different database engines. Any application using the MiniDataAPI spec for interacting with its database requires no modification beyond import and configuration changes to switch database engines. For example, to convert an application from Fastlite running SQLite to FastSQL running PostgreSQL, should require only changing these two lines:

+
+
+

FastLite version

+
from fastlite import *
+db = database('test.db')
+
+

FastSQL version

+
from fastsql import *
+db = Database('postgres:...')
+
+
+

As both libraries adhere to the MiniDataAPI specification, the rest of the code in the application should remain the same. The advantage of the MiniDataAPI spec is that it allows people to use whatever datastores they have access to or prefer.

+
+
+
+ +
+
+Note +
+
+
+

Switching databases won’t migrate any existing data between databases.

+
+
+
+

Easy to learn, quick to implement

+

The MiniDataAPI specification is designed to be easy-to-learn and quick to implement. It focuses on straightforward Create, Read, Update, and Delete (CRUD) operations.

+

MiniDataAPI databases aren’t limited to just row-based systems. In fact, the specification is closer in design to a key/value store than a set of records. What’s exciting about this is we can write implementations for tools like Python dict stored as JSON, Redis, and even the venerable ZODB.

+
+
+

Limitations of the MiniDataAPI Specification

+
+

“Mini refers to the lightweightness of specification, not the data.”

+

– Jeremy Howard

+
+

The advantages of the MiniDataAPI come at a cost. The MiniDataAPI specification focuses a very small set of features compared to what can be found in full-fledged ORMs and query languages. It intentionally avoids nuances or sophisticated features.

+

This means the specification does not include joins or formal foreign keys. Complex data stored over multiple tables that require joins isn’t handled well. For this kind of scenario it’s probably for the best to use more sophisticated ORMs or even direct database queries.

+
+
+

Summary of the MiniDataAPI Design

+
    +
  • Easy-to-learn
  • +
  • Relative quick to implement for new database engines
  • +
  • An API for CRUD operations
  • +
  • For many different types of databases including row- and key/value-based designs
  • +
  • Intentionally small in terms of features: no joins, no foreign keys, no database specific features
  • +
  • Best for simpler designs, complex architectures will need more sophisticated tools.
  • +
+
+
+
+

Connect/construct the database

+

We connect or construct the database by passing in a string connecting to the database endpoint or a filepath representing the database’s location. While this example is for SQLite running in memory, other databases such as PostgreSQL, Redis, MongoDB, might instead use a URI pointing at the database’s filepath or endpoint. The method of connecting to a DB is not part of this API, but part of the underlying library. For instance, for fastlite:

+
+
db = database(':memory:')
+
+

Here’s a complete list of the available methods in the API, all documented below (assuming db is a database and t is a table):

+
    +
  • db.create
  • +
  • t.insert
  • +
  • t.delete
  • +
  • t.update
  • +
  • t[key]
  • +
  • t(...)
  • +
  • t.xtra
  • +
+
+
+

Tables

+

For the sake of expediency, this document uses a SQL example. However, tables can represent anything, not just the fundamental construct of a SQL databases. They might represent keys within a key/value structure or files on a hard-drive.

+
+

Creating tables

+

We use a create() method attached to Database object (db in our example) to create the tables.

+
+
class User: name:str; email: str; year_started:int
+users = db.create(User, pk='name')
+users
+
+
<Table user (name, email, year_started)>
+
+
+
+
class User: name:str; email: str; year_started:int
+users = db.create(User, pk='name')
+users
+
+
<Table user (name, email, year_started)>
+
+
+

If no pk is provided, id is assumed to be the primary key. Regardless of whether you mark a class as a dataclass or not, it will be turned into one – specifically into a flexiclass.

+
+
@dataclass
+class Todo: id: int; title: str; detail: str; status: str; name: str
+todos = db.create(Todo) 
+todos
+
+
<Table todo (id, title, detail, status, name)>
+
+
+
+
+

Compound primary keys

+

The MiniData API spec supports compound primary keys, where more than one column is used to identify records. We’ll also use this example to demonstrate creating a table using a dict of keyword arguments.

+
+
class Publication: authors: str; year: int; title: str
+publications = db.create(Publication, pk=('authors', 'year'))
+
+
+
+

Transforming tables

+

Depending on the database type, this method can include transforms - the ability to modify the tables. Let’s go ahead and add a password field for our table called pwd.

+
+
class User: name:str; email: str; year_started:int; pwd:str
+users = db.create(User, pk='name', transform=True)
+users
+
+
<Table user (name, email, year_started, pwd)>
+
+
+
+
+
+

Manipulating data

+

The specification is designed to provide as straightforward CRUD API (Create, Read, Update, and Delete) as possible. Additional features like joins are out of scope.

+
+

.insert()

+

Add a new record to the database. We want to support as many types as possible, for now we have tests for Python classes, dataclasses, and dicts. Returns an instance of the new record.

+

Here’s how to add a record using a Python class:

+
+
users.insert(User(name='Braden', email='b@example.com', year_started=2018))
+
+
User(name='Braden', email='b@example.com', year_started=2018, pwd=None)
+
+
+

We can also use keyword arguments directly:

+
+
users.insert(name='Alma', email='a@example.com', year_started=2019)
+
+
User(name='Alma', email='a@example.com', year_started=2019, pwd=None)
+
+
+

And now Charlie gets added via a Python dict.

+
+
users.insert({'name': 'Charlie', 'email': 'c@example.com', 'year_started': 2018})
+
+
User(name='Charlie', email='c@example.com', year_started=2018, pwd=None)
+
+
+

And now TODOs. Note that the inserted row is returned:

+
+
todos.insert(Todo(title='Write MiniDataAPI spec', status='open', name='Braden'))
+todos.insert(title='Implement SSE in FastHTML', status='open', name='Alma')
+todo = todos.insert(dict(title='Finish development of FastHTML', status='closed', name='Charlie'))
+todo
+
+
Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')
+
+
+

Let’s do the same with the Publications table.

+
+
publications.insert(Publication(authors='Alma', year=2019, title='FastHTML'))
+publications.insert(authors='Alma', year=2030, title='FastHTML and beyond')
+publication= publications.insert((dict(authors='Alma', year=2035, title='FastHTML, the early years')))
+publication
+
+
Publication(authors='Alma', year=2035, title='FastHTML, the early years')
+
+
+
+ + +
+

.update()

+

Update an existing record of the database. Must accept Python dict, dataclasses, and standard classes. Uses the primary key for identifying the record to be changed. Returns an instance of the updated record.

+

Here’s with a normal Python class:

+
+
user
+
+
User(name='Alma', email='a@example.com', year_started=2019, pwd=None)
+
+
+
+
user.year_started = 2099
+users.update(user)
+
+
User(name='Alma', email='a@example.com', year_started=2099, pwd=None)
+
+
+

Or use a dict:

+
+
users.update(dict(name='Alma', year_started=2199, email='a@example.com'))
+
+
User(name='Alma', email='a@example.com', year_started=2199, pwd=None)
+
+
+

Or use kwargs:

+
+
users.update(name='Alma', year_started=2149)
+
+
User(name='Alma', email='a@example.com', year_started=2149, pwd=None)
+
+
+

If the primary key doesn’t match a record, raise a NotFoundError.

+

John hasn’t started with us yet so doesn’t get the chance yet to travel in time.

+
+
try: users.update(User(name='John', year_started=2024, email='j@example.com'))
+except NotFoundError: print('User not found')
+
+
User not found
+
+
+
+
+

.delete()

+

Delete a record of the database. Uses the primary key for identifying the record to be removed. Returns a table object.

+

Charlie decides to not travel in time. He exits our little group.

+
+
users.delete('Charlie')
+
+
<Table user (name, email, year_started, pwd)>
+
+
+

If the primary key value can’t be found, raises a NotFoundError.

+
+
try: users.delete('Charlies')
+except NotFoundError: print('User not found')
+
+
User not found
+
+
+

In John’s case, he isn’t time travelling with us yet so can’t be removed.

+
+
try: users.delete('John')
+except NotFoundError: print('User not found')
+
+
User not found
+
+
+

Deleting records with compound primary keys requires providing the entire key.

+
+
publications.delete(['Alma' , 2035])
+
+
<Table publication (authors, year, title)>
+
+
+
+
+

in keyword

+

Are Alma and John contained in the Users table? Or, to be technically precise, is the item with the specified primary key value in this table?

+
+
'Alma' in users, 'John' in users
+
+
(True, False)
+
+
+

Also works with compound primary keys, as shown below. You’ll note that the operation can be done with either a list or tuple.

+
+
['Alma', 2019] in  publications
+
+
True
+
+
+

And now for a False result, where John has no publications.

+
+
('John', 1967) in publications
+
+
False
+
+
+
+
+

.xtra()

+

If we set fields within the .xtra function to a particular value, then indexing is also filtered by those. This applies to every database method except for record creation. This makes it easier to limit users (or other objects) access to only things for which they have permission. This is a one-way operation, once set it can’t be undone for a particular table object.

+

For example, if we query all our records below without setting values via the .xtra function, we can see todos for everyone. Pay special attention to the id values of all three records, as we are about to filter most of them away.

+
+
todos()
+
+
[Todo(id=1, title='Write MiniDataAPI spec', detail=None, status='open', name='Braden'),
+ Todo(id=2, title='Implement SSE in FastHTML', detail=None, status='open', name='Alma'),
+ Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')]
+
+
+

Let’s use .xtra to constrain results just to Charlie. We set the name field in Todos, but it could be any field defined for this table.

+
+
todos.xtra(name='Charlie')
+
+

We’ve now set a field to a value with .xtra, if we loop over all the records again, only those assigned to records with a name of Charlie will be displayed.

+
+
todos()
+
+
[Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')]
+
+
+

The in keyword is also affected. Only records with a name of Charlie will evaluate to be True. Let’s demonstrate by testing it with a Charlie record:

+
+
ct = todos[3]
+ct
+
+
Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')
+
+
+

Charlie’s record has an ID of 3. Here we demonstrate that Charlie’s TODO can be found in the list of todos:

+
+
ct.id in todos
+
+
True
+
+
+

If we try in with the other IDs the query fails because the filtering is now set to just records with a name of Charlie.

+
+
1 in todos, 2 in todos
+
+
(False, False)
+
+
+
+
try: todos[2]
+except NotFoundError: print('Record not found')
+
+
Record not found
+
+
+

We are also constrained by what records we can update. In the following example we try to update a TODO not named ‘Charlie’. Because the name is wrong, the .update function will raise a NotFoundError.

+
+
try: todos.update(Todo(id=1, title='Finish MiniDataAPI Spec', status='closed', name='Braden'))
+except NotFoundError as e: print('Record not updated')
+
+
Record not updated
+
+
+

Unlike poor Braden, Charlie isn’t filtered out. Let’s update his TODO.

+
+
todos.update(Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie'))
+
+
Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')
+
+
+

Finally, once constrained by .xtra, only records with Charlie as the name can be deleted.

+
+
try: todos.delete(1)
+except NotFoundError as e: print('Record not updated')
+
+
Record not updated
+
+
+

Charlie’s TODO was to finish development of FastHTML. While the framework will stabilize, like any good project it will see new features added and the odd bug corrected for many years to come. Therefore, Charlie’s TODO is nonsensical. Let’s delete it.

+
+
todos.delete(ct.id)
+
+
<Table todo (id, title, detail, status, name)>
+
+
+

When a TODO is inserted, the xtra fields are automatically set. This ensures that we don’t accidentally, for instance, insert items for others users. Note that here we don’t set the name field, but it’s still included in the resultant row:

+
+
ct = todos.insert(Todo(title='Rewrite personal site in FastHTML', status='open'))
+ct
+
+
Todo(id=3, title='Rewrite personal site in FastHTML', detail=None, status='open', name='Charlie')
+
+
+

If we try to change the username to someone else, the change is ignored, due to xtra:

+
+
ct.name = 'Braden'
+todos.update(ct)
+
+
Todo(id=3, title='Rewrite personal site in FastHTML', detail=None, status='open', name='Charlie')
+
+
+
+
+
+

SQL-first design

+
+
users = None
+User = None
+
+
+
users = db.t.user
+users
+
+
<Table user (name, email, year_started, pwd)>
+
+
+

(This section needs to be documented properly.)

+

From the table objects we can extract a Dataclass version of our tables. Usually this is given an singular uppercase version of our table name, which in this case is User.

+
+
User = users.dataclass()
+
+
+
User(name='Braden', email='b@example.com', year_started=2018)
+
+
User(name='Braden', email='b@example.com', year_started=2018, pwd=UNSET)
+
+
+
+
+

Implementations

+
+

Implementing MiniDataAPI for a new datastore

+

For creating new implementations, the code examples in this specification are the test case for the API. New implementations should pass the tests in order to be compliant with the specification.

+
+
+

Implementations

+
    +
  • fastlite - The original implementation, only for Sqlite
  • +
  • fastsql - An SQL database agnostic implementation based on the excellent SQLAlchemy library.
  • +
+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/explains/minidataapi.html.md b/explains/minidataapi.html.md new file mode 100644 index 00000000..0259fc3b --- /dev/null +++ b/explains/minidataapi.html.md @@ -0,0 +1,668 @@ +# MiniDataAPI Spec + + + + +The `MiniDataAPI` is a persistence API specification that designed to be +small and relatively easy to implement across a wide range of +datastores. While early implementations have been SQL-based, the +specification can be quickly implemented in key/value stores, document +databases, and more. + +
+ +> **Work in Progress** +> +> The MiniData API spec is a work in progress, subject to change. While +> the majority of design is complete, expect there could be breaking +> changes. + +
+ +## Why? + +The MiniDataAPI specification allows us to use the same API for many +different database engines. Any application using the MiniDataAPI spec +for interacting with its database requires no modification beyond import +and configuration changes to switch database engines. For example, to +convert an application from Fastlite running SQLite to FastSQL running +PostgreSQL, should require only changing these two lines: + +
+ +
+ +FastLite version + +``` python +from fastlite import * +db = database('test.db') +``` + +
+ +
+ +FastSQL version + +``` python +from fastsql import * +db = Database('postgres:...') +``` + +
+ +
+ +As both libraries adhere to the MiniDataAPI specification, the rest of +the code in the application should remain the same. The advantage of the +MiniDataAPI spec is that it allows people to use whatever datastores +they have access to or prefer. + +
+ +> **Note** +> +> Switching databases won’t migrate any existing data between databases. + +
+ +### Easy to learn, quick to implement + +The MiniDataAPI specification is designed to be easy-to-learn and quick +to implement. It focuses on straightforward Create, Read, Update, and +Delete (CRUD) operations. + +MiniDataAPI databases aren’t limited to just row-based systems. In fact, +the specification is closer in design to a key/value store than a set of +records. What’s exciting about this is we can write implementations for +tools like Python dict stored as JSON, Redis, and even the venerable +ZODB. + +### Limitations of the MiniDataAPI Specification + +> “Mini refers to the lightweightness of specification, not the data.” +> +> – Jeremy Howard + +The advantages of the MiniDataAPI come at a cost. The MiniDataAPI +specification focuses a very small set of features compared to what can +be found in full-fledged ORMs and query languages. It intentionally +avoids nuances or sophisticated features. + +This means the specification does not include joins or formal foreign +keys. Complex data stored over multiple tables that require joins isn’t +handled well. For this kind of scenario it’s probably for the best to +use more sophisticated ORMs or even direct database queries. + +### Summary of the MiniDataAPI Design + +- Easy-to-learn +- Relative quick to implement for new database engines +- An API for CRUD operations +- For many different types of databases including row- and + key/value-based designs +- Intentionally small in terms of features: no joins, no foreign keys, + no database specific features +- Best for simpler designs, complex architectures will need more + sophisticated tools. + +## Connect/construct the database + +We connect or construct the database by passing in a string connecting +to the database endpoint or a filepath representing the database’s +location. While this example is for SQLite running in memory, other +databases such as PostgreSQL, Redis, MongoDB, might instead use a URI +pointing at the database’s filepath or endpoint. The method of +connecting to a DB is *not* part of this API, but part of the underlying +library. For instance, for fastlite: + +``` python +db = database(':memory:') +``` + +Here’s a complete list of the available methods in the API, all +documented below (assuming `db` is a database and `t` is a table): + +- `db.create` +- `t.insert` +- `t.delete` +- `t.update` +- `t[key]` +- `t(...)` +- `t.xtra` + +## Tables + +For the sake of expediency, this document uses a SQL example. However, +tables can represent anything, not just the fundamental construct of a +SQL databases. They might represent keys within a key/value structure or +files on a hard-drive. + +### Creating tables + +We use a `create()` method attached to `Database` object (`db` in our +example) to create the tables. + +``` python +class User: name:str; email: str; year_started:int +users = db.create(User, pk='name') +users +``` + + + +``` python +class User: name:str; email: str; year_started:int +users = db.create(User, pk='name') +users +``` + +
+ +If no `pk` is provided, `id` is assumed to be the primary key. +Regardless of whether you mark a class as a dataclass or not, it will be +turned into one – specifically into a +[`flexiclass`](https://fastcore.fast.ai/xtras.html#flexiclass). + +``` python +@dataclass +class Todo: id: int; title: str; detail: str; status: str; name: str +todos = db.create(Todo) +todos +``` + +
+ +### Compound primary keys + +The MiniData API spec supports compound primary keys, where more than +one column is used to identify records. We’ll also use this example to +demonstrate creating a table using a dict of keyword arguments. + +``` python +class Publication: authors: str; year: int; title: str +publications = db.create(Publication, pk=('authors', 'year')) +``` + +### Transforming tables + +Depending on the database type, this method can include transforms - the +ability to modify the tables. Let’s go ahead and add a password field +for our table called `pwd`. + +``` python +class User: name:str; email: str; year_started:int; pwd:str +users = db.create(User, pk='name', transform=True) +users +``` + +
+ +## Manipulating data + +The specification is designed to provide as straightforward CRUD API +(Create, Read, Update, and Delete) as possible. Additional features like +joins are out of scope. + +### .insert() + +Add a new record to the database. We want to support as many types as +possible, for now we have tests for Python classes, dataclasses, and +dicts. Returns an instance of the new record. + +Here’s how to add a record using a Python class: + +``` python +users.insert(User(name='Braden', email='b@example.com', year_started=2018)) +``` + + User(name='Braden', email='b@example.com', year_started=2018, pwd=None) + +We can also use keyword arguments directly: + +``` python +users.insert(name='Alma', email='a@example.com', year_started=2019) +``` + + User(name='Alma', email='a@example.com', year_started=2019, pwd=None) + +And now Charlie gets added via a Python dict. + +``` python +users.insert({'name': 'Charlie', 'email': 'c@example.com', 'year_started': 2018}) +``` + + User(name='Charlie', email='c@example.com', year_started=2018, pwd=None) + +And now TODOs. Note that the inserted row is returned: + +``` python +todos.insert(Todo(title='Write MiniDataAPI spec', status='open', name='Braden')) +todos.insert(title='Implement SSE in FastHTML', status='open', name='Alma') +todo = todos.insert(dict(title='Finish development of FastHTML', status='closed', name='Charlie')) +todo +``` + + Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie') + +Let’s do the same with the `Publications` table. + +``` python +publications.insert(Publication(authors='Alma', year=2019, title='FastHTML')) +publications.insert(authors='Alma', year=2030, title='FastHTML and beyond') +publication= publications.insert((dict(authors='Alma', year=2035, title='FastHTML, the early years'))) +publication +``` + + Publication(authors='Alma', year=2035, title='FastHTML, the early years') + +### Square bracket search \[\] + +Get a single record by entering a primary key into a table object within +square brackets. Let’s see if we can find Alma. + +``` python +user = users['Alma'] +user +``` + + User(name='Alma', email='a@example.com', year_started=2019, pwd=None) + +If no record is found, a `NotFoundError` error is raised. Here we look +for David, who hasn’t yet been added to our users table. + +``` python +try: users['David'] +except NotFoundError: print(f'User not found') +``` + + User not found + +Here’s a demonstration of a ticket search, demonstrating how this works +with non-string primary keys. + +``` python +todos[1] +``` + + Todo(id=1, title='Write MiniDataAPI spec', detail=None, status='open', name='Braden') + +Compound primary keys can be supplied in lists or tuples, in the order +they were defined. In this case it is the `authors` and `year` columns. + +Here’s a query by compound primary key done with a `list`: + +``` python +publications[['Alma', 2019]] +``` + + Publication(authors='Alma', year=2019, title='FastHTML') + +Here’s the same query done directly with index args. + +``` python +publications['Alma', 2030] +``` + + Publication(authors='Alma', year=2030, title='FastHTML and beyond') + +### Parentheses search () + +Get zero to many records by entering values with parentheses searches. +If nothing is in the parentheses, then everything is returned. + +``` python +users() +``` + + [User(name='Braden', email='b@example.com', year_started=2018, pwd=None), + User(name='Alma', email='a@example.com', year_started=2019, pwd=None), + User(name='Charlie', email='c@example.com', year_started=2018, pwd=None)] + +We can order the results. + +``` python +users(order_by='name') +``` + + [User(name='Alma', email='a@example.com', year_started=2019, pwd=None), + User(name='Braden', email='b@example.com', year_started=2018, pwd=None), + User(name='Charlie', email='c@example.com', year_started=2018, pwd=None)] + +We can filter on the results: + +``` python +users(where="name='Alma'") +``` + + [User(name='Alma', email='a@example.com', year_started=2019, pwd=None)] + +Generally you probably want to use placeholders, to avoid SQL injection +attacks: + +``` python +users("name=?", ('Alma',)) +``` + + [User(name='Alma', email='a@example.com', year_started=2019, pwd=None)] + +We can limit results with the `limit` keyword: + +``` python +users(limit=1) +``` + + [User(name='Braden', email='b@example.com', year_started=2018, pwd=None)] + +If we’re using the `limit` keyword, we can also use the `offset` keyword +to start the query later. + +``` python +users(limit=5, offset=1) +``` + + [User(name='Alma', email='a@example.com', year_started=2019, pwd=None), + User(name='Charlie', email='c@example.com', year_started=2018, pwd=None)] + +### .update() + +Update an existing record of the database. Must accept Python dict, +dataclasses, and standard classes. Uses the primary key for identifying +the record to be changed. Returns an instance of the updated record. + +Here’s with a normal Python class: + +``` python +user +``` + + User(name='Alma', email='a@example.com', year_started=2019, pwd=None) + +``` python +user.year_started = 2099 +users.update(user) +``` + + User(name='Alma', email='a@example.com', year_started=2099, pwd=None) + +Or use a dict: + +``` python +users.update(dict(name='Alma', year_started=2199, email='a@example.com')) +``` + + User(name='Alma', email='a@example.com', year_started=2199, pwd=None) + +Or use kwargs: + +``` python +users.update(name='Alma', year_started=2149) +``` + + User(name='Alma', email='a@example.com', year_started=2149, pwd=None) + +If the primary key doesn’t match a record, raise a `NotFoundError`. + +John hasn’t started with us yet so doesn’t get the chance yet to travel +in time. + +``` python +try: users.update(User(name='John', year_started=2024, email='j@example.com')) +except NotFoundError: print('User not found') +``` + + User not found + +### .delete() + +Delete a record of the database. Uses the primary key for identifying +the record to be removed. Returns a table object. + +Charlie decides to not travel in time. He exits our little group. + +``` python +users.delete('Charlie') +``` + +
+ +If the primary key value can’t be found, raises a `NotFoundError`. + +``` python +try: users.delete('Charlies') +except NotFoundError: print('User not found') +``` + + User not found + +In John’s case, he isn’t time travelling with us yet so can’t be +removed. + +``` python +try: users.delete('John') +except NotFoundError: print('User not found') +``` + + User not found + +Deleting records with compound primary keys requires providing the +entire key. + +``` python +publications.delete(['Alma' , 2035]) +``` + +
+ +### `in` keyword + +Are `Alma` and `John` contained `in` the Users table? Or, to be +technically precise, is the item with the specified primary key value +`in` this table? + +``` python +'Alma' in users, 'John' in users +``` + + (True, False) + +Also works with compound primary keys, as shown below. You’ll note that +the operation can be done with either a `list` or `tuple`. + +``` python +['Alma', 2019] in publications +``` + + True + +And now for a `False` result, where John has no publications. + +``` python +('John', 1967) in publications +``` + + False + +### .xtra() + +If we set fields within the `.xtra` function to a particular value, then +indexing is also filtered by those. This applies to every database +method except for record creation. This makes it easier to limit users +(or other objects) access to only things for which they have permission. +This is a one-way operation, once set it can’t be undone for a +particular table object. + +For example, if we query all our records below without setting values +via the `.xtra` function, we can see todos for everyone. Pay special +attention to the `id` values of all three records, as we are about to +filter most of them away. + +``` python +todos() +``` + + [Todo(id=1, title='Write MiniDataAPI spec', detail=None, status='open', name='Braden'), + Todo(id=2, title='Implement SSE in FastHTML', detail=None, status='open', name='Alma'), + Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')] + +Let’s use `.xtra` to constrain results just to Charlie. We set the +`name` field in Todos, but it could be any field defined for this table. + +``` python +todos.xtra(name='Charlie') +``` + +We’ve now set a field to a value with `.xtra`, if we loop over all the +records again, only those assigned to records with a `name` of `Charlie` +will be displayed. + +``` python +todos() +``` + + [Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')] + +The `in` keyword is also affected. Only records with a `name` of Charlie +will evaluate to be `True`. Let’s demonstrate by testing it with a +Charlie record: + +``` python +ct = todos[3] +ct +``` + + Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie') + +Charlie’s record has an ID of 3. Here we demonstrate that Charlie’s TODO +can be found in the list of todos: + +``` python +ct.id in todos +``` + + True + +If we try `in` with the other IDs the query fails because the filtering +is now set to just records with a name of Charlie. + +``` python +1 in todos, 2 in todos +``` + + (False, False) + +``` python +try: todos[2] +except NotFoundError: print('Record not found') +``` + + Record not found + +We are also constrained by what records we can update. In the following +example we try to update a TODO not named ‘Charlie’. Because the name is +wrong, the `.update` function will raise a `NotFoundError`. + +``` python +try: todos.update(Todo(id=1, title='Finish MiniDataAPI Spec', status='closed', name='Braden')) +except NotFoundError as e: print('Record not updated') +``` + + Record not updated + +Unlike poor Braden, Charlie isn’t filtered out. Let’s update his TODO. + +``` python +todos.update(Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')) +``` + + Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie') + +Finally, once constrained by `.xtra`, only records with Charlie as the +name can be deleted. + +``` python +try: todos.delete(1) +except NotFoundError as e: print('Record not updated') +``` + + Record not updated + +Charlie’s TODO was to finish development of FastHTML. While the +framework will stabilize, like any good project it will see new features +added and the odd bug corrected for many years to come. Therefore, +Charlie’s TODO is nonsensical. Let’s delete it. + +``` python +todos.delete(ct.id) +``` + +
+ +When a TODO is inserted, the `xtra` fields are automatically set. This +ensures that we don’t accidentally, for instance, insert items for +others users. Note that here we don’t set the `name` field, but it’s +still included in the resultant row: + +``` python +ct = todos.insert(Todo(title='Rewrite personal site in FastHTML', status='open')) +ct +``` + + Todo(id=3, title='Rewrite personal site in FastHTML', detail=None, status='open', name='Charlie') + +If we try to change the username to someone else, the change is ignored, +due to `xtra`: + +``` python +ct.name = 'Braden' +todos.update(ct) +``` + + Todo(id=3, title='Rewrite personal site in FastHTML', detail=None, status='open', name='Charlie') + +## SQL-first design + +``` python +users = None +User = None +``` + +``` python +users = db.t.user +users +``` + +
+ +(This section needs to be documented properly.) + +From the table objects we can extract a Dataclass version of our tables. +Usually this is given an singular uppercase version of our table name, +which in this case is `User`. + +``` python +User = users.dataclass() +``` + +``` python +User(name='Braden', email='b@example.com', year_started=2018) +``` + + User(name='Braden', email='b@example.com', year_started=2018, pwd=UNSET) + +## Implementations + +### Implementing MiniDataAPI for a new datastore + +For creating new implementations, the code examples in this +specification are the test case for the API. New implementations should +pass the tests in order to be compliant with the specification. + +### Implementations + +- [fastlite](https://github.com/AnswerDotAI/fastlite) - The original + implementation, only for Sqlite +- [fastsql](https://github.com/AnswerDotAI/fastsql) - An SQL database + agnostic implementation based on the excellent SQLAlchemy library. diff --git a/explains/oauth.html b/explains/oauth.html new file mode 100644 index 00000000..ecf6f481 --- /dev/null +++ b/explains/oauth.html @@ -0,0 +1,1086 @@ + + + + + + + + + +OAuth – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

OAuth

+
+ + + +
+ + + + +
+ + + +
+ + + +

OAuth is an open standard for ‘access delegation’, commonly used as a way for Internet users to grant websites or applications access to their information on other websites but without giving them the passwords. It is the mechanism that enables “Log in with Google” on many sites, saving you from having to remember and manage yet another password. Like many auth-related topics, there’s a lot of depth and complexity to the OAuth standard, but once you understand the basic usage it can be a very convenient alternative to managing your own user accounts.

+

On this page you’ll see how to use OAuth with FastHTML to implement some common pieces of functionality.

+
+

Creating an Client

+

FastHTML has Client classes for managing settings and state for different OAuth providers. Currently implemented are: GoogleAppClient, GitHubAppClient, HuggingFaceClient and DiscordAppClient - see the source if you need to add other providers. You’ll need a client_id and client_secret from the provider (see the from-scratch example later in this page for an example of registering with GitHub) to create the client. We recommend storing these in environment variables, rather than hardcoding them in your code.

+
+
import os
+from fasthtml.oauth import GoogleAppClient
+client = GoogleAppClient(os.getenv("AUTH_CLIENT_ID"),
+                         os.getenv("AUTH_CLIENT_SECRET"))
+
+

The client is used to obtain a login link and to manage communications between your app and the OAuth provider (client.login_link(redirect_uri="/redirect")).

+
+
+

Using the OAuth class

+

Once you’ve set up a client, adding OAuth to a FastHTML app can be as simple as:

+
+
from fasthtml.oauth import OAuth
+from fasthtml.common import FastHTML, RedirectResponse
+
+class Auth(OAuth):
+    def get_auth(self, info, ident, session, state):
+        email = info.email or ''
+        if info.email_verified and email.split('@')[-1]=='answer.ai':
+            return RedirectResponse('/', status_code=303)
+
+app = FastHTML()
+oauth = Auth(app, client)
+
+@app.get('/')
+def home(auth): return P('Logged in!'), A('Log out', href='/logout')
+
+@app.get('/login')
+def login(req): return Div(P("Not logged in"), A('Log in', href=oauth.login_link(req)))
+
+

There’s a fair bit going on here, so let’s unpack what’s happening in that code:

+
    +
  • OAuth (and by extension our custom Auth class) has a number of default arguments, including some key URLs: redir_path='/redirect', error_path='/error', logout_path='/logout', login_path='/login'. It will create and handle the redirect and logout paths, and it’s up to you to handle /login (where unsuccessful login attempts will be redirected) and /error (for oauth errors).
  • +
  • When we run oauth = Auth(app, client) it adds the redirect and logout paths to the app and also adds some beforeware. This beforeware runs on any requests (apart from any specified with the skip parameter).
  • +
+

The added beforeware specifies some app behaviour:

+
    +
  • If someone who isn’t logged in attempts to visit our homepage (/) here, they will be redirected to /login.
  • +
  • If they are logged in, it calls a check_invalid method. This defaults to False, which let’s the user continue to the page they requested. The behaviour can be modified by defining your own check_invalid method in the Auth class - for example, you could have this forcibly log out users who have recently been banned.
  • +
+

So how does someone log in? If they visit (or are redirected to) the login page at /login, we show them a login link. This sends them to the OAuth provider, where they’ll go through the steps of selecting their account, giving permissions etc. Once done they will be redirected back to /redirect. Behind the scenes a code that comes as part of their request gets turned into user info, which is then passed to the key function get_auth(self, info, ident, session, state). Here is where you’d handle looking up or adding a user in a database, checking for some condition (for example, this code checks if the email is an answer.ai email address) or choosing the destination based on state. The arguments are:

+
    +
  • self: the Auth object, which you can use to access the client (self.cli)
  • +
  • info: the information provided by the OAuth provider, typically including a unique user id, email address, username and other metadata.
  • +
  • ident: a unique identifier for this user. What this looks like varies between providers. This is useful for managing a database of users, for example.
  • +
  • session: the current session, that you can store information in securely
  • +
  • state: you can optionally pass in some state when creating the login link. This persists and is returned after the user goes through the Oath steps, which is useful for returning them to the same page they left. It can also be used as added security against CSRF attacks.
  • +
+

In our example, we check the email in info (we use a GoogleAppClient, not all providers will include an email). If we aren’t happy, and get_auth returns False or nothing (as in the case here for non-answerai people) then the user is redirected back to the login page. But if everything looks good we return a redirect to the homepage, and an auth key is added to the session and the scope containing the users identity ident. So, for example, in the homepage route we could use auth to look up this particular user’s profile info and customize the page accordingly. This auth will persist in their session until they clear the browser cache, so by default they’ll stay logged in. To log them out, remove it ( session.pop('auth', None)) or send them to /logout which will do that for you.

+
+
+

Explaining OAuth with a from-scratch implementation

+

Hopefully the example above is enough to get you started. You can also check out the (fairly minimal) source code where this is implemented, and the examples here.

+

If you’re wanting to learn more about how this works, and to see where you might add additional functionality, the rest of this page will walk through some examples without the OAuth convenience class, to illustrate the concepts. This was writted before said OAuth class was available, and is kep here for educational purposes - we recommend you stick with the new approach shown above in most cases.

+
+
+

A Minimal Login Flow (GitHub)

+

Let’s begin by building a minimal ‘Sign in with GitHub’ flow. This will demonstrate the basic steps of OAuth.

+

OAuth requires a “provider” (in this case, GitHub) to authenticate the user. So the first step when setting up our app is to register with GitHub to set things up.

+

Go to https://github.com/settings/developers and click “New OAuth App”. Fill in the form with the following values, then click ‘Register application’.

+
    +
  • Application name: Your app name
  • +
  • Homepage URL: http://localhost:8000 (or whatever URL you’re using - you can change this later)
  • +
  • Authorization callback URL: http://localhost:8000/auth_redirect (you can modify this later too)
  • +
+
+Setting up an OAuth app in GitHub +
+

After you register, you’ll see a screen where you can view the client ID and generate a client secret. Store these values in a safe place. You’ll use them to create a GitHubAppClient object in FastHTML.

+

This client object is responsible for handling the parts of the OAuth flow which depend on direct communication between your app and GitHub, as opposed to interactions which go through the user’s browser via redirects.

+

Here is how to setup the client object:

+
client = GitHubAppClient(
+    client_id="your_client_id",
+    client_secret="your_client_secret"
+)
+

You should also save the path component of the authorization callback URL which you provided on registration.

+

This route is where GitHub will redirect the user’s browser in order to send an authorization code to your app. You should save only the URL’s path component rather than the entire URL because you want your code to work automatically in deployment, when the host and port part of the URL change from localhost:8000 to your real DNS name.

+

Save the special authorization callback path under an obvious name:

+
auth_callback_path = "/auth_redirect"
+
+
+
+ +
+
+Note +
+
+
+

It’s recommended to store the client ID, and secret, in environment variables, rather than hardcoding them in your code.

+
+
+

When the user visit a normal page of your app, if they are not already logged in, then you’ll want to redirect them to your app’s login page, which will live at the /login path. We accomplish that by using this piece of “beforeware”, which defines logic which runs before other work for all routes except ones we specify to be skipped:

+
def before(req, session):
+    auth = req.scope['auth'] = session.get('user_id', None)
+    if not auth: return RedirectResponse('/login', status_code=303)
+    counts.xtra(name=auth)
+bware = Beforeware(before, skip=['/login', auth_callback_path])
+

We configure the beforeware to skip /login because that’s where the user goes to login, and we also skip the special authorization callback path because that is used by OAuth itself to receive information from GitHub.

+

It’s only at your login page that we start the OAuth flow. To start the OAuth flow, you need to give the user a link to GitHub’s login for your app. You’ll need the client object to generate that link, and the client object will in turn need the full authorization callback URL, which we need to build from the authorization callback path, so it is a multi-step process to produce this GitHub login link.

+

Here is an implementation of your own /login route handler. It generates the GitHub login link and presents it to the user:

+
@app.get('/login')
+def login(request)
+    redir = redir_url(request,auth_callback_path)
+    login_link = client.login_link(redir)
+    return P(A('Login with GitHub', href=login_link))    
+

Once the user follows that link, GitHub will ask them to grant permission to your app to access their GitHub account. If they agree, GitHub will redirect them back to your app’s authorization callback URL, carrying an authorization code which your app can use to generate an access token. To receive this code, you need to set up a route in FastHTML that listens for requests at the authorization callback path. For example:

+
@app.get(auth_callback_path)
+def auth_redirect(code:str):
+    return P(f"code: {code}")
+

This authorization code is temporary, and is used by your app to directly ask the provider for user information like an access token.

+

To recap, you can think of the exchange so far as:

+
    +
  • User to us: “I want to log in with you, app.”
  • +
  • Us to User: “Okay but first, here’s a special link to log in with GitHub”
  • +
  • User to GitHub: “I want to log in with you, GitHub, to use this app.”
  • +
  • GitHub to User: “OK, redirecting you back to the app’s URL (with an auth code)”
  • +
  • User to Us: “Hi again, app. Here’s the GitHub auth code you need to ask GitHub for info about me” (delivered via /auth_redirect?code=...)
  • +
+

The final steps we need to implement are as follows:

+
    +
  • Us to GitHUb: “A user just gave me this auth code. May I have the user info (e.g., an access token)?”
  • +
  • GitHub to us: “Since you have an auth code, here’s the user info”
  • +
+

It’s critical for us to derive the user info from the auth code immediately in the authorization callback, because the auth code may be used only once. So we use it that once in order to get information like an access token, which will remain valid for longer.

+

To go from the auth code to user info, you use info = client.retr_info(code,redirect_uri). From the user info, you can extract the user_id, which is a unique identifier for the user:

+
@app.get(auth_callback_path)
+def auth_redirect(code:str, request):
+    redir = redir_url(request, auth_callback_path)
+    user_info = client.retr_info(code, redir)
+    user_id = info[client.id_key]
+    return P(f"User id: {user_id}")
+

But we want the user ID not to print it but to remember the user.

+

So let us store it in the session object, to remember who is logged in:

+
@app.get(auth_callback_path)
+def auth_redirect(code:str, request, session):
+    redir = redir_url(request, auth_callback_path)
+    user_info = client.retr_info(code, redir)
+    user_id = user_info[client.id_key] # get their ID
+    session['user_id'] = user_id # save ID in the session
+    return RedirectResponse('/', status_code=303)
+

The session object is derived from values visible to the user’s browser, but it is cryptographically signed so the user can’t read it themselves. This makes it safe to store even information we don’t want to expose to the user.

+

For larger quantities of data, we’d want to save that information in a database and use the session to hold keys to lookup information from that database.

+

Here’s a minimal app that puts all these pieces together. It uses the user info to get the user_id. It stores that in the session object. It then uses the user_id as a key into a database, which tracks how frequently every user has hit an increment button.

+
import os
+from fasthtml.common import *
+from fasthtml.oauth import GitHubAppClient, redir_url
+
+db = database('data/counts.db')
+counts = db.t.counts
+if counts not in db.t: counts.create(dict(name=str, count=int), pk='name')
+Count = counts.dataclass()
+
+# Auth client setup for GitHub
+client = GitHubAppClient(os.getenv("AUTH_CLIENT_ID"), 
+                         os.getenv("AUTH_CLIENT_SECRET"))
+auth_callback_path = "/auth_redirect"
+
+def before(req, session):
+    # if not logged in, we send them to our login page
+    # logged in means:
+    # - 'user_id' in the session object, 
+    # - 'auth' in the request object
+    auth = req.scope['auth'] = session.get('user_id', None)
+    if not auth: return RedirectResponse('/login', status_code=303)
+    counts.xtra(name=auth)
+bware = Beforeware(before, skip=['/login', auth_callback_path])
+
+app = FastHTML(before=bware)
+
+# User asks us to Login
+@app.get('/login')
+def login(request):
+    redir = redir_url(request,auth_callback_path)
+    login_link = client.login_link(redir)
+    # we tell user to login at github
+    return P(A('Login with GitHub', href=login_link))    
+
+# User comes back to us with an auth code from Github
+@app.get(auth_callback_path)
+def auth_redirect(code:str, request, session):
+    redir = redir_url(request, auth_callback_path)
+    user_info = client.retr_info(code, redir)
+    user_id = user_info[client.id_key] # get their ID
+    session['user_id'] = user_id # save ID in the session
+    # create a db entry for the user
+    if user_id not in counts: counts.insert(name=user_id, count=0)
+    return RedirectResponse('/', status_code=303)
+
+@app.get('/')
+def home(auth):
+    return Div(
+        P("Count demo"),
+        P(f"Count: ", Span(counts[auth].count, id='count')),
+        Button('Increment', hx_get='/increment', hx_target='#count'),
+        P(A('Logout', href='/logout'))
+    )
+
+@app.get('/increment')
+def increment(auth):
+    c = counts[auth]
+    c.count += 1
+    return counts.upsert(c).count
+
+@app.get('/logout')
+def logout(session):
+    session.pop('user_id', None)
+    return RedirectResponse('/login', status_code=303)
+
+serve()
+

Some things to note:

+
    +
  • The before function is used to check if the user is authenticated. If not, they are redirected to the login page.
  • +
  • To log the user out, we remove the user ID from the session.
  • +
  • Calling counts.xtra(name=auth) ensures that only the row corresponding to the current user is accessible when responding to a request. This is often nicer than trying to remember to filter the data in every route, and lowers the risk of accidentally leaking data.
  • +
  • In the auth_redirect route, we store the user ID in the session and create a new row in the user_counts table if it doesn’t already exist.
  • +
+

You can find more heavily-commented version of this code in the oauth directory in fasthtml-example, along with an even more minimal example. More examples may be added in the future.

+
+

Revoking Tokens (Google)

+

When the user in the example above logs out, we remove their user ID from the session. However, the user is still logged in to GitHub. If they click ‘Login with GitHub’ again, they’ll be redirected back to our site without having to log in again. This is because GitHub remembers that they’ve already granted our app permission to access their account. Most of the time this is convenient, but for testing or security purposes you may want a way to revoke this permission.

+

As a user, you can usually revoke access to an app from the provider’s website (for example, https://github.com/settings/applications). But as a developer, you can also revoke access programmatically - at least with some providers. This requires keeping track of the access token (stored in client.token["access_token"] after you call retr_info), and sending a request to the provider’s revoke URL:

+
auth_revoke_url = "https://accounts.google.com/o/oauth2/revoke"
+def revoke_token(token):
+    response = requests.post(auth_revoke_url, params={"token": token})
+    return response.status_code == 200 # True if successful
+

Not all providers support token revocation, and it is not built into FastHTML clients at the moment.

+
+
+

Using State (Hugging Face)

+

Imagine a user (not logged in) comes to your AI image editing site, starts testing things out, and then realizes they need to sign in before they can click “Run (Pro)” on the edit they’re working on. They click “Sign in with Hugging Face”, log in, and are redirected back to your site. But now they’ve lost their in-progress edit and are left just looking at the homepage! This is an example of a case where you might want to keep track of some additional state. Another strong use case for being able to pass some uniqie state through the OAuth flow is to prevent something called a CSRF attack. To add a state string to the OAuth flow, you can use client.login_link_with_state(state) instead of client.login_link(), like so:

+
# in login page:
+link = A('Login with GitHub', href=client.login_link_with_state(state='current_prompt: add a unicorn'))
+
+# in auth_redirect:
+@app.get('/auth_redirect')
+def auth_redirect(code:str, session, state:str=None):
+    print(f"state: {state}") # Use as needed
+    ...
+

The state string is passed through the OAuth flow and back to your site.

+
+
+

A Work in Progress

+

This page (and OAuth support in FastHTML) is a work in progress. Questions, PRs, and feedback are welcome!

+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/explains/oauth.html.md b/explains/oauth.html.md new file mode 100644 index 00000000..98169736 --- /dev/null +++ b/explains/oauth.html.md @@ -0,0 +1,480 @@ +# OAuth + + + + +OAuth is an open standard for ‘access delegation’, commonly used as a +way for Internet users to grant websites or applications access to their +information on other websites but without giving them the passwords. It +is the mechanism that enables “Log in with Google” on many sites, saving +you from having to remember and manage yet another password. Like many +auth-related topics, there’s a lot of depth and complexity to the OAuth +standard, but once you understand the basic usage it can be a very +convenient alternative to managing your own user accounts. + +On this page you’ll see how to use OAuth with FastHTML to implement some +common pieces of functionality. + +## Creating an Client + +FastHTML has Client classes for managing settings and state for +different OAuth providers. Currently implemented are: GoogleAppClient, +GitHubAppClient, HuggingFaceClient and DiscordAppClient - see the +[source](https://github.com/AnswerDotAI/fasthtml/blob/main/nbs/api/08_oauth.ipynb) +if you need to add other providers. You’ll need a `client_id` and +`client_secret` from the provider (see the from-scratch example later in +this page for an example of registering with GitHub) to create the +client. We recommend storing these in environment variables, rather than +hardcoding them in your code. + +``` python +import os +from fasthtml.oauth import GoogleAppClient +client = GoogleAppClient(os.getenv("AUTH_CLIENT_ID"), + os.getenv("AUTH_CLIENT_SECRET")) +``` + +The client is used to obtain a login link and to manage communications +between your app and the OAuth provider +(`client.login_link(redirect_uri="/redirect")`). + +## Using the OAuth class + +Once you’ve set up a client, adding OAuth to a FastHTML app can be as +simple as: + +``` python +from fasthtml.oauth import OAuth +from fasthtml.common import FastHTML, RedirectResponse + +class Auth(OAuth): + def get_auth(self, info, ident, session, state): + email = info.email or '' + if info.email_verified and email.split('@')[-1]=='answer.ai': + return RedirectResponse('/', status_code=303) + +app = FastHTML() +oauth = Auth(app, client) + +@app.get('/') +def home(auth): return P('Logged in!'), A('Log out', href='/logout') + +@app.get('/login') +def login(req): return Div(P("Not logged in"), A('Log in', href=oauth.login_link(req))) +``` + +There’s a fair bit going on here, so let’s unpack what’s happening in +that code: + +- OAuth (and by extension our custom Auth class) has a number of default + arguments, including some key URLs: + `redir_path='/redirect', error_path='/error', logout_path='/logout', login_path='/login'`. + It will create and handle the redirect and logout paths, and it’s up + to you to handle `/login` (where unsuccessful login attempts will be + redirected) and `/error` (for oauth errors). +- When we run `oauth = Auth(app, client)` it adds the redirect and + logout paths to the app and also adds some beforeware. This beforeware + runs on any requests (apart from any specified with the `skip` + parameter). + +The added beforeware specifies some app behaviour: + +- If someone who isn’t logged in attempts to visit our homepage (`/`) + here, they will be redirected to `/login`. +- If they are logged in, it calls a `check_invalid` method. This + defaults to False, which let’s the user continue to the page they + requested. The behaviour can be modified by defining your own + `check_invalid` method in the Auth class - for example, you could have + this forcibly log out users who have recently been banned. + +So how does someone log in? If they visit (or are redirected to) the +login page at `/login`, we show them a login link. This sends them to +the OAuth provider, where they’ll go through the steps of selecting +their account, giving permissions etc. Once done they will be redirected +back to `/redirect`. Behind the scenes a code that comes as part of +their request gets turned into user info, which is then passed to the +key function `get_auth(self, info, ident, session, state)`. Here is +where you’d handle looking up or adding a user in a database, checking +for some condition (for example, this code checks if the email is an +answer.ai email address) or choosing the destination based on state. The +arguments are: + +- `self`: the Auth object, which you can use to access the client + (`self.cli`) +- `info`: the information provided by the OAuth provider, typically + including a unique user id, email address, username and other + metadata. +- `ident`: a unique identifier for this user. What this looks like + varies between providers. This is useful for managing a database of + users, for example. +- `session`: the current session, that you can store information in + securely +- `state`: you can optionally pass in some state when creating the login + link. This persists and is returned after the user goes through the + Oath steps, which is useful for returning them to the same page they + left. It can also be used as added security against CSRF attacks. + +In our example, we check the email in `info` (we use a GoogleAppClient, +not all providers will include an email). If we aren’t happy, and +get_auth returns False or nothing (as in the case here for non-answerai +people) then the user is redirected back to the login page. But if +everything looks good we return a redirect to the homepage, and an +`auth` key is added to the session and the scope containing the users +identity `ident`. So, for example, in the homepage route we could use +`auth` to look up this particular user’s profile info and customize the +page accordingly. This auth will persist in their session until they +clear the browser cache, so by default they’ll stay logged in. To log +them out, remove it ( `session.pop('auth', None)`) or send them to +`/logout` which will do that for you. + +## Explaining OAuth with a from-scratch implementation + +Hopefully the example above is enough to get you started. You can also +check out the (fairly minimal) [source +code](https://github.com/AnswerDotAI/fasthtml/blob/main/nbs/api/08_oauth.ipynb) +where this is implemented, and the [examples +here](https://github.com/AnswerDotAI/fasthtml-example/blob/main/oauth_example). + +If you’re wanting to learn more about how this works, and to see where +you might add additional functionality, the rest of this page will walk +through some examples **without** the OAuth convenience class, to +illustrate the concepts. This was writted before said OAuth class was +available, and is kep here for educational purposes - we recommend you +stick with the new approach shown above in most cases. + +## A Minimal Login Flow (GitHub) + +Let’s begin by building a minimal ‘Sign in with GitHub’ flow. This will +demonstrate the basic steps of OAuth. + +OAuth requires a “provider” (in this case, GitHub) to authenticate the +user. So the first step when setting up our app is to register with +GitHub to set things up. + +Go to https://github.com/settings/developers and click “New OAuth App”. +Fill in the form with the following values, then click ‘Register +application’. + +- Application name: Your app name +- Homepage URL: http://localhost:8000 (or whatever URL you’re using - + you can change this later) +- Authorization callback URL: http://localhost:8000/auth_redirect (you + can modify this later too) + +
+ +Setting up an OAuth app in GitHub + +
+ +After you register, you’ll see a screen where you can view the client ID +and generate a client secret. Store these values in a safe place. You’ll +use them to create a +[`GitHubAppClient`](https://AnswerDotAI.github.io/fasthtml/api/oauth.html#githubappclient) +object in FastHTML. + +This `client` object is responsible for handling the parts of the OAuth +flow which depend on direct communication between your app and GitHub, +as opposed to interactions which go through the user’s browser via +redirects. + +Here is how to setup the client object: + +``` python +client = GitHubAppClient( + client_id="your_client_id", + client_secret="your_client_secret" +) +``` + +You should also save the path component of the authorization callback +URL which you provided on registration. + +This route is where GitHub will redirect the user’s browser in order to +send an authorization code to your app. You should save only the URL’s +path component rather than the entire URL because you want your code to +work automatically in deployment, when the host and port part of the URL +change from `localhost:8000` to your real DNS name. + +Save the special authorization callback path under an obvious name: + +``` python +auth_callback_path = "/auth_redirect" +``` + +
+ +> **Note** +> +> It’s recommended to store the client ID, and secret, in environment +> variables, rather than hardcoding them in your code. + +
+ +When the user visit a normal page of your app, if they are not already +logged in, then you’ll want to redirect them to your app’s login page, +which will live at the `/login` path. We accomplish that by using this +piece of “beforeware”, which defines logic which runs before other work +for all routes except ones we specify to be skipped: + +``` python +def before(req, session): + auth = req.scope['auth'] = session.get('user_id', None) + if not auth: return RedirectResponse('/login', status_code=303) + counts.xtra(name=auth) +bware = Beforeware(before, skip=['/login', auth_callback_path]) +``` + +We configure the beforeware to skip `/login` because that’s where the +user goes to login, and we also skip the special authorization callback +path because that is used by OAuth itself to receive information from +GitHub. + +It’s only at your login page that we start the OAuth flow. To start the +OAuth flow, you need to give the user a link to GitHub’s login for your +app. You’ll need the `client` object to generate that link, and the +client object will in turn need the full authorization callback URL, +which we need to build from the authorization callback path, so it is a +multi-step process to produce this GitHub login link. + +Here is an implementation of your own `/login` route handler. It +generates the GitHub login link and presents it to the user: + +``` python +@app.get('/login') +def login(request) + redir = redir_url(request,auth_callback_path) + login_link = client.login_link(redir) + return P(A('Login with GitHub', href=login_link)) +``` + +Once the user follows that link, GitHub will ask them to grant +permission to your app to access their GitHub account. If they agree, +GitHub will redirect them back to your app’s authorization callback URL, +carrying an authorization code which your app can use to generate an +access token. To receive this code, you need to set up a route in +FastHTML that listens for requests at the authorization callback path. +For example: + +``` python +@app.get(auth_callback_path) +def auth_redirect(code:str): + return P(f"code: {code}") +``` + +This authorization code is temporary, and is used by your app to +directly ask the provider for user information like an access token. + +To recap, you can think of the exchange so far as: + +- User to us: “I want to log in with you, app.” +- Us to User: “Okay but first, here’s a special link to log in with + GitHub” +- User to GitHub: “I want to log in with you, GitHub, to use this app.” +- GitHub to User: “OK, redirecting you back to the app’s URL (with an + auth code)” +- User to Us: “Hi again, app. Here’s the GitHub auth code you need to + ask GitHub for info about me” (delivered via + `/auth_redirect?code=...`) + +The final steps we need to implement are as follows: + +- Us to GitHUb: “A user just gave me this auth code. May I have the user + info (e.g., an access token)?” +- GitHub to us: “Since you have an auth code, here’s the user info” + +It’s critical for us to derive the user info from the auth code +immediately in the authorization callback, because the auth code may be +used only once. So we use it that once in order to get information like +an access token, which will remain valid for longer. + +To go from the auth code to user info, you use +`info = client.retr_info(code,redirect_uri)`. From the user info, you +can extract the `user_id`, which is a unique identifier for the user: + +``` python +@app.get(auth_callback_path) +def auth_redirect(code:str, request): + redir = redir_url(request, auth_callback_path) + user_info = client.retr_info(code, redir) + user_id = info[client.id_key] + return P(f"User id: {user_id}") +``` + +But we want the user ID not to print it but to remember the user. + +So let us store it in the `session` object, to remember who is logged +in: + +``` python +@app.get(auth_callback_path) +def auth_redirect(code:str, request, session): + redir = redir_url(request, auth_callback_path) + user_info = client.retr_info(code, redir) + user_id = user_info[client.id_key] # get their ID + session['user_id'] = user_id # save ID in the session + return RedirectResponse('/', status_code=303) +``` + +The session object is derived from values visible to the user’s browser, +but it is cryptographically signed so the user can’t read it themselves. +This makes it safe to store even information we don’t want to expose to +the user. + +For larger quantities of data, we’d want to save that information in a +database and use the session to hold keys to lookup information from +that database. + +Here’s a minimal app that puts all these pieces together. It uses the +user info to get the user_id. It stores that in the session object. It +then uses the user_id as a key into a database, which tracks how +frequently every user has hit an increment button. + +``` python +import os +from fasthtml.common import * +from fasthtml.oauth import GitHubAppClient, redir_url + +db = database('data/counts.db') +counts = db.t.counts +if counts not in db.t: counts.create(dict(name=str, count=int), pk='name') +Count = counts.dataclass() + +# Auth client setup for GitHub +client = GitHubAppClient(os.getenv("AUTH_CLIENT_ID"), + os.getenv("AUTH_CLIENT_SECRET")) +auth_callback_path = "/auth_redirect" + +def before(req, session): + # if not logged in, we send them to our login page + # logged in means: + # - 'user_id' in the session object, + # - 'auth' in the request object + auth = req.scope['auth'] = session.get('user_id', None) + if not auth: return RedirectResponse('/login', status_code=303) + counts.xtra(name=auth) +bware = Beforeware(before, skip=['/login', auth_callback_path]) + +app = FastHTML(before=bware) + +# User asks us to Login +@app.get('/login') +def login(request): + redir = redir_url(request,auth_callback_path) + login_link = client.login_link(redir) + # we tell user to login at github + return P(A('Login with GitHub', href=login_link)) + +# User comes back to us with an auth code from Github +@app.get(auth_callback_path) +def auth_redirect(code:str, request, session): + redir = redir_url(request, auth_callback_path) + user_info = client.retr_info(code, redir) + user_id = user_info[client.id_key] # get their ID + session['user_id'] = user_id # save ID in the session + # create a db entry for the user + if user_id not in counts: counts.insert(name=user_id, count=0) + return RedirectResponse('/', status_code=303) + +@app.get('/') +def home(auth): + return Div( + P("Count demo"), + P(f"Count: ", Span(counts[auth].count, id='count')), + Button('Increment', hx_get='/increment', hx_target='#count'), + P(A('Logout', href='/logout')) + ) + +@app.get('/increment') +def increment(auth): + c = counts[auth] + c.count += 1 + return counts.upsert(c).count + +@app.get('/logout') +def logout(session): + session.pop('user_id', None) + return RedirectResponse('/login', status_code=303) + +serve() +``` + +Some things to note: + +- The `before` function is used to check if the user is authenticated. + If not, they are redirected to the login page. +- To log the user out, we remove the user ID from the session. +- Calling `counts.xtra(name=auth)` ensures that only the row + corresponding to the current user is accessible when responding to a + request. This is often nicer than trying to remember to filter the + data in every route, and lowers the risk of accidentally leaking data. +- In the `auth_redirect` route, we store the user ID in the session and + create a new row in the `user_counts` table if it doesn’t already + exist. + +You can find more heavily-commented version of this code in the [oauth +directory in +fasthtml-example](https://github.com/AnswerDotAI/fasthtml-example/tree/main/oauth_example), +along with an even more minimal example. More examples may be added in +the future. + +### Revoking Tokens (Google) + +When the user in the example above logs out, we remove their user ID +from the session. However, the user is still logged in to GitHub. If +they click ‘Login with GitHub’ again, they’ll be redirected back to our +site without having to log in again. This is because GitHub remembers +that they’ve already granted our app permission to access their account. +Most of the time this is convenient, but for testing or security +purposes you may want a way to revoke this permission. + +As a user, you can usually revoke access to an app from the provider’s +website (for example, ). But +as a developer, you can also revoke access programmatically - at least +with some providers. This requires keeping track of the access token +(stored in `client.token["access_token"]` after you call `retr_info`), +and sending a request to the provider’s revoke URL: + +``` python +auth_revoke_url = "https://accounts.google.com/o/oauth2/revoke" +def revoke_token(token): + response = requests.post(auth_revoke_url, params={"token": token}) + return response.status_code == 200 # True if successful +``` + +Not all providers support token revocation, and it is not built into +FastHTML clients at the moment. + +### Using State (Hugging Face) + +Imagine a user (not logged in) comes to your AI image editing site, +starts testing things out, and then realizes they need to sign in before +they can click “Run (Pro)” on the edit they’re working on. They click +“Sign in with Hugging Face”, log in, and are redirected back to your +site. But now they’ve lost their in-progress edit and are left just +looking at the homepage! This is an example of a case where you might +want to keep track of some additional state. Another strong use case for +being able to pass some uniqie state through the OAuth flow is to +prevent something called a [CSRF +attack](https://en.wikipedia.org/wiki/Cross-site_request_forgery). To +add a state string to the OAuth flow, you can use +`client.login_link_with_state(state)` instead of `client.login_link()`, +like so: + +``` python +# in login page: +link = A('Login with GitHub', href=client.login_link_with_state(state='current_prompt: add a unicorn')) + +# in auth_redirect: +@app.get('/auth_redirect') +def auth_redirect(code:str, session, state:str=None): + print(f"state: {state}") # Use as needed + ... +``` + +The state string is passed through the OAuth flow and back to your site. + +### A Work in Progress + +This page (and OAuth support in FastHTML) is a work in progress. +Questions, PRs, and feedback are welcome! diff --git a/explains/routes.html b/explains/routes.html new file mode 100644 index 00000000..963d6677 --- /dev/null +++ b/explains/routes.html @@ -0,0 +1,992 @@ + + + + + + + + + +Routes – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Routes

+
+ + + +
+ + + + +
+ + + +
+ + + +

Behaviour in FastHTML apps is defined by routes. The syntax is largely the same as the wonderful FastAPI (which is what you should be using instead of this if you’re creating a JSON service. FastHTML is mainly for making HTML web apps, not APIs).

+
+
+
+ +
+
+Unfinished +
+
+
+

We haven’t yet written complete documentation of all of FastHTML’s routing features – until we add that, the best place to see all the available functionality is to look over the tests

+
+
+

Note that you need to include the types of your parameters, so that FastHTML knows what to pass to your function. Here, we’re just expecting a string:

+
+
from fasthtml.common import *
+
+
+
app = FastHTML()
+
+@app.get('/user/{nm}')
+def get_nm(nm:str): return f"Good day to you, {nm}!"
+
+

Normally you’d save this into a file such as main.py, and then run it in uvicorn using:

+
uvicorn main:app
+

However, for testing, we can use Starlette’s TestClient to try it out:

+
+
from starlette.testclient import TestClient
+
+
+
client = TestClient(app)
+r = client.get('/user/Jeremy')
+r
+
+
<Response [200 OK]>
+
+
+

TestClient uses httpx behind the scenes, so it returns a httpx.Response, which has a text attribute with our response body:

+
+
r.text
+
+
'Good day to you, Jeremy!'
+
+
+

In the previous example, the function name (get_nm) didn’t actually matter – we could have just called it _, for instance, since we never actually call it directly. It’s just called through HTTP. In fact, we often do call our functions _ when using this style of route, since that’s one less thing we have to worry about, naming.

+

An alternative approach to creating a route is to use app.route instead, in which case, you make the function name the HTTP method you want. Since this is such a common pattern, you might like to give a shorter name to app.route – we normally use rt:

+
+
rt = app.route
+
+@rt('/')
+def post(): return "Going postal!"
+
+client.post('/').text
+
+
'Going postal!'
+
+
+
+

Route-specific functionality

+

FastHTML supports custom decorators for adding specific functionality to routes. This allows you to implement authentication, authorization, middleware, or other custom behaviors for individual routes.

+

Here’s an example of a basic authentication decorator:

+
+
from functools import wraps
+
+def basic_auth(f):
+    @wraps(f)
+    async def wrapper(req, *args, **kwargs):
+        token = req.headers.get("Authorization")
+        if token == 'abc123':
+            return await f(req, *args, **kwargs)
+        return Response('Not Authorized', status_code=401)
+    return wrapper
+
+@app.get("/protected")
+@basic_auth
+async def protected(req):
+    return "Protected Content"
+
+client.get('/protected', headers={'Authorization': 'abc123'}).text
+
+
'Protected Content'
+
+
+

The decorator intercepts the request before the route function executes. If the decorator allows the request to proceed, it calls the original route function, passing along the request and any other arguments.

+

One of the key advantages of this approach is the ability to apply different behaviors to different routes. You can also stack multiple decorators on a single route for combined functionality.

+
+
def app_beforeware():
+    print('App level beforeware')
+
+app = FastHTML(before=Beforeware(app_beforeware))
+client = TestClient(app)
+
+def route_beforeware(f):
+    @wraps(f)
+    async def decorator(*args, **kwargs):
+        print('Route level beforeware')
+        return await f(*args, **kwargs)
+    return decorator
+    
+def second_route_beforeware(f):
+    @wraps(f)
+    async def decorator(*args, **kwargs):
+        print('Second route level beforeware')
+        return await f(*args, **kwargs)
+    return decorator
+
+@app.get("/users")
+@route_beforeware
+@second_route_beforeware
+async def users():
+    return "Users Page"
+
+client.get('/users').text
+
+
App level beforeware
+Route level beforeware
+Second route level beforeware
+
+
+
'Users Page'
+
+
+

This flexiblity allows for granular control over route behaviour, enabling you to tailor each endpoint’s functionality as needed. While app-level beforeware remains useful for global operations, decorators provide a powerful tool for route-specific customization.

+
+
+

Combining Routes

+

Sometimes a FastHTML project can grow so weildy that putting all the routes into main.py becomes unweildy. Or, we install a FastHTML- or Starlette-based package that requires us to add routes.

+

First let’s create a books.py module, that represents all the user-related views:

+
+
# books.py
+books_app, rt = fast_app()
+
+books = ['A Guide to FastHTML', 'FastHTML Cookbook', 'FastHTML in 24 Hours']
+
+@rt("/", name="list")
+def get():
+    return Titled("Books", *[P(book) for book in books])
+
+

Let’s mount it in our main module:

+
from books import books_app
+
+1app, rt = fast_app(routes=[Mount("/books", books_app, name="books")])
+
+@rt("/")
+def get():
+    return Titled("Dashboard",
+2        P(A(href="/books")("Books")),
+        Hr(),
+3        P(A(link=uri("books:list"))("Books")),
+    )
+
+serve()
+
+
1
+
+We use starlette.Mount to add the route to our routes list. We provide the name of books to make discovery and management of the links easier. More on that in items 2 and 3 of this annotations list +
+
2
+
+This example link to the books list view is hand-crafted. Obvious in purpose, it makes changing link patterns in the future harder +
+
3
+
+This example link uses the named URL route for the books. The advantage of this approach is it makes management of large numbers of link items easier. +
+
+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/explains/routes.html.md b/explains/routes.html.md new file mode 100644 index 00000000..a0cf3d28 --- /dev/null +++ b/explains/routes.html.md @@ -0,0 +1,217 @@ +# Routes + + + + +Behaviour in FastHTML apps is defined by routes. The syntax is largely +the same as the wonderful [FastAPI](https://fastapi.tiangolo.com/) +(which is what you should be using instead of this if you’re creating a +JSON service. FastHTML is mainly for making HTML web apps, not APIs). + +
+ +> **Unfinished** +> +> We haven’t yet written complete documentation of all of FastHTML’s +> routing features – until we add that, the best place to see all the +> available functionality is to look over [the +> tests](../api/core.html#tests) + +
+ +Note that you need to include the types of your parameters, so that +[`FastHTML`](https://AnswerDotAI.github.io/fasthtml/api/core.html#fasthtml) +knows what to pass to your function. Here, we’re just expecting a +string: + +``` python +from fasthtml.common import * +``` + +``` python +app = FastHTML() + +@app.get('/user/{nm}') +def get_nm(nm:str): return f"Good day to you, {nm}!" +``` + +Normally you’d save this into a file such as main.py, and then run it in +`uvicorn` using: + + uvicorn main:app + +However, for testing, we can use Starlette’s `TestClient` to try it out: + +``` python +from starlette.testclient import TestClient +``` + +``` python +client = TestClient(app) +r = client.get('/user/Jeremy') +r +``` + + + +TestClient uses `httpx` behind the scenes, so it returns a +`httpx.Response`, which has a `text` attribute with our response body: + +``` python +r.text +``` + + 'Good day to you, Jeremy!' + +In the previous example, the function name (`get_nm`) didn’t actually +matter – we could have just called it `_`, for instance, since we never +actually call it directly. It’s just called through HTTP. In fact, we +often do call our functions `_` when using this style of route, since +that’s one less thing we have to worry about, naming. + +An alternative approach to creating a route is to use `app.route` +instead, in which case, you make the function name the HTTP method you +want. Since this is such a common pattern, you might like to give a +shorter name to `app.route` – we normally use `rt`: + +``` python +rt = app.route + +@rt('/') +def post(): return "Going postal!" + +client.post('/').text +``` + + 'Going postal!' + +### Route-specific functionality + +FastHTML supports custom decorators for adding specific functionality to +routes. This allows you to implement authentication, authorization, +middleware, or other custom behaviors for individual routes. + +Here’s an example of a basic authentication decorator: + +``` python +from functools import wraps + +def basic_auth(f): + @wraps(f) + async def wrapper(req, *args, **kwargs): + token = req.headers.get("Authorization") + if token == 'abc123': + return await f(req, *args, **kwargs) + return Response('Not Authorized', status_code=401) + return wrapper + +@app.get("/protected") +@basic_auth +async def protected(req): + return "Protected Content" + +client.get('/protected', headers={'Authorization': 'abc123'}).text +``` + + 'Protected Content' + +The decorator intercepts the request before the route function executes. +If the decorator allows the request to proceed, it calls the original +route function, passing along the request and any other arguments. + +One of the key advantages of this approach is the ability to apply +different behaviors to different routes. You can also stack multiple +decorators on a single route for combined functionality. + +``` python +def app_beforeware(): + print('App level beforeware') + +app = FastHTML(before=Beforeware(app_beforeware)) +client = TestClient(app) + +def route_beforeware(f): + @wraps(f) + async def decorator(*args, **kwargs): + print('Route level beforeware') + return await f(*args, **kwargs) + return decorator + +def second_route_beforeware(f): + @wraps(f) + async def decorator(*args, **kwargs): + print('Second route level beforeware') + return await f(*args, **kwargs) + return decorator + +@app.get("/users") +@route_beforeware +@second_route_beforeware +async def users(): + return "Users Page" + +client.get('/users').text +``` + + App level beforeware + Route level beforeware + Second route level beforeware + + 'Users Page' + +This flexiblity allows for granular control over route behaviour, +enabling you to tailor each endpoint’s functionality as needed. While +app-level beforeware remains useful for global operations, decorators +provide a powerful tool for route-specific customization. + +## Combining Routes + +Sometimes a FastHTML project can grow so weildy that putting all the +routes into `main.py` becomes unweildy. Or, we install a FastHTML- or +Starlette-based package that requires us to add routes. + +First let’s create a `books.py` module, that represents all the +user-related views: + +``` python +# books.py +books_app, rt = fast_app() + +books = ['A Guide to FastHTML', 'FastHTML Cookbook', 'FastHTML in 24 Hours'] + +@rt("/", name="list") +def get(): + return Titled("Books", *[P(book) for book in books]) +``` + +Let’s mount it in our main module: + +``` python +from books import books_app + +app, rt = fast_app(routes=[Mount("/books", books_app, name="books")]) + +@rt("/") +def get(): + return Titled("Dashboard", + P(A(href="/books")("Books")), + Hr(), + P(A(link=uri("books:list"))("Books")), + ) + +serve() +``` + +Line 3 +We use `starlette.Mount` to add the route to our routes list. We provide +the name of `books` to make discovery and management of the links +easier. More on that in items 2 and 3 of this annotations list + +Line 8 +This example link to the books list view is hand-crafted. Obvious in +purpose, it makes changing link patterns in the future harder + +Line 10 +This example link uses the named URL route for the books. The advantage +of this approach is it makes management of large numbers of link items +easier. diff --git a/explains/websockets.html b/explains/websockets.html new file mode 100644 index 00000000..119800ac --- /dev/null +++ b/explains/websockets.html @@ -0,0 +1,948 @@ + + + + + + + + + +WebSockets – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

WebSockets

+
+ + + +
+ + + + +
+ + + +
+ + + +

Websockets are a protocol for two-way, persistent communication between a client and server. This is different from HTTP, which uses a request/response model where the client sends a request and the server responds. With websockets, either party can send messages at any time, and the other party can respond.

+

This allows for different applications to be built, including things like chat apps, live-updating dashboards, and real-time collaborative tools, which would require constant polling of the server for updates with HTTP.

+

In FastHTML, you can create a websocket route using the @app.ws decorator. This decorator takes a route path, and optional conn and disconn parameters representing the on_connect and on_disconnect callbacks in websockets, respectively. The function decorated by @app.ws is the main function that is called when a message is received.

+

Here’s an example of a basic websocket route:

+
@app.ws('/ws', conn=on_conn, disconn=on_disconn)
+async def on_message(msg:str, send):
+    await send(Div('Hello ' + msg, id='notifications'))
+    await send(Div('Goodbye ' + msg, id='notifications'))
+

The on_message function is the main function that is called when a message is received and can be named however you like. Similar to standard routes, the arguments to on_message are automatically parsed from the websocket payload for you, so you don’t need to manually parse the message content. However, certain argument names are reserved for special purposes. Here are the most important ones:

+
    +
  • send is a function that can be used to send text data to the client.
  • +
  • data is a dictionary containing the data sent by the client.
  • +
  • ws is a reference to the websocket object.
  • +
+

For example, we can send a message to the client that just connected like this:

+
async def on_conn(send):
+    await send(Div('Hello, world!'))
+

Or if we receive a message from the client, we can send a message back to them:

+
@app.ws('/ws', conn=on_conn, disconn=on_disconn)
+async def on_message(msg:str, send):
+    await send(Div('You said: ' + msg, id='notifications'))
+    # or...
+    return Div('You said: ' + msg, id='notifications')
+

On the client side, we can use HTMX’s websocket extension to open a websocket connection and send/receive messages. For example:

+
from fasthtml.common import *
+
+app = FastHTML(exts='ws')
+
+@app.get('/')
+def home():
+    cts = Div(
+        Div(id='notifications'),
+        Form(Input(id='msg'), id='form', ws_send=True),
+        hx_ext='ws', ws_connect='/ws')
+    return Titled('Websocket Test', cts)
+

This will create a websocket connection to the server on route /ws, and send any form submissions to the server via the websocket. The server will then respond by sending a message back to the client. The client will then update the message div with the message from the server using Out of Band Swaps, which means that the content is swapped with the same id without reloading the page.

+
+
+
+ +
+
+Note +
+
+
+

Make sure you set exts='ws' when creating your FastHTML object if you want to use websockets so the extension is loaded.

+
+
+

Putting it all together, the code for the client and server should look like this:

+
from fasthtml.common import *
+
+app = FastHTML(exts='ws')
+rt = app.route
+
+@rt('/')
+def get():
+    cts = Div(
+        Div(id='notifications'),
+        Form(Input(id='msg'), id='form', ws_send=True),
+        hx_ext='ws', ws_connect='/ws')
+    return Titled('Websocket Test', cts)
+
+@app.ws('/ws')
+async def ws(msg:str, send):
+    await send(Div('Hello ' + msg, id='notifications'))
+
+serve()
+

This is a fairly simple example and could be done just as easily with standard HTTP requests, but it illustrates the basic idea of how websockets work. Let’s look at a more complex example next.

+
+

Session data in Websockets

+

Session data is shared between standard HTTP routes and Websockets. This means you can access, for example, logged in user ID inside websocket handler:

+
from fasthtml.common import *
+
+app = FastHTML(exts='ws')
+rt = app.route
+
+@rt('/login')
+def get(session):
+    session["person"] = "Bob"
+    return "ok"
+
+@app.ws('/ws')
+async def ws(msg:str, send, session):
+    await send(Div(f'Hello {session.get("person")}' + msg, id='notifications'))
+
+serve()
+
+
+

Real-Time Chat App

+

Let’s put our new websocket knowledge to use by building a simple chat app. We will create a chat app where multiple users can send and receive messages in real time.

+

Let’s start by defining the app and the home page:

+
from fasthtml.common import *
+
+app = FastHTML(exts='ws')
+rt = app.route
+
+msgs = []
+@rt('/')
+def home(): return Div(
+    Div(Ul(*[Li(m) for m in msgs], id='msg-list')),
+    Form(Input(id='msg'), id='form', ws_send=True),
+    hx_ext='ws', ws_connect='/ws')
+

Now, let’s handle the websocket connection. We’ll add a new route for this along with an on_conn and on_disconn function to keep track of the users currently connected to the websocket. Finally, we will handle the logic for sending messages to all connected users.

+
users = {}
+def on_conn(ws, send): users[str(id(ws))] = send
+def on_disconn(ws): users.pop(str(id(ws)), None)
+
+@app.ws('/ws', conn=on_conn, disconn=on_disconn)
+async def ws(msg:str):
+    msgs.append(msg)
+    # Use associated `send` function to send message to each user
+    for u in users.values(): await u(Ul(*[Li(m) for m in msgs], id='msg-list'))
+
+serve()
+

We can now run this app with python chat_ws.py and open multiple browser tabs to http://localhost:5001. You should be able to send messages in one tab and see them appear in the other tabs.

+
+

A Work in Progress

+

This page (and Websocket support in FastHTML) is a work in progress. Questions, PRs, and feedback are welcome!

+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/explains/websockets.html.md b/explains/websockets.html.md new file mode 100644 index 00000000..536682de --- /dev/null +++ b/explains/websockets.html.md @@ -0,0 +1,197 @@ +# WebSockets + + + + +Websockets are a protocol for two-way, persistent communication between +a client and server. This is different from HTTP, which uses a +request/response model where the client sends a request and the server +responds. With websockets, either party can send messages at any time, +and the other party can respond. + +This allows for different applications to be built, including things +like chat apps, live-updating dashboards, and real-time collaborative +tools, which would require constant polling of the server for updates +with HTTP. + +In FastHTML, you can create a websocket route using the `@app.ws` +decorator. This decorator takes a route path, and optional `conn` and +`disconn` parameters representing the `on_connect` and `on_disconnect` +callbacks in websockets, respectively. The function decorated by +`@app.ws` is the main function that is called when a message is +received. + +Here’s an example of a basic websocket route: + +``` python +@app.ws('/ws', conn=on_conn, disconn=on_disconn) +async def on_message(msg:str, send): + await send(Div('Hello ' + msg, id='notifications')) + await send(Div('Goodbye ' + msg, id='notifications')) +``` + +The `on_message` function is the main function that is called when a +message is received and can be named however you like. Similar to +standard routes, the arguments to `on_message` are automatically parsed +from the websocket payload for you, so you don’t need to manually parse +the message content. However, certain argument names are reserved for +special purposes. Here are the most important ones: + +- `send` is a function that can be used to send text data to the client. +- `data` is a dictionary containing the data sent by the client. +- `ws` is a reference to the websocket object. + +For example, we can send a message to the client that just connected +like this: + +``` python +async def on_conn(send): + await send(Div('Hello, world!')) +``` + +Or if we receive a message from the client, we can send a message back +to them: + +``` python +@app.ws('/ws', conn=on_conn, disconn=on_disconn) +async def on_message(msg:str, send): + await send(Div('You said: ' + msg, id='notifications')) + # or... + return Div('You said: ' + msg, id='notifications') +``` + +On the client side, we can use HTMX’s websocket extension to open a +websocket connection and send/receive messages. For example: + +``` python +from fasthtml.common import * + +app = FastHTML(exts='ws') + +@app.get('/') +def home(): + cts = Div( + Div(id='notifications'), + Form(Input(id='msg'), id='form', ws_send=True), + hx_ext='ws', ws_connect='/ws') + return Titled('Websocket Test', cts) +``` + +This will create a websocket connection to the server on route `/ws`, +and send any form submissions to the server via the websocket. The +server will then respond by sending a message back to the client. The +client will then update the message div with the message from the server +using Out of Band Swaps, which means that the content is swapped with +the same id without reloading the page. + +
+ +> **Note** +> +> Make sure you set `exts='ws'` when creating your +> [`FastHTML`](https://AnswerDotAI.github.io/fasthtml/api/core.html#fasthtml) +> object if you want to use websockets so the extension is loaded. + +
+ +Putting it all together, the code for the client and server should look +like this: + +``` python +from fasthtml.common import * + +app = FastHTML(exts='ws') +rt = app.route + +@rt('/') +def get(): + cts = Div( + Div(id='notifications'), + Form(Input(id='msg'), id='form', ws_send=True), + hx_ext='ws', ws_connect='/ws') + return Titled('Websocket Test', cts) + +@app.ws('/ws') +async def ws(msg:str, send): + await send(Div('Hello ' + msg, id='notifications')) + +serve() +``` + +This is a fairly simple example and could be done just as easily with +standard HTTP requests, but it illustrates the basic idea of how +websockets work. Let’s look at a more complex example next. + +## Session data in Websockets + +Session data is shared between standard HTTP routes and Websockets. This +means you can access, for example, logged in user ID inside websocket +handler: + +``` python +from fasthtml.common import * + +app = FastHTML(exts='ws') +rt = app.route + +@rt('/login') +def get(session): + session["person"] = "Bob" + return "ok" + +@app.ws('/ws') +async def ws(msg:str, send, session): + await send(Div(f'Hello {session.get("person")}' + msg, id='notifications')) + +serve() +``` + +## Real-Time Chat App + +Let’s put our new websocket knowledge to use by building a simple chat +app. We will create a chat app where multiple users can send and receive +messages in real time. + +Let’s start by defining the app and the home page: + +``` python +from fasthtml.common import * + +app = FastHTML(exts='ws') +rt = app.route + +msgs = [] +@rt('/') +def home(): return Div( + Div(Ul(*[Li(m) for m in msgs], id='msg-list')), + Form(Input(id='msg'), id='form', ws_send=True), + hx_ext='ws', ws_connect='/ws') +``` + +Now, let’s handle the websocket connection. We’ll add a new route for +this along with an `on_conn` and `on_disconn` function to keep track of +the users currently connected to the websocket. Finally, we will handle +the logic for sending messages to all connected users. + +``` python +users = {} +def on_conn(ws, send): users[str(id(ws))] = send +def on_disconn(ws): users.pop(str(id(ws)), None) + +@app.ws('/ws', conn=on_conn, disconn=on_disconn) +async def ws(msg:str): + msgs.append(msg) + # Use associated `send` function to send message to each user + for u in users.values(): await u(Ul(*[Li(m) for m in msgs], id='msg-list')) + +serve() +``` + +We can now run this app with `python chat_ws.py` and open multiple +browser tabs to `http://localhost:5001`. You should be able to send +messages in one tab and see them appear in the other tabs. + +### A Work in Progress + +This page (and Websocket support in FastHTML) is a work in progress. +Questions, PRs, and feedback are welcome! diff --git a/favicon.ico b/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..82ef3bab5996bc30a77e3fe0901081eed5eb8c61 GIT binary patch literal 15086 zcmeHO&r2&u9G{r#A_$4U5JX9oR;m}D2wvmOLiJd@X%8Y574_hyylQx@?@+-)p|^q{ z1?3_5CuqTYFM_l@YWnC=`$W`6U1yis$!2!;XOfbEPrfrd^ZkB4-Yh>+JX;8Pd_F62u99@opq&&KJM`Y$h? zIKu4rFJF&Pu^a-@k6|j3LxB1#l0(4yBhMiO{gLJnlK#kY2vL6|IfSe~^c)oR-@&+Z z3I2e-yd+j=_f`of-}O7+@L2yQjQ3$hEkWoQw(Pg^PPhk zNnQy{KeGDDK`z_u4O73b9887sO4#~6=OCAD_C}%KQx2v=c_k|Su5*yfHhZJgkEGwL zzQ>sg<&~)QKh?PYw>7T+yb}65J3Hy_?yhTGTwI7}Z$Hg_7*LMB2KL1TkO;vOLx3;#@jg1Z0I6FHN7rY|(9}mwj zqHd#I*Kf?kvsJo&>+^3;TK)Vnr$Vn(^ZE0ORjO@ENk81n>Cn)SYmAJH(3_hZp4iR# z69-49jWZ?vcJHq$$KZ1xy}G*cq#xoFaCFKcRQ>RI)t7#S9I9TwLk=~eUy_4G|MK!O zg*7j>w6r`00oV)neg2?!qu5*z7XA3VcY1mnC<=vw=j+$d&08S{i+-Lcq`LC`lh-WP zxg2UzKjy=7_&KC`E*I8z{_{kG@r2|F-tN)Hm_?b`CY#J=m`yR8U={-C;~;=;A%Jcn zz_ZTFk{HQZf4IcOd)!>moFbzKV1JU4<-?s({ z|9j~B)dp{oR4AU$Nr+SK{QpbfaB6ze5FkJ5duex#%Z{n=kB^TRySlpW5wM4z$M5GQ z#nn8h@Q;p;et;bCwya}jW=2AkrSE^e1B= zXUp=-u6v^|nM~$yZ*ProW%*$|J{;R=@WtQO)^?}LUq*v3e!NGx;kVIX^5btNR#sMA zVSRmFY5kFxFWg#~{Kg{su+sJC`LWJ4{OE&EbhYs3V0R@K0{~MFwq$Z714sc901Xfa z!~g`b3CI9a0L=|xn?lP2V0%J?y$#%&(P)-vxKDxG5@~?BPXYfmvn!Cu7h*E3{hi%- z@tDLP!#Kup@OO+M_~VyAzcJvK>c)QpHbmV#_3#zC-_^;=t8qlbrw zCXSAdXt7wt;U+)M1GHb`K`0h~K6k9cZ*8t09v*O<#h;6z`%g|zUJefrzY{SwHg?Ir kXDIP!F}}s%3Ptxnfz>(L1Y{UeU=sih5HF7$^j!n~0~&?pHUIzs literal 0 HcmV?d00001 diff --git a/index.html b/index.html new file mode 100644 index 00000000..9f4041c3 --- /dev/null +++ b/index.html @@ -0,0 +1,938 @@ + + + + + + + + + + +FastHTML – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

FastHTML

+
+ +
+
+ The fastest, most powerful way to create an HTML app +
+
+ + +
+ + + + +
+ + + +
+ + + +

Welcome to the official FastHTML documentation.

+

FastHTML is a new next-generation web framework for fast, scalable web applications with minimal, compact code. It’s designed to be:

+
    +
  • Powerful and expressive enough to build the most advanced, interactive web apps you can imagine.
  • +
  • Fast and lightweight, so you can write less code and get more done.
  • +
  • Easy to learn and use, with a simple, intuitive syntax that makes it easy to build complex apps quickly.
  • +
+

FastHTML apps are just Python code, so you can use FastHTML with the full power of the Python language and ecosystem. FastHTML’s functionality maps 1:1 directly to HTML and HTTP, but allows them to be encapsulated using good software engineering practices—so you’ll need to understand these foundations to use this library fully. To understand how and why this works, please read this first: about.fastht.ml.

+
+

Installation

+

Since fasthtml is a Python library, you can install it with:

+
pip install python-fasthtml
+

In the near future, we hope to add component libraries that can likewise be installed via pip.

+
+
+

Usage

+

For a minimal app, create a file “main.py” as follows:

+
+
+
main.py
+
+
from fasthtml.common import *
+
+app,rt = fast_app()
+
+@rt('/')
+def get(): return Div(P('Hello World!'), hx_get="/change")
+
+serve()
+
+

Running the app with python main.py prints out a link to your running app: http://localhost:5001. Visit that link in your browser and you should see a page with the text “Hello World!”. Congratulations, you’ve just created your first FastHTML app!

+

Adding interactivity is surprisingly easy, thanks to HTMX. Modify the file to add this function:

+
+
+
main.py
+
+
@rt('/change')
+def get(): return P('Nice to be here!')
+
+

You now have a page with a clickable element that changes the text when clicked. When clicking on this link, the server will respond with an “HTML partial”—that is, just a snippet of HTML which will be inserted into the existing page. In this case, the returned element will replace the original P element (since that’s the default behavior of HTMX) with the new version returned by the second route.

+

This “hypermedia-based” approach to web development is a powerful way to build web applications.

+
+

Getting help from AI

+

Because FastHTML is newer than most LLMs, AI systems like Cursor, ChatGPT, Claude, and Copilot won’t give useful answers about it. To fix that problem, we’ve provided an LLM-friendly guide that teaches them how to use FastHTML. To use it, add this link for your AI helper to use:

+ +

This example is in a format based on recommendations from Anthropic for use with Claude Projects. This works so well that we’ve actually found that Claude can provide even better information than our own documentation! For instance, read through this annotated Claude chat for some great getting-started information, entirely generated from a project using the above text file as context.

+

If you use Cursor, type @doc then choose “Add new doc”, and use the /llms-ctx.txt link above. The context file is auto-generated from our llms.txt (our proposed standard for providing AI-friendly information)—you can generate alternative versions suitable for other models as needed.

+
+
+
+

Next Steps

+

Start with the official sources to learn more about FastHTML:

+
    +
  • About: Learn about the core ideas behind FastHTML
  • +
  • Documentation: Learn from examples how to write FastHTML code
  • +
  • Idiomatic app: Heavily commented source code walking through a complete application, including custom authentication, JS library connections, and database use.
  • +
+

We also have a 1-hour intro video:

+
+

The capabilities of FastHTML are vast and growing, and not all the features and patterns have been documented yet. Be prepared to invest time into studying and modifying source code, such as the main FastHTML repo’s notebooks and the official FastHTML examples repo:

+ +

Then explore the small but growing third-party ecosystem of FastHTML tutorials, notebooks, libraries, and components:

+ +

Finally, join the FastHTML community to ask questions, share your work, and learn from others:

+ +
+ + +
+ +
+ + + + + \ No newline at end of file diff --git a/index.html.md b/index.html.md new file mode 100644 index 00000000..4e58cee1 --- /dev/null +++ b/index.html.md @@ -0,0 +1,203 @@ +# FastHTML + + + + +Welcome to the official FastHTML documentation. + +FastHTML is a new next-generation web framework for fast, scalable web +applications with minimal, compact code. It’s designed to be: + +- Powerful and expressive enough to build the most advanced, interactive + web apps you can imagine. +- Fast and lightweight, so you can write less code and get more done. +- Easy to learn and use, with a simple, intuitive syntax that makes it + easy to build complex apps quickly. + +FastHTML apps are just Python code, so you can use FastHTML with the +full power of the Python language and ecosystem. FastHTML’s +functionality maps 1:1 directly to HTML and HTTP, but allows them to be +encapsulated using good software engineering practices—so you’ll need to +understand these foundations to use this library fully. To understand +how and why this works, please read this first: +[about.fastht.ml](https://about.fastht.ml/). + +## Installation + +Since `fasthtml` is a Python library, you can install it with: + +``` sh +pip install python-fasthtml +``` + +In the near future, we hope to add component libraries that can likewise +be installed via `pip`. + +## Usage + +For a minimal app, create a file “main.py” as follows: + +
+ +**main.py** + +``` python +from fasthtml.common import * + +app,rt = fast_app() + +@rt('/') +def get(): return Div(P('Hello World!'), hx_get="/change") + +serve() +``` + +
+ +Running the app with `python main.py` prints out a link to your running +app: `http://localhost:5001`. Visit that link in your browser and you +should see a page with the text “Hello World!”. Congratulations, you’ve +just created your first FastHTML app! + +Adding interactivity is surprisingly easy, thanks to HTMX. Modify the +file to add this function: + +
+ +**main.py** + +``` python +@rt('/change') +def get(): return P('Nice to be here!') +``` + +
+ +You now have a page with a clickable element that changes the text when +clicked. When clicking on this link, the server will respond with an +“HTML partial”—that is, just a snippet of HTML which will be inserted +into the existing page. In this case, the returned element will replace +the original P element (since that’s the default behavior of HTMX) with +the new version returned by the second route. + +This “hypermedia-based” approach to web development is a powerful way to +build web applications. + +### Getting help from AI + +Because FastHTML is newer than most LLMs, AI systems like Cursor, +ChatGPT, Claude, and Copilot won’t give useful answers about it. To fix +that problem, we’ve provided an LLM-friendly guide that teaches them how +to use FastHTML. To use it, add this link for your AI helper to use: + +- [/llms-ctx.txt](https://docs.fastht.ml/llms-ctx.txt) + +This example is in a format based on recommendations from Anthropic for +use with [Claude +Projects](https://support.anthropic.com/en/articles/9517075-what-are-projects). +This works so well that we’ve actually found that Claude can provide +even better information than our own documentation! For instance, read +through [this annotated Claude +chat](https://gist.github.com/jph00/9559b0a563f6a370029bec1d1cc97b74) +for some great getting-started information, entirely generated from a +project using the above text file as context. + +If you use Cursor, type `@doc` then choose “*Add new doc*”, and use the +/llms-ctx.txt link above. The context file is auto-generated from our +[`llms.txt`](https://llmstxt.org/) (our proposed standard for providing +AI-friendly information)—you can generate alternative versions suitable +for other models as needed. + +## Next Steps + +Start with the official sources to learn more about FastHTML: + +- [About](https://about.fastht.ml): Learn about the core ideas behind + FastHTML +- [Documentation](https://docs.fastht.ml): Learn from examples how to + write FastHTML code +- [Idiomatic + app](https://github.com/AnswerDotAI/fasthtml/blob/main/examples/adv_app.py): + Heavily commented source code walking through a complete application, + including custom authentication, JS library connections, and database + use. + +We also have a 1-hour intro video: + + + +The capabilities of FastHTML are vast and growing, and not all the +features and patterns have been documented yet. Be prepared to invest +time into studying and modifying source code, such as the main FastHTML +repo’s notebooks and the official FastHTML examples repo: + +- [FastHTML Examples Repo on + GitHub](https://github.com/AnswerDotAI/fasthtml-example) +- [FastHTML Repo on GitHub](https://github.com/AnswerDotAI/fasthtml) + +Then explore the small but growing third-party ecosystem of FastHTML +tutorials, notebooks, libraries, and components: + +- [FastHTML Gallery](https://gallery.fastht.ml): Learn from minimal + examples of components (ie chat bubbles, click-to-edit, infinite + scroll, etc) +- [Creating Custom FastHTML Tags for Markdown + Rendering](https://isaac-flath.github.io/website/posts/boots/FasthtmlTutorial.html) + by Isaac Flath +- [How to Build a Simple Login System in + FastHTML](https://blog.mariusvach.com/posts/login-fasthtml) by Marius + Vach +- Your tutorial here! + +Finally, join the FastHTML community to ask questions, share your work, +and learn from others: + +- [Discord](https://discord.gg/qcXvcxMhdP) + +## Other languages and related projects + +If you’re not a Python user, or are keen to try out a new language, +we’ll list here other projects that have a similar approach to FastHTML. +(Please reach out if you know of any other projects that you’d like to +see added.) + +- [htmgo](https://htmgo.dev/) (Go): “*htmgo is a lightweight pure go way + to build interactive websites / web applications using go & htmx. By + combining the speed & simplicity of go + hypermedia attributes (htmx) + to add interactivity to websites, all conveniently wrapped in pure go, + you can build simple, fast, interactive websites without touching + javascript. All compiled to a single deployable binary*” + +If you’re just interested in functional HTML components, rather than a +full HTMX server solution, consider: + +- [fastcore.xml.FT](https://fastcore.fast.ai/xml.html): This is actually + what FastHTML uses behind the scenes +- [htpy](https://htpy.dev/): Similar to + [`fastcore.xml.FT`](https://fastcore.fast.ai/xml.html#ft), but with a + somewhat different syntax +- [elm-html](https://package.elm-lang.org/packages/elm/html/latest/): + Elm’s built-in HTML library with a type-safe functional approach +- [hiccup](https://github.com/weavejester/hiccup): Popular library for + representing HTML in Clojure using vectors +- [hiccl](https://github.com/garlic0x1/hiccl): HTML generation library + for Common Lisp inspired by Clojure’s Hiccup +- [Falco.Markup](https://github.com/pimbrouwers/Falco): F# HTML DSL and + web framework with type-safe HTML generation +- [Lucid](https://github.com/chrisdone/lucid): Type-safe HTML generation + for Haskell using monad transformers +- [dream-html](https://github.com/aantron/dream): Part of the Dream web + framework for OCaml, provides type-safe HTML templating + +For other hypermedia application platforms, not based on HTMX, take a +look at: + +- [Hotwire/Turbo](https://turbo.hotwired.dev/): Rails-oriented framework + that similarly uses HTML-over-the-wire +- [LiveView](https://hexdocs.pm/phoenix_live_view/Phoenix.LiveView.html): + Phoenix framework’s solution for building interactive web apps with + minimal JavaScript +- [Unpoly](https://unpoly.com/): Another HTML-over-the-wire framework + with progressive enhancement +- [Livewire](https://laravel-livewire.com/): Laravel’s take on building + dynamic interfaces with minimal JavaScript diff --git a/listings.json b/listings.json new file mode 100644 index 00000000..0f6cf2ba --- /dev/null +++ b/listings.json @@ -0,0 +1,11 @@ +[ + { + "listing": "/tutorials/index.html", + "items": [ + "/tutorials/by_example.html", + "/tutorials/quickstart_for_web_devs.html", + "/tutorials/e2e.html", + "/tutorials/jupyter_and_fasthtml.html" + ] + } +] \ No newline at end of file diff --git a/llms-ctx-full.txt b/llms-ctx-full.txt new file mode 100644 index 00000000..bb7c0e09 --- /dev/null +++ b/llms-ctx-full.txt @@ -0,0 +1,4857 @@ +Things to remember when writing FastHTML apps: + +- Although parts of its API are inspired by FastAPI, it is *not* compatible with FastAPI syntax and is not targeted at creating API services +- FastHTML includes support for Pico CSS and the fastlite sqlite library, although using both are optional; sqlalchemy can be used directly or via the fastsql library, and any CSS framework can be used. Support for the Surreal and css-scope-inline libraries are also included, but both are optional +- FastHTML is compatible with JS-native web components and any vanilla JS library, but not with React, Vue, or Svelte +- Use `serve()` for running uvicorn (`if __name__ == "__main__"` is not needed since it's automatic) +- When a title is needed with a response, use `Titled`; note that that already wraps children in `Container`, and already includes both the meta title as well as the H1 element.# Web Devs Quickstart + + + +## Installation + +``` bash +pip install python-fasthtml +``` + +## A Minimal Application + +A minimal FastHTML application looks something like this: + +
+ +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/") +def get(): + return Titled("FastHTML", P("Let's do this!")) + +serve() +``` + +
+ +Line 1 +We import what we need for rapid development! A carefully-curated set of +FastHTML functions and other Python objects is brought into our global +namespace for convenience. + +Line 3 +We instantiate a FastHTML app with the `fast_app()` utility function. +This provides a number of really useful defaults that we’ll take +advantage of later in the tutorial. + +Line 5 +We use the `rt()` decorator to tell FastHTML what to return when a user +visits `/` in their browser. + +Line 6 +We connect this route to HTTP GET requests by defining a view function +called `get()`. + +Line 7 +A tree of Python function calls that return all the HTML required to +write a properly formed web page. You’ll soon see the power of this +approach. + +Line 9 +The +[`serve()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#serve) +utility configures and runs FastHTML using a library called `uvicorn`. + +Run the code: + +``` bash +python main.py +``` + +The terminal will look like this: + +``` bash +INFO: Uvicorn running on http://0.0.0.0:5001 (Press CTRL+C to quit) +INFO: Started reloader process [58058] using WatchFiles +INFO: Started server process [58060] +INFO: Waiting for application startup. +INFO: Application startup complete. +``` + +Confirm FastHTML is running by opening your web browser to +[127.0.0.1:5001](http://127.0.0.1:5001). You should see something like +the image below: + +![](quickstart-web-dev/quickstart-fasthtml.png) + +
+ +> **Note** +> +> While some linters and developers will complain about the wildcard +> import, it is by design here and perfectly safe. FastHTML is very +> deliberate about the objects it exports in `fasthtml.common`. If it +> bothers you, you can import the objects you need individually, though +> it will make the code more verbose and less readable. +> +> If you want to learn more about how FastHTML handles imports, we cover +> that [here](https://docs.fastht.ml/explains/faq.html#why-use-import). + +
+ +## A Minimal Charting Application + +The +[`Script`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#script) +function allows you to include JavaScript. You can use Python to +generate parts of your JS or JSON like this: + +``` python +import json +from fasthtml.common import * + +app, rt = fast_app(hdrs=(Script(src="https://cdn.plot.ly/plotly-2.32.0.min.js"),)) + +data = json.dumps({ + "data": [{"x": [1, 2, 3, 4],"type": "scatter"}, + {"x": [1, 2, 3, 4],"y": [16, 5, 11, 9],"type": "scatter"}], + "title": "Plotly chart in FastHTML ", + "description": "This is a demo dashboard", + "type": "scatter" +}) + + +@rt("/") +def get(): + return Titled("Chart Demo", Div(id="myDiv"), + Script(f"var data = {data}; Plotly.newPlot('myDiv', data);")) + +serve() +``` + +## Debug Mode + +When we can’t figure out a bug in FastHTML, we can run it in `DEBUG` +mode. When an error is thrown, the error screen is displayed in the +browser. This error setting should never be used in a deployed app. + +``` python +from fasthtml.common import * + +app, rt = fast_app(debug=True) + +@rt("/") +def get(): + 1/0 + return Titled("FastHTML Error!", P("Let's error!")) + +serve() +``` + +Line 3 +`debug=True` sets debug mode on. + +Line 7 +Python throws an error when it tries to divide an integer by zero. + +## Routing + +FastHTML builds upon FastAPI’s friendly decorator pattern for specifying +URLs, with extra features: + +
+ +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/") +def get(): + return Titled("FastHTML", P("Let's do this!")) + +@rt("/hello") +def get(): + return Titled("Hello, world!") + +serve() +``` + +
+ +Line 5 +The “/” URL on line 5 is the home of a project. This would be accessed +at [127.0.0.1:5001](http://127.0.0.1:5001). + +Line 9 +“/hello” URL on line 9 will be found by the project if the user visits +[127.0.0.1:5001/hello](http://127.0.0.1:5001/hello). + +
+ +> **Tip** +> +> It looks like `get()` is being defined twice, but that’s not the case. +> Each function decorated with `rt` is totally separate, and is injected +> into the router. We’re not calling them in the module’s namespace +> (`locals()`). Rather, we’re loading them into the routing mechanism +> using the `rt` decorator. + +
+ +You can do more! Read on to learn what we can do to make parts of the +URL dynamic. + +## Variables in URLs + +You can add variable sections to a URL by marking them with +`{variable_name}`. Your function then receives the `{variable_name}` as +a keyword argument, but only if it is the correct type. Here’s an +example: + +
+ +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/{name}/{age}") +def get(name: str, age: int): + return Titled(f"Hello {name.title()}, age {age}") + +serve() +``` + +
+ +Line 5 +We specify two variable names, `name` and `age`. + +Line 6 +We define two function arguments named identically to the variables. You +will note that we specify the Python types to be passed. + +Line 7 +We use these functions in our project. + +Try it out by going to this address: +[127.0.0.1:5001/uma/5](http://127.0.0.1:5001/uma/5). You should get a +page that says, + +> “Hello Uma, age 5”. + +### What happens if we enter incorrect data? + +The [127.0.0.1:5001/uma/5](http://127.0.0.1:5001/uma/5) URL works +because `5` is an integer. If we enter something that is not, such as +[127.0.0.1:5001/uma/five](http://127.0.0.1:5001/uma/five), then FastHTML +will return an error instead of a web page. + +
+ +> **FastHTML URL routing supports more complex types** +> +> The two examples we provide here use Python’s built-in `str` and `int` +> types, but you can use your own types, including more complex ones +> such as those defined by libraries like +> [attrs](https://pypi.org/project/attrs/), +> [pydantic](https://pypi.org/project/pydantic/), and even +> [sqlmodel](https://pypi.org/project/sqlmodel/). + +
+ +## HTTP Methods + +FastHTML matches function names to HTTP methods. So far the URL routes +we’ve defined have been for HTTP GET methods, the most common method for +web pages. + +Form submissions often are sent as HTTP POST. When dealing with more +dynamic web page designs, also known as Single Page Apps (SPA for +short), the need can arise for other methods such as HTTP PUT and HTTP +DELETE. The way FastHTML handles this is by changing the function name. + +
+ +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/") +def get(): + return Titled("HTTP GET", P("Handle GET")) + +@rt("/") +def post(): + return Titled("HTTP POST", P("Handle POST")) + +serve() +``` + +
+ +Line 6 +On line 6 because the `get()` function name is used, this will handle +HTTP GETs going to the `/` URI. + +Line 10 +On line 10 because the `post()` function name is used, this will handle +HTTP POSTs going to the `/` URI. + +## CSS Files and Inline Styles + +Here we modify default headers to demonstrate how to use the [Sakura CSS +microframework](https://github.com/oxalorg/sakura) instead of FastHTML’s +default of Pico CSS. + +
+ +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app( + pico=False, + hdrs=( + Link(rel='stylesheet', href='assets/normalize.min.css', type='text/css'), + Link(rel='stylesheet', href='assets/sakura.css', type='text/css'), + Style("p {color: red;}") +)) + +@app.get("/") +def home(): + return Titled("FastHTML", + P("Let's do this!"), + ) + +serve() +``` + +
+ +Line 4 +By setting `pico` to `False`, FastHTML will not include `pico.min.css`. + +Line 7 +This will generate an HTML `` tag for sourcing the css for Sakura. + +Line 8 +If you want an inline styles, the +[`Style()`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#style) +function will put the result into the HTML. + +## Other Static Media File Locations + +As you saw, +[`Script`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#script) +and `Link` are specific to the most common static media use cases in web +apps: including JavaScript, CSS, and images. But it also works with +videos and other static media files. The default behavior is to look for +these files in the root directory - typically we don’t do anything +special to include them. We can change the default directory that is +looked in for files by adding the `static_path` parameter to the +`fast_app` function. + +``` python +app, rt = fast_app(static_path='public') +``` + +FastHTML also allows us to define a route that uses `FileResponse` to +serve the file at a specified path. This is useful for serving images, +videos, and other media files from a different directory without having +to change the paths of many files. So if we move the directory +containing the media files, we only need to change the path in one +place. In the example below, we call images from a directory called +`public`. + +``` python +@rt("/{fname:path}.{ext:static}") +async def get(fname:str, ext:str): + return FileResponse(f'public/{fname}.{ext}') +``` + +## Rendering Markdown + +``` python +from fasthtml.common import * + +hdrs = (MarkdownJS(), HighlightJS(langs=['python', 'javascript', 'html', 'css']), ) + +app, rt = fast_app(hdrs=hdrs) + +content = """ +Here are some _markdown_ elements. + +- This is a list item +- This is another list item +- And this is a third list item + +**Fenced code blocks work here.** +""" + +@rt('/') +def get(req): + return Titled("Markdown rendering example", Div(content,cls="marked")) + +serve() +``` + +## Code highlighting + +Here’s how to highlight code without any markdown configuration. + +``` python +from fasthtml.common import * + +# Add the HighlightJS built-in header +hdrs = (HighlightJS(langs=['python', 'javascript', 'html', 'css']),) + +app, rt = fast_app(hdrs=hdrs) + +code_example = """ +import datetime +import time + +for i in range(10): + print(f"{datetime.datetime.now()}") + time.sleep(1) +""" + +@rt('/') +def get(req): + return Titled("Markdown rendering example", + Div( + # The code example needs to be surrounded by + # Pre & Code elements + Pre(Code(code_example)) + )) + +serve() +``` + +## Defining new `ft` components + +We can build our own `ft` components and combine them with other +components. The simplest method is defining them as a function. + +``` python +from fasthtml.common import * +``` + +``` python +def hero(title, statement): + return Div(H1(title),P(statement), cls="hero") + +# usage example +Main( + hero("Hello World", "This is a hero statement") +) +``` + +``` html +
+

Hello World

+

This is a hero statement

+
+
+``` + +### Pass through components + +For when we need to define a new component that allows zero-to-many +components to be nested within them, we lean on Python’s `*args` and +`**kwargs` mechanism. Useful for creating page layout controls. + +``` python +def layout(*args, **kwargs): + """Dashboard layout for all our dashboard views""" + return Main( + H1("Dashboard"), + Div(*args, **kwargs), + cls="dashboard", + ) + +# usage example +layout( + Ul(*[Li(o) for o in range(3)]), + P("Some content", cls="description"), +) +``` + +``` html +

Dashboard

+
+
    +
  • 0
  • +
  • 1
  • +
  • 2
  • +
+

Some content

+
+
+``` + +### Dataclasses as ft components + +While functions are easy to read, for more complex components some might +find it easier to use a dataclass. + +``` python +from dataclasses import dataclass + +@dataclass +class Hero: + title: str + statement: str + + def __ft__(self): + """ The __ft__ method renders the dataclass at runtime.""" + return Div(H1(self.title),P(self.statement), cls="hero") + +# usage example +Main( + Hero("Hello World", "This is a hero statement") +) +``` + +``` html +
+

Hello World

+

This is a hero statement

+
+
+``` + +## Testing views in notebooks + +Because of the ASGI event loop it is currently impossible to run +FastHTML inside a notebook. However, we can still test the output of our +views. To do this, we leverage Starlette, an ASGI toolkit that FastHTML +uses. + +``` python +# First we instantiate our app, in this case we remove the +# default headers to reduce the size of the output. +app, rt = fast_app(default_hdrs=False) + +# Setting up the Starlette test client +from starlette.testclient import TestClient +client = TestClient(app) + +# Usage example +@rt("/") +def get(): + return Titled("FastHTML is awesome", + P("The fastest way to create web apps in Python")) + +print(client.get("/").text) +``` + + + + + FastHTML is awesome + +

FastHTML is awesome

+

The fastest way to create web apps in Python

+
+ + +## Forms + +To validate data coming from users, first define a dataclass +representing the data you want to check. Here’s an example representing +a signup form. + +``` python +from dataclasses import dataclass + +@dataclass +class Profile: email:str; phone:str; age:int +``` + +Create an FT component representing an empty version of that form. Don’t +pass in any value to fill the form, that gets handled later. + +``` python +profile_form = Form(method="post", action="/profile")( + Fieldset( + Label('Email', Input(name="email")), + Label("Phone", Input(name="phone")), + Label("Age", Input(name="age")), + ), + Button("Save", type="submit"), + ) +profile_form +``` + +``` html +
+``` + +Once the dataclass and form function are completed, we can add data to +the form. To do that, instantiate the profile dataclass: + +``` python +profile = Profile(email='john@example.com', phone='123456789', age=5) +profile +``` + + Profile(email='john@example.com', phone='123456789', age=5) + +Then add that data to the `profile_form` using FastHTML’s +[`fill_form`](https://AnswerDotAI.github.io/fasthtml/api/components.html#fill_form) +class: + +``` python +fill_form(profile_form, profile) +``` + +``` html +
+``` + +### Forms with views + +The usefulness of FastHTML forms becomes more apparent when they are +combined with FastHTML views. We’ll show how this works by using the +test client from above. First, let’s create a SQlite database: + +``` python +db = database("profiles.db") +profiles = db.create(Profile, pk="email") +``` + +Now we insert a record into the database: + +``` python +profiles.insert(profile) +``` + + Profile(email='john@example.com', phone='123456789', age=5) + +And we can then demonstrate in the code that form is filled and +displayed to the user. + +``` python +@rt("/profile/{email}") +def profile(email:str): + profile = profiles[email] + filled_profile_form = fill_form(profile_form, profile) + return Titled(f'Profile for {profile.email}', filled_profile_form) + +print(client.get(f"/profile/john@example.com").text) +``` + +Line 3 +Fetch the profile using the profile table’s `email` primary key + +Line 4 +Fill the form for display. + + + + + + Profile for john@example.com + +

Profile for john@example.com

+
+ + +And now let’s demonstrate making a change to the data. + +``` python +@rt("/profile") +def post(profile: Profile): + profiles.update(profile) + return RedirectResponse(url=f"/profile/{profile.email}") + +new_data = dict(email='john@example.com', phone='7654321', age=25) +print(client.post("/profile", data=new_data).text) +``` + +Line 2 +We use the `Profile` dataclass definition to set the type for the +incoming `profile` content. This validates the field types for the +incoming data + +Line 3 +Taking our validated data, we updated the profiles table + +Line 4 +We redirect the user back to their profile view + +Line 7 +The display is of the profile form view showing the changes in data. + + + + + + Profile for john@example.com + +

Profile for john@example.com

+
+ + +## Strings and conversion order + +The general rules for rendering are: - `__ft__` method will be called +(for default components like `P`, `H2`, etc. or if you define your own +components) - If you pass a string, it will be escaped - On other python +objects, `str()` will be called + +As a consequence, if you want to include plain HTML tags directly into +e.g. a `Div()` they will get escaped by default (as a security measure +to avoid code injections). This can be avoided by using `NotStr()`, a +convenient way to reuse python code that returns already HTML. If you +use pandas, you can use `pandas.DataFrame.to_html()` to get a nice +table. To include the output a FastHTML, wrap it in `NotStr()`, like +`Div(NotStr(df.to_html()))`. + +Above we saw how a dataclass behaves with the `__ft__` method defined. +On a plain dataclass, `str()` will be called (but not escaped). + +``` python +from dataclasses import dataclass + +@dataclass +class Hero: + title: str + statement: str + +# rendering the dataclass with the default method +Main( + Hero("

Hello World

", "This is a hero statement") +) +``` + +``` html +
Hero(title='

Hello World

', statement='This is a hero statement')
+``` + +``` python +# This will display the HTML as text on your page +Div("Let's include some HTML here:
Some HTML
") +``` + +``` html +
Let's include some HTML here: <div>Some HTML</div>
+``` + +``` python +# Keep the string untouched, will be rendered on the page +Div(NotStr("

Some HTML

")) +``` + +``` html +

Some HTML

+``` + +## Custom exception handlers + +FastHTML allows customization of exception handlers, but does so +gracefully. What this means is by default it includes all the `` +tags needed to display attractive content. Try it out! + +``` python +from fasthtml.common import * + +def not_found(req, exc): return Titled("404: I don't exist!") + +exception_handlers = {404: not_found} + +app, rt = fast_app(exception_handlers=exception_handlers) + +@rt('/') +def get(): + return (Titled("Home page", P(A(href="/oops")("Click to generate 404 error")))) + +serve() +``` + +We can also use lambda to make things more terse: + +``` python +from fasthtml.common import * + +exception_handlers={ + 404: lambda req, exc: Titled("404: I don't exist!"), + 418: lambda req, exc: Titled("418: I'm a teapot!") +} + +app, rt = fast_app(exception_handlers=exception_handlers) + +@rt('/') +def get(): + return (Titled("Home page", P(A(href="/oops")("Click to generate 404 error")))) + +serve() +``` + +## Cookies + +We can set cookies using the +[`cookie()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#cookie) +function. In our example, we’ll create a `timestamp` cookie. + +``` python +from datetime import datetime +from IPython.display import HTML +``` + +``` python +@rt("/settimestamp") +def get(req): + now = datetime.now() + return P(f'Set to {now}'), cookie('now', datetime.now()) + +HTML(client.get('/settimestamp').text) +``` + + + + +FastHTML page + +

Set to 2024-09-26 15:33:48.141869

+ + + +Now let’s get it back using the same name for our parameter as the +cookie name. + +``` python +@rt('/gettimestamp') +def get(now:parsed_date): return f'Cookie was set at time {now.time()}' + +client.get('/gettimestamp').text +``` + + 'Cookie was set at time 15:33:48.141903' + +## Sessions + +For convenience and security, FastHTML has a mechanism for storing small +amounts of data in the user’s browser. We can do this by adding a +`session` argument to routes. FastHTML sessions are Python dictionaries, +and we can leverage to our benefit. The example below shows how to +concisely set and get sessions. + +``` python +@rt('/adder/{num}') +def get(session, num: int): + session.setdefault('sum', 0) + session['sum'] = session.get('sum') + num + return Response(f'The sum is {session["sum"]}.') +``` + +## Toasts (also known as Messages) + +Toasts, sometimes called “Messages” are small notifications usually in +colored boxes used to notify users that something has happened. Toasts +can be of four types: + +- info +- success +- warning +- error + +Examples toasts might include: + +- “Payment accepted” +- “Data submitted” +- “Request approved” + +Toasts require the use of the `setup_toasts()` function plus every view +needs these two features: + +- The session argument +- Must return FT components + +``` python +setup_toasts(app) + +@rt('/toasting') +def get(session): + # Normally one toast is enough, this allows us to see + # different toast types in action. + add_toast(session, f"Toast is being cooked", "info") + add_toast(session, f"Toast is ready", "success") + add_toast(session, f"Toast is getting a bit crispy", "warning") + add_toast(session, f"Toast is burning!", "error") + return Titled("I like toast") +``` + +Line 1 +`setup_toasts` is a helper function that adds toast dependencies. +Usually this would be declared right after `fast_app()` + +Line 4 +Toasts require sessions + +Line 11 +Views with Toasts must return FT or FtResponse components. + +💡 `setup_toasts` takes a `duration` input that allows you to specify +how long a toast will be visible before disappearing. For example +`setup_toasts(duration=5)` sets the toasts duration to 5 seconds. By +default toasts disappear after 10 seconds. + +## Authentication and authorization + +In FastHTML the tasks of authentication and authorization are handled +with Beforeware. Beforeware are functions that run before the route +handler is called. They are useful for global tasks like ensuring users +are authenticated or have permissions to access a view. + +First, we write a function that accepts a request and session arguments: + +``` python +# Status code 303 is a redirect that can change POST to GET, +# so it's appropriate for a login page. +login_redir = RedirectResponse('/login', status_code=303) + +def user_auth_before(req, sess): + # The `auth` key in the request scope is automatically provided + # to any handler which requests it, and can not be injected + # by the user using query params, cookies, etc, so it should + # be secure to use. + auth = req.scope['auth'] = sess.get('auth', None) + # If the session key is not there, it redirects to the login page. + if not auth: return login_redir +``` + +Now we pass our `user_auth_before` function as the first argument into a +[`Beforeware`](https://AnswerDotAI.github.io/fasthtml/api/core.html#beforeware) +class. We also pass a list of regular expressions to the `skip` +argument, designed to allow users to still get to the home and login +pages. + +``` python +beforeware = Beforeware( + user_auth_before, + skip=[r'/favicon\.ico', r'/static/.*', r'.*\.css', r'.*\.js', '/login', '/'] +) + +app, rt = fast_app(before=beforeware) +``` + +## Server-sent events (SSE) + +With [server-sent +events](https://developer.mozilla.org/en-US/docs/Web/API/Server-sent_events), +it’s possible for a server to send new data to a web page at any time, +by pushing messages to the web page. Unlike WebSockets, SSE can only go +in one direction: server to client. SSE is also part of the HTTP +specification unlike WebSockets which uses its own specification. + +FastHTML introduces several tools for working with SSE which are covered +in the example below. While concise, there’s a lot going on in this +function so we’ve annotated it quite a bit. + +``` python +import random +from asyncio import sleep +from fasthtml.common import * + +hdrs=(Script(src="https://unpkg.com/htmx-ext-sse@2.2.1/sse.js"),) +app,rt = fast_app(hdrs=hdrs) + +@rt +def index(): + return Titled("SSE Random Number Generator", + P("Generate pairs of random numbers, as the list grows scroll downwards."), + Div(hx_ext="sse", + sse_connect="/number-stream", + hx_swap="beforeend show:bottom", + sse_swap="message")) + +shutdown_event = signal_shutdown() + +async def number_generator(): + while not shutdown_event.is_set(): + data = Article(random.randint(1, 100)) + yield sse_message(data) + await sleep(1) + +@rt("/number-stream") +async def get(): return EventStream(number_generator()) +``` + +Line 5 +Import the HTMX SSE extension + +Line 12 +Tell HTMX to load the SSE extension + +Line 13 +Look at the `/number-stream` endpoint for SSE content + +Line 14 +When new items come in from the SSE endpoint, add them at the end of the +current content within the div. If they go beyond the screen, scroll +downwards + +Line 15 +Specify the name of the event. FastHTML’s default event name is +“message”. Only change if you have more than one call to SSE endpoints +within a view + +Line 17 +Set up the asyncio event loop + +Line 19 +Don’t forget to make this an `async` function! + +Line 20 +Iterate through the asyncio event loop + +Line 22 +We yield the data. Data ideally should be comprised of FT components as +that plugs nicely into HTMX in the browser + +Line 26 +The endpoint view needs to be an async function that returns a +[`EventStream`](https://AnswerDotAI.github.io/fasthtml/api/core.html#eventstream) + +## Websockets + +With websockets we can have bi-directional communications between a +browser and client. Websockets are useful for things like chat and +certain types of games. While websockets can be used for single +direction messages from the server (i.e. telling users that a process is +finished), that task is arguably better suited for SSE. + +FastHTML provides useful tools for adding websockets to your pages. + +``` python +from fasthtml.common import * +from asyncio import sleep + +app, rt = fast_app(exts='ws') + +def mk_inp(): return Input(id='msg', autofocus=True) + +@rt('/') +async def get(request): + cts = Div( + Div(id='notifications'), + Form(mk_inp(), id='form', ws_send=True), + hx_ext='ws', ws_connect='/ws') + return Titled('Websocket Test', cts) + +async def on_connect(send): + print('Connected!') + await send(Div('Hello, you have connected', id="notifications")) + +async def on_disconnect(ws): + print('Disconnected!') + +@app.ws('/ws', conn=on_connect, disconn=on_disconnect) +async def ws(msg:str, send): + await send(Div('Hello ' + msg, id="notifications")) + await sleep(2) + return Div('Goodbye ' + msg, id="notifications"), mk_inp() +``` + +Line 4 +To use websockets in FastHTML, you must instantiate the app with `exts` +set to ‘ws’ + +Line 6 +As we want to use websockets to reset the form, we define the `mk_input` +function that can be called from multiple locations + +Line 12 +We create the form and mark it with the `ws_send` attribute, which is +documented here in the [HTMX websocket +specification](https://v1.htmx.org/extensions/web-sockets/). This tells +HTMX to send a message to the nearest websocket based on the trigger for +the form element, which for forms is pressing the `enter` key, an action +considered to be a form submission + +Line 13 +This is where the HTMX extension is loaded (`hx_ext='ws'`) and the +nearest websocket is defined (`ws_connect='/ws'`) + +Line 16 +When a websocket first connects we can optionally have it call a +function that accepts a `send` argument. The `send` argument will push a +message to the browser. + +Line 18 +Here we use the `send` function that was passed into the `on_connect` +function to send a `Div` with an `id` of `notifications` that HTMX +assigns to the element in the page that already has an `id` of +`notifications` + +Line 20 +When a websocket disconnects we can call a function which takes no +arguments. Typically the role of this function is to notify the server +to take an action. In this case, we print a simple message to the +console + +Line 23 +We use the `app.ws` decorator to mark that `/ws` is the route for our +websocket. We also pass in the two optional `conn` and `disconn` +parameters to this decorator. As a fun experiment, remove the `conn` and +`disconn` arguments and see what happens + +Line 24 +Define the `ws` function as async. This is necessary for ASGI to be able +to serve websockets. The function accepts two arguments, a `msg` that is +user input from the browser, and a `send` function for pushing data back +to the browser + +Line 25 +The `send` function is used here to send HTML back to the page. As the +HTML has an `id` of `notifications`, HTMX will overwrite what is already +on the page with the same ID + +Line 27 +The websocket function can also be used to return a value. In this case, +it is a tuple of two HTML elements. HTMX will take the elements and +replace them where appropriate. As both have `id` specified +(`notifications` and `msg` respectively), they will replace their +predecessor on the page. + +## File Uploads + +A common task in web development is uploading files. The examples below +are for uploading files to the hosting server, with information about +the uploaded file presented to the user. + +
+ +> **File uploads in production can be dangerous** +> +> File uploads can be the target of abuse, accidental or intentional. +> That means users may attempt to upload files that are too large or +> present a security risk. This is especially of concern for public +> facing apps. File upload security is outside the scope of this +> tutorial, for now we suggest reading the [OWASP File Upload Cheat +> Sheet](https://cheatsheetseries.owasp.org/cheatsheets/File_Upload_Cheat_Sheet.html). + +
+ +### Single File Uploads + +``` python +from fasthtml.common import * +from pathlib import Path + +app, rt = fast_app() + +upload_dir = Path("filez") +upload_dir.mkdir(exist_ok=True) + +@rt('/') +def get(): + return Titled("File Upload Demo", + Article( + Form(hx_post=upload, hx_target="#result-one")( + Input(type="file", name="file"), + Button("Upload", type="submit", cls='secondary'), + ), + Div(id="result-one") + ) + ) + +def FileMetaDataCard(file): + return Article( + Header(H3(file.filename)), + Ul( + Li('Size: ', file.size), + Li('Content Type: ', file.content_type), + Li('Headers: ', file.headers), + ) + ) + +@rt +async def upload(file: UploadFile): + card = FileMetaDataCard(file) + filebuffer = await file.read() + (upload_dir / file.filename).write_bytes(filebuffer) + return card + +serve() +``` + +Line 13 +Every form rendered with the +[`Form`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#form) FT +component defaults to `enctype="multipart/form-data"` + +Line 14 +Don’t forget to set the `Input` FT Component’s type to `file` + +Line 32 +The upload view should receive a [Starlette +UploadFile](https://www.starlette.io/requests/#request-files) type. You +can add other form variables + +Line 33 +We can access the metadata of the card (filename, size, content_type, +headers), a quick and safe process. We set that to the card variable + +Line 34 +In order to access the contents contained within a file we use the +`await` method to read() it. As files may be quite large or contain bad +data, this is a seperate step from accessing metadata + +Line 35 +This step shows how to use Python’s built-in `pathlib.Path` library to +write the file to disk. + +### Multiple File Uploads + +``` python +from fasthtml.common import * +from pathlib import Path + +app, rt = fast_app() + +upload_dir = Path("filez") +upload_dir.mkdir(exist_ok=True) + +@rt('/') +def get(): + return Titled("Multiple File Upload Demo", + Article( + Form(hx_post=upload_many, hx_target="#result-many")( + Input(type="file", name="files", multiple=True), + Button("Upload", type="submit", cls='secondary'), + ), + Div(id="result-many") + ) + ) + +def FileMetaDataCard(file): + return Article( + Header(H3(file.filename)), + Ul( + Li('Size: ', file.size), + Li('Content Type: ', file.content_type), + Li('Headers: ', file.headers), + ) + ) + +@rt +async def upload_many(files: list[UploadFile]): + cards = [] + for file in files: + cards.append(FileMetaDataCard(file)) + filebuffer = await file.read() + (upload_dir / file.filename).write_bytes(filebuffer) + return cards + +serve() +``` + +Line 13 +Every form rendered with the +[`Form`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#form) FT +component defaults to `enctype="multipart/form-data"` + +Line 14 +Don’t forget to set the `Input` FT Component’s type to `file` and assign +the multiple attribute to `True` + +Line 32 +The upload view should receive a `list` containing the [Starlette +UploadFile](https://www.starlette.io/requests/#request-files) type. You +can add other form variables + +Line 34 +Iterate through the files + +Line 35 +We can access the metadata of the card (filename, size, content_type, +headers), a quick and safe process. We add that to the cards variable + +Line 36 +In order to access the contents contained within a file we use the +`await` method to read() it. As files may be quite large or contain bad +data, this is a seperate step from accessing metadata + +Line 37 +This step shows how to use Python’s built-in `pathlib.Path` library to +write the file to disk.
+++ +title = "Reference" ++++ + +## Contents + +* [htmx Core Attributes](#attributes) +* [htmx Additional Attributes](#attributes-additional) +* [htmx CSS Classes](#classes) +* [htmx Request Headers](#request_headers) +* [htmx Response Headers](#response_headers) +* [htmx Events](#events) +* [htmx Extensions](/extensions) +* [JavaScript API](#api) +* [Configuration Options](#config) + +## Core Attribute Reference {#attributes} + +The most common attributes when using htmx. + +
+ +| Attribute | Description | +|--------------------------------------------------|--------------------------------------------------------------------------------------------------------------------| +| [`hx-get`](@/attributes/hx-get.md) | issues a `GET` to the specified URL | +| [`hx-post`](@/attributes/hx-post.md) | issues a `POST` to the specified URL | +| [`hx-on*`](@/attributes/hx-on.md) | handle events with inline scripts on elements | +| [`hx-push-url`](@/attributes/hx-push-url.md) | push a URL into the browser location bar to create history | +| [`hx-select`](@/attributes/hx-select.md) | select content to swap in from a response | +| [`hx-select-oob`](@/attributes/hx-select-oob.md) | select content to swap in from a response, somewhere other than the target (out of band) | +| [`hx-swap`](@/attributes/hx-swap.md) | controls how content will swap in (`outerHTML`, `beforeend`, `afterend`, ...) | +| [`hx-swap-oob`](@/attributes/hx-swap-oob.md) | mark element to swap in from a response (out of band) | +| [`hx-target`](@/attributes/hx-target.md) | specifies the target element to be swapped | +| [`hx-trigger`](@/attributes/hx-trigger.md) | specifies the event that triggers the request | +| [`hx-vals`](@/attributes/hx-vals.md) | add values to submit with the request (JSON format) | + +
+ +## Additional Attribute Reference {#attributes-additional} + +All other attributes available in htmx. + +
+ +| Attribute | Description | +|------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------| +| [`hx-boost`](@/attributes/hx-boost.md) | add [progressive enhancement](https://en.wikipedia.org/wiki/Progressive_enhancement) for links and forms | +| [`hx-confirm`](@/attributes/hx-confirm.md) | shows a `confirm()` dialog before issuing a request | +| [`hx-delete`](@/attributes/hx-delete.md) | issues a `DELETE` to the specified URL | +| [`hx-disable`](@/attributes/hx-disable.md) | disables htmx processing for the given node and any children nodes | +| [`hx-disabled-elt`](@/attributes/hx-disabled-elt.md) | adds the `disabled` attribute to the specified elements while a request is in flight | +| [`hx-disinherit`](@/attributes/hx-disinherit.md) | control and disable automatic attribute inheritance for child nodes | +| [`hx-encoding`](@/attributes/hx-encoding.md) | changes the request encoding type | +| [`hx-ext`](@/attributes/hx-ext.md) | extensions to use for this element | +| [`hx-headers`](@/attributes/hx-headers.md) | adds to the headers that will be submitted with the request | +| [`hx-history`](@/attributes/hx-history.md) | prevent sensitive data being saved to the history cache | +| [`hx-history-elt`](@/attributes/hx-history-elt.md) | the element to snapshot and restore during history navigation | +| [`hx-include`](@/attributes/hx-include.md) | include additional data in requests | +| [`hx-indicator`](@/attributes/hx-indicator.md) | the element to put the `htmx-request` class on during the request | +| [`hx-inherit`](@/attributes/hx-inherit.md) | control and enable automatic attribute inheritance for child nodes if it has been disabled by default | +| [`hx-params`](@/attributes/hx-params.md) | filters the parameters that will be submitted with a request | +| [`hx-patch`](@/attributes/hx-patch.md) | issues a `PATCH` to the specified URL | +| [`hx-preserve`](@/attributes/hx-preserve.md) | specifies elements to keep unchanged between requests | +| [`hx-prompt`](@/attributes/hx-prompt.md) | shows a `prompt()` before submitting a request | +| [`hx-put`](@/attributes/hx-put.md) | issues a `PUT` to the specified URL | +| [`hx-replace-url`](@/attributes/hx-replace-url.md) | replace the URL in the browser location bar | +| [`hx-request`](@/attributes/hx-request.md) | configures various aspects of the request | +| [`hx-sync`](@/attributes/hx-sync.md) | control how requests made by different elements are synchronized | +| [`hx-validate`](@/attributes/hx-validate.md) | force elements to validate themselves before a request | +| [`hx-vars`](@/attributes/hx-vars.md) | adds values dynamically to the parameters to submit with the request (deprecated, please use [`hx-vals`](@/attributes/hx-vals.md)) | + +
+ +## CSS Class Reference {#classes} + +
+ +| Class | Description | +|-----------|-------------| +| `htmx-added` | Applied to a new piece of content before it is swapped, removed after it is settled. +| `htmx-indicator` | A dynamically generated class that will toggle visible (opacity:1) when a `htmx-request` class is present +| `htmx-request` | Applied to either the element or the element specified with [`hx-indicator`](@/attributes/hx-indicator.md) while a request is ongoing +| `htmx-settling` | Applied to a target after content is swapped, removed after it is settled. The duration can be modified via [`hx-swap`](@/attributes/hx-swap.md). +| `htmx-swapping` | Applied to a target before any content is swapped, removed after it is swapped. The duration can be modified via [`hx-swap`](@/attributes/hx-swap.md). + +
+ +## HTTP Header Reference {#headers} + +### Request Headers Reference {#request_headers} + +
+ +| Header | Description | +|--------|-------------| +| `HX-Boosted` | indicates that the request is via an element using [hx-boost](@/attributes/hx-boost.md) +| `HX-Current-URL` | the current URL of the browser +| `HX-History-Restore-Request` | "true" if the request is for history restoration after a miss in the local history cache +| `HX-Prompt` | the user response to an [hx-prompt](@/attributes/hx-prompt.md) +| `HX-Request` | always "true" +| `HX-Target` | the `id` of the target element if it exists +| `HX-Trigger-Name` | the `name` of the triggered element if it exists +| `HX-Trigger` | the `id` of the triggered element if it exists + +
+ +### Response Headers Reference {#response_headers} + +
+ +| Header | Description | +|------------------------------------------------------|-------------| +| [`HX-Location`](@/headers/hx-location.md) | allows you to do a client-side redirect that does not do a full page reload +| [`HX-Push-Url`](@/headers/hx-push-url.md) | pushes a new url into the history stack +| [`HX-Redirect`](@/headers/hx-redirect.md) | can be used to do a client-side redirect to a new location +| `HX-Refresh` | if set to "true" the client-side will do a full refresh of the page +| [`HX-Replace-Url`](@/headers/hx-replace-url.md) | replaces the current URL in the location bar +| `HX-Reswap` | allows you to specify how the response will be swapped. See [hx-swap](@/attributes/hx-swap.md) for possible values +| `HX-Retarget` | a CSS selector that updates the target of the content update to a different element on the page +| `HX-Reselect` | a CSS selector that allows you to choose which part of the response is used to be swapped in. Overrides an existing [`hx-select`](@/attributes/hx-select.md) on the triggering element +| [`HX-Trigger`](@/headers/hx-trigger.md) | allows you to trigger client-side events +| [`HX-Trigger-After-Settle`](@/headers/hx-trigger.md) | allows you to trigger client-side events after the settle step +| [`HX-Trigger-After-Swap`](@/headers/hx-trigger.md) | allows you to trigger client-side events after the swap step + +
+ +## Event Reference {#events} + +
+ +| Event | Description | +|-------|-------------| +| [`htmx:abort`](@/events.md#htmx:abort) | send this event to an element to abort a request +| [`htmx:afterOnLoad`](@/events.md#htmx:afterOnLoad) | triggered after an AJAX request has completed processing a successful response +| [`htmx:afterProcessNode`](@/events.md#htmx:afterProcessNode) | triggered after htmx has initialized a node +| [`htmx:afterRequest`](@/events.md#htmx:afterRequest) | triggered after an AJAX request has completed +| [`htmx:afterSettle`](@/events.md#htmx:afterSettle) | triggered after the DOM has settled +| [`htmx:afterSwap`](@/events.md#htmx:afterSwap) | triggered after new content has been swapped in +| [`htmx:beforeCleanupElement`](@/events.md#htmx:beforeCleanupElement) | triggered before htmx [disables](@/attributes/hx-disable.md) an element or removes it from the DOM +| [`htmx:beforeOnLoad`](@/events.md#htmx:beforeOnLoad) | triggered before any response processing occurs +| [`htmx:beforeProcessNode`](@/events.md#htmx:beforeProcessNode) | triggered before htmx initializes a node +| [`htmx:beforeRequest`](@/events.md#htmx:beforeRequest) | triggered before an AJAX request is made +| [`htmx:beforeSwap`](@/events.md#htmx:beforeSwap) | triggered before a swap is done, allows you to configure the swap +| [`htmx:beforeSend`](@/events.md#htmx:beforeSend) | triggered just before an ajax request is sent +| [`htmx:beforeTransition`](@/events.md#htmx:beforeTransition) | triggered before the [View Transition](https://developer.mozilla.org/en-US/docs/Web/API/View_Transitions_API) wrapped swap occurs +| [`htmx:configRequest`](@/events.md#htmx:configRequest) | triggered before the request, allows you to customize parameters, headers +| [`htmx:confirm`](@/events.md#htmx:confirm) | triggered after a trigger occurs on an element, allows you to cancel (or delay) issuing the AJAX request +| [`htmx:historyCacheError`](@/events.md#htmx:historyCacheError) | triggered on an error during cache writing +| [`htmx:historyCacheMiss`](@/events.md#htmx:historyCacheMiss) | triggered on a cache miss in the history subsystem +| [`htmx:historyCacheMissError`](@/events.md#htmx:historyCacheMissError) | triggered on a unsuccessful remote retrieval +| [`htmx:historyCacheMissLoad`](@/events.md#htmx:historyCacheMissLoad) | triggered on a successful remote retrieval +| [`htmx:historyRestore`](@/events.md#htmx:historyRestore) | triggered when htmx handles a history restoration action +| [`htmx:beforeHistorySave`](@/events.md#htmx:beforeHistorySave) | triggered before content is saved to the history cache +| [`htmx:load`](@/events.md#htmx:load) | triggered when new content is added to the DOM +| [`htmx:noSSESourceError`](@/events.md#htmx:noSSESourceError) | triggered when an element refers to a SSE event in its trigger, but no parent SSE source has been defined +| [`htmx:onLoadError`](@/events.md#htmx:onLoadError) | triggered when an exception occurs during the onLoad handling in htmx +| [`htmx:oobAfterSwap`](@/events.md#htmx:oobAfterSwap) | triggered after an out of band element as been swapped in +| [`htmx:oobBeforeSwap`](@/events.md#htmx:oobBeforeSwap) | triggered before an out of band element swap is done, allows you to configure the swap +| [`htmx:oobErrorNoTarget`](@/events.md#htmx:oobErrorNoTarget) | triggered when an out of band element does not have a matching ID in the current DOM +| [`htmx:prompt`](@/events.md#htmx:prompt) | triggered after a prompt is shown +| [`htmx:pushedIntoHistory`](@/events.md#htmx:pushedIntoHistory) | triggered after an url is pushed into history +| [`htmx:responseError`](@/events.md#htmx:responseError) | triggered when an HTTP response error (non-`200` or `300` response code) occurs +| [`htmx:sendAbort`](@/events.md#htmx:sendAbort) | triggered when a request is aborted +| [`htmx:sendError`](@/events.md#htmx:sendError) | triggered when a network error prevents an HTTP request from happening +| [`htmx:sseError`](@/events.md#htmx:sseError) | triggered when an error occurs with a SSE source +| [`htmx:sseOpen`](/events#htmx:sseOpen) | triggered when a SSE source is opened +| [`htmx:swapError`](@/events.md#htmx:swapError) | triggered when an error occurs during the swap phase +| [`htmx:targetError`](@/events.md#htmx:targetError) | triggered when an invalid target is specified +| [`htmx:timeout`](@/events.md#htmx:timeout) | triggered when a request timeout occurs +| [`htmx:validation:validate`](@/events.md#htmx:validation:validate) | triggered before an element is validated +| [`htmx:validation:failed`](@/events.md#htmx:validation:failed) | triggered when an element fails validation +| [`htmx:validation:halted`](@/events.md#htmx:validation:halted) | triggered when a request is halted due to validation errors +| [`htmx:xhr:abort`](@/events.md#htmx:xhr:abort) | triggered when an ajax request aborts +| [`htmx:xhr:loadend`](@/events.md#htmx:xhr:loadend) | triggered when an ajax request ends +| [`htmx:xhr:loadstart`](@/events.md#htmx:xhr:loadstart) | triggered when an ajax request starts +| [`htmx:xhr:progress`](@/events.md#htmx:xhr:progress) | triggered periodically during an ajax request that supports progress events + +
+ +## JavaScript API Reference {#api} + +
+ +| Method | Description | +|-------|-------------| +| [`htmx.addClass()`](@/api.md#addClass) | Adds a class to the given element +| [`htmx.ajax()`](@/api.md#ajax) | Issues an htmx-style ajax request +| [`htmx.closest()`](@/api.md#closest) | Finds the closest parent to the given element matching the selector +| [`htmx.config`](@/api.md#config) | A property that holds the current htmx config object +| [`htmx.createEventSource`](@/api.md#createEventSource) | A property holding the function to create SSE EventSource objects for htmx +| [`htmx.createWebSocket`](@/api.md#createWebSocket) | A property holding the function to create WebSocket objects for htmx +| [`htmx.defineExtension()`](@/api.md#defineExtension) | Defines an htmx [extension](https://htmx.org/extensions) +| [`htmx.find()`](@/api.md#find) | Finds a single element matching the selector +| [`htmx.findAll()` `htmx.findAll(elt, selector)`](@/api.md#find) | Finds all elements matching a given selector +| [`htmx.logAll()`](@/api.md#logAll) | Installs a logger that will log all htmx events +| [`htmx.logger`](@/api.md#logger) | A property set to the current logger (default is `null`) +| [`htmx.off()`](@/api.md#off) | Removes an event listener from the given element +| [`htmx.on()`](@/api.md#on) | Creates an event listener on the given element, returning it +| [`htmx.onLoad()`](@/api.md#onLoad) | Adds a callback handler for the `htmx:load` event +| [`htmx.parseInterval()`](@/api.md#parseInterval) | Parses an interval declaration into a millisecond value +| [`htmx.process()`](@/api.md#process) | Processes the given element and its children, hooking up any htmx behavior +| [`htmx.remove()`](@/api.md#remove) | Removes the given element +| [`htmx.removeClass()`](@/api.md#removeClass) | Removes a class from the given element +| [`htmx.removeExtension()`](@/api.md#removeExtension) | Removes an htmx [extension](https://htmx.org/extensions) +| [`htmx.swap()`](@/api.md#swap) | Performs swapping (and settling) of HTML content +| [`htmx.takeClass()`](@/api.md#takeClass) | Takes a class from other elements for the given element +| [`htmx.toggleClass()`](@/api.md#toggleClass) | Toggles a class from the given element +| [`htmx.trigger()`](@/api.md#trigger) | Triggers an event on an element +| [`htmx.values()`](@/api.md#values) | Returns the input values associated with the given element + +
+ + +## Configuration Reference {#config} + +Htmx has some configuration options that can be accessed either programmatically or declaratively. They are +listed below: + +
+ +| Config Variable | Info | +|---------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| `htmx.config.historyEnabled` | defaults to `true`, really only useful for testing | +| `htmx.config.historyCacheSize` | defaults to 10 | +| `htmx.config.refreshOnHistoryMiss` | defaults to `false`, if set to `true` htmx will issue a full page refresh on history misses rather than use an AJAX request | +| `htmx.config.defaultSwapStyle` | defaults to `innerHTML` | +| `htmx.config.defaultSwapDelay` | defaults to 0 | +| `htmx.config.defaultSettleDelay` | defaults to 20 | +| `htmx.config.includeIndicatorStyles` | defaults to `true` (determines if the indicator styles are loaded) | +| `htmx.config.indicatorClass` | defaults to `htmx-indicator` | +| `htmx.config.requestClass` | defaults to `htmx-request` | +| `htmx.config.addedClass` | defaults to `htmx-added` | +| `htmx.config.settlingClass` | defaults to `htmx-settling` | +| `htmx.config.swappingClass` | defaults to `htmx-swapping` | +| `htmx.config.allowEval` | defaults to `true`, can be used to disable htmx's use of eval for certain features (e.g. trigger filters) | +| `htmx.config.allowScriptTags` | defaults to `true`, determines if htmx will process script tags found in new content | +| `htmx.config.inlineScriptNonce` | defaults to `''`, meaning that no nonce will be added to inline scripts | +| `htmx.config.inlineStyleNonce` | defaults to `''`, meaning that no nonce will be added to inline styles | +| `htmx.config.attributesToSettle` | defaults to `["class", "style", "width", "height"]`, the attributes to settle during the settling phase | +| `htmx.config.wsReconnectDelay` | defaults to `full-jitter` | +| `htmx.config.wsBinaryType` | defaults to `blob`, the [the type of binary data](https://developer.mozilla.org/docs/Web/API/WebSocket/binaryType) being received over the WebSocket connection | +| `htmx.config.disableSelector` | defaults to `[hx-disable], [data-hx-disable]`, htmx will not process elements with this attribute on it or a parent | +| `htmx.config.disableInheritance` | defaults to `false`. If it is set to `true`, the inheritance of attributes is completely disabled and you can explicitly specify the inheritance with the [hx-inherit](@/attributes/hx-inherit.md) attribute. +| `htmx.config.withCredentials` | defaults to `false`, allow cross-site Access-Control requests using credentials such as cookies, authorization headers or TLS client certificates | +| `htmx.config.timeout` | defaults to 0, the number of milliseconds a request can take before automatically being terminated | +| `htmx.config.scrollBehavior` | defaults to 'instant', the scroll behavior when using the [show](@/attributes/hx-swap.md#scrolling-scroll-show) modifier with `hx-swap`. The allowed values are `instant` (scrolling should happen instantly in a single jump), `smooth` (scrolling should animate smoothly) and `auto` (scroll behavior is determined by the computed value of [scroll-behavior](https://developer.mozilla.org/en-US/docs/Web/CSS/scroll-behavior)). | +| `htmx.config.defaultFocusScroll` | if the focused element should be scrolled into view, defaults to false and can be overridden using the [focus-scroll](@/attributes/hx-swap.md#focus-scroll) swap modifier. | +| `htmx.config.getCacheBusterParam` | defaults to false, if set to true htmx will append the target element to the `GET` request in the format `org.htmx.cache-buster=targetElementId` | +| `htmx.config.globalViewTransitions` | if set to `true`, htmx will use the [View Transition](https://developer.mozilla.org/en-US/docs/Web/API/View_Transitions_API) API when swapping in new content. | +| `htmx.config.methodsThatUseUrlParams` | defaults to `["get", "delete"]`, htmx will format requests with these methods by encoding their parameters in the URL, not the request body | +| `htmx.config.selfRequestsOnly` | defaults to `true`, whether to only allow AJAX requests to the same domain as the current document | +| `htmx.config.ignoreTitle` | defaults to `false`, if set to `true` htmx will not update the title of the document when a `title` tag is found in new content | +| `htmx.config.scrollIntoViewOnBoost` | defaults to `true`, whether or not the target of a boosted element is scrolled into the viewport. If `hx-target` is omitted on a boosted element, the target defaults to `body`, causing the page to scroll to the top. | +| `htmx.config.triggerSpecsCache` | defaults to `null`, the cache to store evaluated trigger specifications into, improving parsing performance at the cost of more memory usage. You may define a simple object to use a never-clearing cache, or implement your own system using a [proxy object](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Proxy) | +| `htmx.config.responseHandling` | the default [Response Handling](@/docs.md#response-handling) behavior for response status codes can be configured here to either swap or error | +| `htmx.config.allowNestedOobSwaps` | defaults to `true`, whether to process OOB swaps on elements that are nested within the main response element. See [Nested OOB Swaps](@/attributes/hx-swap-oob.md#nested-oob-swaps). | + +
+ +You can set them directly in javascript, or you can use a `meta` tag: + +```html + +```
# 🗿 Surreal +### Tiny jQuery alternative for plain Javascript with inline [Locality of Behavior](https://htmx.org/essays/locality-of-behaviour/)! + +![cover](https://user-images.githubusercontent.com/24665/171092805-b41286b2-be4a-4aab-9ee6-d604699cc507.png) +(Art by [shahabalizadeh](https://www.deviantart.com/shahabalizadeh)) + + +## Why does this exist? + +For devs who love ergonomics! You may appreciate Surreal if: + +* You want to stay as close as possible to Vanilla JS. +* Hate typing `document.querySelector` over.. and over.. +* Hate typing `addEventListener` over.. and over.. +* Really wish `document.querySelectorAll` had Array functions.. +* Really wish `this` would work in any inline ` + +``` + +See the [Live Example](https://gnat.github.io/surreal/example.html)! Then [view source](https://github.com/gnat/surreal/blob/main/example.html). + +## 🎁 Install + +Surreal is only 320 lines. No build step. No dependencies. + +[📥 Download](https://raw.githubusercontent.com/gnat/surreal/main/surreal.js) into your project, and add `` in your `` + +Or, 🌐 via CDN: `` + +## ⚡ Usage + +### 🔍️ DOM Selection + +* Select **one** element: `me(...)` + * Can be any of: + * CSS selector: `".button"`, `"#header"`, `"h1"`, `"body > .block"` + * Variables: `body`, `e`, `some_element` + * Events: `event.currentTarget` will be used. + * Surreal selectors: `me()`,`any()` + * Choose the start location in the DOM with the 2nd arg. (Default: `document`) + * 🔥 `any('button', me('#header')).classAdd('red')` + * Add `.red` to any ` + +``` +See the [Live Example](https://gnat.github.io/css-scope-inline/example.html)! Then [view source](https://github.com/gnat/css-scope-inline/blob/main/example.html). + +## 🌘 How does it work? + +This uses `MutationObserver` to monitor the DOM, and the moment a ` + red +
green
+
green
+
green
+
yellow
+
blue
+
green
+
green
+ + +
+ red +
green
+
green
+
green
+
yellow
+
blue
+
green
+
green
+
+``` + +### CSS variables and child elements +At first glance, **Tailwind Example 2** looks very promising! Exciting ...but: +* 🔴 **Every child style requires an explicit selector.** + * Tailwinds' shorthand advantages sadly disappear. + * Any more child styles added in Tailwind will become longer than vanilla CSS. + * This limited example is the best case scenario for Tailwind. +* 🔴 Not visible on github: **no highlighting for properties and units** begins to be painful. +```html + + + + + + + + + +
+ + + + + + +
+ + + + + + + + +``` +## 🔎 Technical FAQ +* Why do you use `querySelectorAll()` and not just process the `MutationObserver` results directly? + * This was indeed the original design; it will work well up until you begin recieving subtrees (ex: DOM swaps with [htmx](https://htmx.org), ajax, jquery, etc.) which requires walking all subtree elements to ensure we do not miss a ` + + +``` +See the [Live Example](https://gnat.github.io/css-scope-inline/example.html)! Then [view source](https://github.com/gnat/css-scope-inline/blob/main/example.html). + +## 🌘 How does it work? + +This uses `MutationObserver` to monitor the DOM, and the moment a ` + red +
green
+
green
+
green
+
yellow
+
blue
+
green
+
green
+ + +
+ red +
green
+
green
+
green
+
yellow
+
blue
+
green
+
green
+
+``` + +### CSS variables and child elements +At first glance, **Tailwind Example 2** looks very promising! Exciting ...but: +* 🔴 **Every child style requires an explicit selector.** + * Tailwinds' shorthand advantages sadly disappear. + * Any more child styles added in Tailwind will become longer than vanilla CSS. + * This limited example is the best case scenario for Tailwind. +* 🔴 Not visible on github: **no highlighting for properties and units** begins to be painful. +```html + + + + + + + + + +
+ + + + + + +
+ + + + + + + + +``` +## 🔎 Technical FAQ +* Why do you use `querySelectorAll()` and not just process the `MutationObserver` results directly? + * This was indeed the original design; it will work well up until you begin recieving subtrees (ex: DOM swaps with [htmx](https://htmx.org), ajax, jquery, etc.) which requires walking all subtree elements to ensure we do not miss a ` + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Custom Components

+
+ + + +
+ + + + +
+ + + +
+ + + +

The majority of the time the default ft components are all you need (for example Div, P, H1, etc.).

+
+
+
+ +
+
+Pre-requisite Knowledge +
+
+
+

If you don’t know what an ft component is, you should read the explaining ft components explainer first.

+
+
+

However, there are many situations where you need a custom ft component that creates a unique HTML tag (for example <zero-md></zero-md>). There are many options in FastHTML to do this, and this section will walk through them. Generally you want to use the highest level option that fits your needs.

+
+
+
+ +
+
+Real-world example +
+
+
+

This external tutorial walks through a practical situation where you may want to create a custom HTML tag using a custom ft component. Seeing a real-world example is a good way to understand why the contents of this guide is useful.

+
+
+
+

NotStr

+

The first way is to use the NotStr class to use an HTML tag as a string. It works as a one-off but quickly becomes harder to work with as complexity grows. However we can see that you can genenrate the same xml using NotStr as the out-of-the-box components.

+
+
from fasthtml.common import NotStr,Div, to_xml
+
+
+
div_NotStr = NotStr('<div></div>') 
+print(div_NotStr)
+
+
<div></div>
+
+
+
+
+

Automatic Creation

+

The next (and better) approach is to let FastHTML generate the component function for you. As you can see in our assert this creates a function that creates the HTML just as we wanted. This works even though there is not a Some_never_before_used_tag function in the fasthtml.components source code (you can verify this yourself by looking at the source code).

+
+
+
+ +
+
+Tip +
+
+
+

Typically these tags are needed because a CSS or Javascript library created a new XML tag that isn’t default HTML. For example the zero-md javascript library looks for a <zero-md></zero-md> tag to know what to run its javascript code on. Most CSS libraries work by creating styling based on the class attribute, but they can also apply styling to an arbitrary HTML tag that they made up.

+
+
+
+
from fasthtml.components import Some_never_before_used_tag
+
+Some_never_before_used_tag()
+
+
<some-never-before-used-tag></some-never-before-used-tag>
+
+
+
+
+

Manual Creation

+

The automatic creation isn’t magic. It’s just calling a python function __getattr__ and you can call it yourself to get the same result.

+
+
import fasthtml
+
+auto_called = fasthtml.components.Some_never_before_used_tag()
+manual_called = fasthtml.components.__getattr__('Some_never_before_used_tag')()
+
+# Proving they generate the same xml
+assert to_xml(auto_called) == to_xml(manual_called)
+
+

Knowing that, we know that it’s possible to create a different function that has different behavior than FastHTMLs default behavior by modifying how the ___getattr__ function creates the components! It’s only a few lines of code and reading that what it does is a great way to understand components more deeply.

+
+
+
+ +
+
+Tip +
+
+
+

Dunder methods and functions are special functions that have double underscores at the beginning and end of their name. They are called at specific times in python so you can use them to cause customized behavior that makes sense for your specific use case. They can appear magical if you don’t know how python works, but they are extremely commonly used to modify python’s default behavior (__init__ is probably the most common one).

+

In a module __getattr__ is called to get an attribute. In fasthtml.components, this is defined to create components automatically for you.

+
+
+

For example if you want a component that creates <path></path> that doesn’t conflict names with pathlib.Path you can do that. FastHTML automatically creates new components with a 1:1 mapping and a consistent name, which is almost always what you want. But in some cases you may want to customize that and you can use the ft_hx function to do that differently than the default.

+
+
from fasthtml.common import ft_hx
+
+def ft_path(*c, target_id=None, **kwargs): 
+    return ft_hx('path', *c, target_id=target_id, **kwargs)
+
+ft_path()
+
+
<path></path>
+
+
+

We can add any behavior in that function that we need to, so let’s go through some progressively complex examples that you may need in some of your projects.

+
+

Underscores in tags

+

Now that we understand how FastHTML generates components, we can create our own in all kinds of ways. For example, maybe we need a weird HTML tag that uses underscores. FastHTML replaces _ with - in tags because underscores in tags are highly unusual and rarely what you want, though it does come up rarely.

+
+
def tag_with_underscores(*c, target_id=None, **kwargs): 
+    return ft_hx('tag_with_underscores', *c, target_id=target_id, **kwargs)
+
+tag_with_underscores()
+
+
<tag_with_underscores></tag_with_underscores>
+
+
+
+
+

Symbols (ie @) in tags

+

Sometimes you may need to use a tag that uses characters that are not allowed in function names in python (again, very unusual).

+
+
def tag_with_AtSymbol(*c, target_id=None, **kwargs): 
+    return ft_hx('tag-with-@symbol', *c, target_id=target_id, **kwargs)
+
+tag_with_AtSymbol()
+
+
<tag-with-@symbol></tag-with-@symbol>
+
+
+
+
+

Symbols (ie @) in tag attributes

+

It also may be that an argument in an HTML tag uses characters that can’t be used in python arguments. To handle these you can define those args using a dictionary.

+
+
Div(normal_arg='normal stuff',**{'notNormal:arg:with_varing@symbols!':'123'})
+
+
<div normal-arg="normal stuff" notnormal:arg:with_varing@symbols!="123"></div>
+
+
+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/ref/defining_xt_component.md b/ref/defining_xt_component.md new file mode 100644 index 00000000..d26dbb57 --- /dev/null +++ b/ref/defining_xt_component.md @@ -0,0 +1,202 @@ +# Custom Components + + + + +The majority of the time the default [ft +components](../explains/explaining_xt_components.html) are all you need +(for example `Div`, `P`, `H1`, etc.). + +
+ +> **Pre-requisite Knowledge** +> +> If you don’t know what an ft component is, you should read [the +> explaining ft components explainer +> first](../explains/explaining_xt_components.html). + +
+ +However, there are many situations where you need a custom ft component +that creates a unique HTML tag (for example ``). +There are many options in FastHTML to do this, and this section will +walk through them. Generally you want to use the highest level option +that fits your needs. + +
+ +> **Real-world example** +> +> [This external +> tutorial](https://isaac-flath.github.io/website/posts/boots/FasthtmlTutorial.html) +> walks through a practical situation where you may want to create a +> custom HTML tag using a custom ft component. Seeing a real-world +> example is a good way to understand why the contents of this guide is +> useful. + +
+ +## NotStr + +The first way is to use the `NotStr` class to use an HTML tag as a +string. It works as a one-off but quickly becomes harder to work with as +complexity grows. However we can see that you can genenrate the same xml +using `NotStr` as the out-of-the-box components. + +``` python +from fasthtml.common import NotStr,Div, to_xml +``` + +``` python +div_NotStr = NotStr('
') +print(div_NotStr) +``` + +
+ +## Automatic Creation + +The next (and better) approach is to let FastHTML generate the component +function for you. As you can see in our `assert` this creates a function +that creates the HTML just as we wanted. This works even though there is +not a `Some_never_before_used_tag` function in the `fasthtml.components` +source code (you can verify this yourself by looking at the source +code). + +
+ +> **Tip** +> +> Typically these tags are needed because a CSS or Javascript library +> created a new XML tag that isn’t default HTML. For example the +> `zero-md` javascript library looks for a `` tag to +> know what to run its javascript code on. Most CSS libraries work by +> creating styling based on the `class` attribute, but they can also +> apply styling to an arbitrary HTML tag that they made up. + +
+ +``` python +from fasthtml.components import Some_never_before_used_tag + +Some_never_before_used_tag() +``` + +``` html + +``` + +## Manual Creation + +The automatic creation isn’t magic. It’s just calling a python function +[`__getattr__`](https://AnswerDotAI.github.io/fasthtml/api/components.html#__getattr__) +and you can call it yourself to get the same result. + +``` python +import fasthtml + +auto_called = fasthtml.components.Some_never_before_used_tag() +manual_called = fasthtml.components.__getattr__('Some_never_before_used_tag')() + +# Proving they generate the same xml +assert to_xml(auto_called) == to_xml(manual_called) +``` + +Knowing that, we know that it’s possible to create a different function +that has different behavior than FastHTMLs default behavior by modifying +how the `___getattr__` function creates the components! It’s only a few +lines of code and reading that what it does is a great way to understand +components more deeply. + +
+ +> **Tip** +> +> Dunder methods and functions are special functions that have double +> underscores at the beginning and end of their name. They are called at +> specific times in python so you can use them to cause customized +> behavior that makes sense for your specific use case. They can appear +> magical if you don’t know how python works, but they are extremely +> commonly used to modify python’s default behavior (`__init__` is +> probably the most common one). +> +> In a module +> [`__getattr__`](https://AnswerDotAI.github.io/fasthtml/api/components.html#__getattr__) +> is called to get an attribute. In `fasthtml.components`, this is +> defined to create components automatically for you. + +
+ +For example if you want a component that creates `` that +doesn’t conflict names with `pathlib.Path` you can do that. FastHTML +automatically creates new components with a 1:1 mapping and a consistent +name, which is almost always what you want. But in some cases you may +want to customize that and you can use the +[`ft_hx`](https://AnswerDotAI.github.io/fasthtml/api/components.html#ft_hx) +function to do that differently than the default. + +``` python +from fasthtml.common import ft_hx + +def ft_path(*c, target_id=None, **kwargs): + return ft_hx('path', *c, target_id=target_id, **kwargs) + +ft_path() +``` + +``` html + +``` + +We can add any behavior in that function that we need to, so let’s go +through some progressively complex examples that you may need in some of +your projects. + +### Underscores in tags + +Now that we understand how FastHTML generates components, we can create +our own in all kinds of ways. For example, maybe we need a weird HTML +tag that uses underscores. FastHTML replaces `_` with `-` in tags +because underscores in tags are highly unusual and rarely what you want, +though it does come up rarely. + +``` python +def tag_with_underscores(*c, target_id=None, **kwargs): + return ft_hx('tag_with_underscores', *c, target_id=target_id, **kwargs) + +tag_with_underscores() +``` + +``` html + +``` + +### Symbols (ie @) in tags + +Sometimes you may need to use a tag that uses characters that are not +allowed in function names in python (again, very unusual). + +``` python +def tag_with_AtSymbol(*c, target_id=None, **kwargs): + return ft_hx('tag-with-@symbol', *c, target_id=target_id, **kwargs) + +tag_with_AtSymbol() +``` + +``` html + +``` + +### Symbols (ie @) in tag attributes + +It also may be that an argument in an HTML tag uses characters that +can’t be used in python arguments. To handle these you can define those +args using a dictionary. + +``` python +Div(normal_arg='normal stuff',**{'notNormal:arg:with_varing@symbols!':'123'}) +``` + +``` html +
+``` diff --git a/ref/handlers.html b/ref/handlers.html new file mode 100644 index 00000000..eb2d407a --- /dev/null +++ b/ref/handlers.html @@ -0,0 +1,1653 @@ + + + + + + + + + + +Handling handlers – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Handling handlers

+
+ +
+
+ How handlers work in FastHTML +
+
+ + +
+ + + + +
+ + + +
+ + + +
+
from fasthtml.common import *
+from collections import namedtuple
+from typing import TypedDict
+from datetime import datetime
+import json,time
+
+
+
app = FastHTML()
+
+

The FastHTML class is the main application class for FastHTML apps.

+
+
rt = app.route
+
+

app.route is used to register route handlers. It is a decorator, which means we place it before a function that is used as a handler. Because it’s used frequently in most FastHTML applications, we often alias it as rt, as we do here.

+
+

Basic Route Handling

+
+
@rt("/hi")
+def get(): return 'Hi there'
+
+

Handler functions can return strings directly. These strings are sent as the response body to the client.

+
+
cli = Client(app)
+
+

Client is a test client for FastHTML applications. It allows you to simulate requests to your app without running a server.

+
+
cli.get('/hi').text
+
+
'Hi there'
+
+
+

The get method on a Client instance simulates GET requests to the app. It returns a response object that has a .text attribute, which you can use to access the body of the response. It calls httpx.get internally – all httpx HTTP verbs are supported.

+
+
@rt("/hi")
+def post(): return 'Postal'
+cli.post('/hi').text
+
+
'Postal'
+
+
+

Handler functions can be defined for different HTTP methods on the same route. Here, we define a post handler for the /hi route. The Client instance can simulate different HTTP methods, including POST requests.

+
+
+

Request and Response Objects

+
+
@app.get("/hostie")
+def show_host(req): return req.headers['host']
+cli.get('/hostie').text
+
+
'testserver'
+
+
+

Handler functions can accept a req (or request) parameter, which represents the incoming request. This object contains information about the request, including headers. In this example, we return the host header from the request. The test client uses ‘testserver’ as the default host.

+

In this example, we use @app.get("/hostie") instead of @rt("/hostie"). The @app.get() decorator explicitly specifies the HTTP method (GET) for the route, while @rt() by default handles both GET and POST requests.

+
+
@rt
+def yoyo(): return 'a yoyo'
+cli.post('/yoyo').text
+
+
'a yoyo'
+
+
+

If the @rt decorator is used without arguments, it uses the function name as the route path. Here, the yoyo function becomes the handler for the /yoyo route. This handler responds to GET and POST methods, since a specific method wasn’t provided.

+
+
@rt
+def ft1(): return Html(Div('Text.'))
+print(cli.get('/ft1').text)
+
+
 <html>
+   <div>Text.</div>
+ </html>
+
+
+
+

Handler functions can return FT objects, which are automatically converted to HTML strings. The FT class can take other FT components as arguments, such as Div. This allows for easy composition of HTML elements in your responses.

+
+
@app.get
+def autopost(): return Html(Div('Text.', hx_post=yoyo.to()))
+print(cli.get('/autopost').text)
+
+
 <html>
+   <div hx-post="/yoyo">Text.</div>
+ </html>
+
+
+
+

The rt decorator modifies the yoyo function by adding an rt() method. This method returns the route path associated with the handler. It’s a convenient way to reference the route of a handler function dynamically.

+

In the example, yoyo.to() is used as the value for hx_post. This means when the div is clicked, it will trigger an HTMX POST request to the route of the yoyo handler. This approach allows for flexible, DRY code by avoiding hardcoded route strings and automatically updating if the route changes.

+

This pattern is particularly useful in larger applications where routes might change, or when building reusable components that need to reference their own routes dynamically.

+
+
@app.get
+def autoget(): return Html(Body(Div('Text.', cls='px-2', hx_post=show_host.to(a='b'))))
+print(cli.get('/autoget').text)
+
+
 <html>
+   <body>
+     <div hx-post="/hostie?a=b" class="px-2">Text.</div>
+   </body>
+ </html>
+
+
+
+

The rt() method of handler functions can also accept parameters. When called with parameters, it returns the route path with a query string appended. In this example, show_host.to(a='b') generates the path /hostie?a=b.

+

The Body component is used here to demonstrate nesting of FT components. Div is nested inside Body, showcasing how you can create more complex HTML structures.

+

The cls parameter is used to add a CSS class to the Div. This translates to the class attribute in the rendered HTML. (class can’t be used as a parameter name directly in Python since it’s a reserved word.)

+
+
@rt('/ft2')
+def get(): return Title('Foo'),H1('bar')
+print(cli.get('/ft2').text)
+
+
 <!doctype html>
+ <html>
+   <head>
+     <title>Foo</title>
+     <meta charset="utf-8">
+     <meta name="viewport" content="width=device-width, initial-scale=1, viewport-fit=cover">
+<script src="https://unpkg.com/htmx.org@next/dist/htmx.min.js"></script><script src="https://cdn.jsdelivr.net/gh/answerdotai/fasthtml-js@1.0.4/fasthtml.js"></script><script src="https://cdn.jsdelivr.net/gh/answerdotai/surreal@main/surreal.js"></script><script src="https://cdn.jsdelivr.net/gh/gnat/css-scope-inline@main/script.js"></script><script>
+    function sendmsg() {
+        window.parent.postMessage({height: document.documentElement.offsetHeight}, '*');
+    }
+    window.onload = function() {
+        sendmsg();
+        document.body.addEventListener('htmx:afterSettle',    sendmsg);
+        document.body.addEventListener('htmx:wsAfterMessage', sendmsg);
+    };</script>   </head>
+   <body>
+     <h1>bar</h1>
+   </body>
+ </html>
+
+
+
+

Handler functions can return multiple FT objects as a tuple. The first item is treated as the Title, and the rest are added to the Body. When the request is not an HTMX request, FastHTML automatically adds necessary HTML boilerplate, including default head content with required scripts.

+

When using app.route (or rt), if the function name matches an HTTP verb (e.g., get, post, put, delete), that HTTP method is automatically used for the route. In this case, a path must be explicitly provided as an argument to the decorator.

+
+
hxhdr = {'headers':{'hx-request':"1"}}
+print(cli.get('/ft2', **hxhdr).text)
+
+
 <title>Foo</title>
+ <h1>bar</h1>
+
+
+
+

For HTMX requests (indicated by the hx-request header), FastHTML returns only the specified components without the full HTML structure. This allows for efficient partial page updates in HTMX applications.

+
+
@rt('/ft3')
+def get(): return H1('bar')
+print(cli.get('/ft3', **hxhdr).text)
+
+
 <h1>bar</h1>
+
+
+
+

When a handler function returns a single FT object for an HTMX request, it’s rendered as a single HTML partial.

+
+
@rt('/ft4')
+def get(): return Html(Head(Title('hi')), Body(P('there')))
+
+print(cli.get('/ft4').text)
+
+
 <html>
+   <head>
+     <title>hi</title>
+   </head>
+   <body>
+     <p>there</p>
+   </body>
+ </html>
+
+
+
+

Handler functions can return a complete Html structure, including Head and Body components. When a full HTML structure is returned, FastHTML doesn’t add any additional boilerplate. This gives you full control over the HTML output when needed.

+
+
@rt
+def index(): return "welcome!"
+print(cli.get('/').text)
+
+
welcome!
+
+
+

The index function is a special handler in FastHTML. When defined without arguments to the @rt decorator, it automatically becomes the handler for the root path ('/'). This is a convenient way to define the main page or entry point of your application.

+
+
+

Path and Query Parameters

+
+
@rt('/user/{nm}', name='gday')
+def get(nm:str=''): return f"Good day to you, {nm}!"
+cli.get('/user/Alexis').text
+
+
'Good day to you, Alexis!'
+
+
+

Handler functions can use path parameters, defined using curly braces in the route – this is implemented by Starlette directly, so all Starlette path parameters can be used. These parameters are passed as arguments to the function.

+

The name parameter in the decorator allows you to give the route a name, which can be used for URL generation.

+

In this example, {nm} in the route becomes the nm parameter in the function. The function uses this parameter to create a personalized greeting.

+
+
@app.get
+def autolink(): return Html(Div('Text.', link=uri('gday', nm='Alexis')))
+print(cli.get('/autolink').text)
+
+
 <html>
+   <div href="/user/Alexis">Text.</div>
+ </html>
+
+
+
+

The uri function is used to generate URLs for named routes. It takes the route name as its first argument, followed by any path or query parameters needed for that route.

+

In this example, uri('gday', nm='Alexis') generates the URL for the route named ‘gday’ (which we defined earlier as ‘/user/{nm}’), with ‘Alexis’ as the value for the ‘nm’ parameter.

+

The link parameter in FT components sets the href attribute of the rendered HTML element. By using uri(), we can dynamically generate correct URLs even if the underlying route structure changes.

+

This approach promotes maintainable code by centralizing route definitions and avoiding hardcoded URLs throughout the application.

+
+
@rt('/link')
+def get(req): return f"{req.url_for('gday', nm='Alexis')}; {req.url_for('show_host')}"
+
+cli.get('/link').text
+
+
'http://testserver/user/Alexis; http://testserver/hostie'
+
+
+

The url_for method of the request object can be used to generate URLs for named routes. It takes the route name as its first argument, followed by any path parameters needed for that route.

+

In this example, req.url_for('gday', nm='Alexis') generates the full URL for the route named ‘gday’, including the scheme and host. Similarly, req.url_for('show_host') generates the URL for the ‘show_host’ route.

+

This method is particularly useful when you need to generate absolute URLs, such as for email links or API responses. It ensures that the correct host and scheme are included, even if the application is accessed through different domains or protocols.

+
+
app.url_path_for('gday', nm='Jeremy')
+
+
'/user/Jeremy'
+
+
+

The url_path_for method of the application can be used to generate URL paths for named routes. Unlike url_for, it returns only the path component of the URL, without the scheme or host.

+

In this example, app.url_path_for('gday', nm='Jeremy') generates the path ‘/user/Jeremy’ for the route named ‘gday’.

+

This method is useful when you need relative URLs or just the path component, such as for internal links or when constructing URLs in a host-agnostic manner.

+
+
@rt('/oops')
+def get(nope): return nope
+r = cli.get('/oops?nope=1')
+print(r)
+r.text
+
+
<Response [200 OK]>
+
+
+
/Users/wgilliam/development/projects/aai/fasthtml/fasthtml/core.py:188: UserWarning: `nope has no type annotation and is not a recognised special name, so is ignored.
+  if arg!='resp': warn(f"`{arg} has no type annotation and is not a recognised special name, so is ignored.")
+
+
+
''
+
+
+

Handler functions can include parameters, but they must be type-annotated or have special names (like req) to be recognized. In this example, the nope parameter is not annotated, so it’s ignored, resulting in a warning.

+

When a parameter is ignored, it doesn’t receive the value from the query string. This can lead to unexpected behavior, as the function attempts to return nope, which is undefined.

+

The cli.get('/oops?nope=1') call succeeds with a 200 OK status because the handler doesn’t raise an exception, but it returns an empty response, rather than the intended value.

+

To fix this, you should either add a type annotation to the parameter (e.g., def get(nope: str):) or use a recognized special name like req.

+
+
@rt('/html/{idx}')
+def get(idx:int): return Body(H4(f'Next is {idx+1}.'))
+print(cli.get('/html/1', **hxhdr).text)
+
+
 <body>
+   <h4>Next is 2.</h4>
+ </body>
+
+
+
+

Path parameters can be type-annotated, and FastHTML will automatically convert them to the specified type if possible. In this example, idx is annotated as int, so it’s converted from the string in the URL to an integer.

+
+
reg_re_param("imgext", "ico|gif|jpg|jpeg|webm")
+
+@rt(r'/static/{path:path}{fn}.{ext:imgext}')
+def get(fn:str, path:str, ext:str): return f"Getting {fn}.{ext} from /{path}"
+
+print(cli.get('/static/foo/jph.ico').text)
+
+
Getting jph.ico from /foo/
+
+
+

The reg_re_param function is used to register custom path parameter types using regular expressions. Here, we define a new path parameter type called “imgext” that matches common image file extensions.

+

Handler functions can use complex path patterns with multiple parameters and custom types. In this example, the route pattern r'/static/{path:path}{fn}.{ext:imgext}' uses three path parameters:

+
    +
  1. path: A Starlette built-in type that matches any path segments
  2. +
  3. fn: The filename without extension
  4. +
  5. ext: Our custom “imgext” type that matches specific image extensions
  6. +
+
+
ModelName = str_enum('ModelName', "alexnet", "resnet", "lenet")
+
+@rt("/models/{nm}")
+def get(nm:ModelName): return nm
+
+print(cli.get('/models/alexnet').text)
+
+
alexnet
+
+
+

We define ModelName as an enum with three possible values: “alexnet”, “resnet”, and “lenet”. Handler functions can use these enum types as parameter annotations. In this example, the nm parameter is annotated with ModelName, which ensures that only valid model names are accepted.

+

When a request is made with a valid model name, the handler function returns that name. This pattern is useful for creating type-safe APIs with a predefined set of valid values.

+
+
@rt("/files/{path}")
+async def get(path: Path): return path.with_suffix('.txt')
+print(cli.get('/files/foo').text)
+
+
foo.txt
+
+
+

Handler functions can use Path objects as parameter types. The Path type is from Python’s standard library pathlib module, which provides an object-oriented interface for working with file paths. In this example, the path parameter is annotated with Path, so FastHTML automatically converts the string from the URL to a Path object.

+

This approach is particularly useful when working with file-related routes, as it provides a convenient and platform-independent way to handle file paths.

+
+
fake_db = [{"name": "Foo"}, {"name": "Bar"}]
+
+@rt("/items/")
+def get(idx:int|None = 0): return fake_db[idx]
+print(cli.get('/items/?idx=1').text)
+
+
{"name":"Bar"}
+
+
+

Handler functions can use query parameters, which are automatically parsed from the URL. In this example, idx is a query parameter with a default value of 0. It’s annotated as int|None, allowing it to be either an integer or None.

+

The function uses this parameter to index into a fake database (fake_db). When a request is made with a valid idx query parameter, the handler returns the corresponding item from the database.

+
+
print(cli.get('/items/').text)
+
+
{"name":"Foo"}
+
+
+

When no idx query parameter is provided, the handler function uses the default value of 0. This results in returning the first item from the fake_db list, which is {"name":"Foo"}.

+

This behavior demonstrates how default values for query parameters work in FastHTML. They allow the API to have a sensible default behavior when optional parameters are not provided.

+
+
print(cli.get('/items/?idx=g'))
+
+
<Response [404 Not Found]>
+
+
+

When an invalid value is provided for a typed query parameter, FastHTML returns a 404 Not Found response. In this example, ‘g’ is not a valid integer for the idx parameter, so the request fails with a 404 status.

+

This behavior ensures type safety and prevents invalid inputs from reaching the handler function.

+
+
@app.get("/booly/")
+def _(coming:bool=True): return 'Coming' if coming else 'Not coming'
+print(cli.get('/booly/?coming=true').text)
+print(cli.get('/booly/?coming=no').text)
+
+
Coming
+Not coming
+
+
+

Handler functions can use boolean query parameters. In this example, coming is a boolean parameter with a default value of True. FastHTML automatically converts string values like ‘true’, ‘false’, ‘1’, ‘0’, ‘on’, ‘off’, ‘yes’, and ‘no’ to their corresponding boolean values.

+

The underscore _ is used as the function name in this example to indicate that the function’s name is not important or won’t be referenced elsewhere. This is a common Python convention for throwaway or unused variables, and it works here because FastHTML uses the route decorator parameter, when provided, to determine the URL path, not the function name. By default, both get and post methods can be used in routes that don’t specify an http method (by either using app.get, def get, or the methods parameter to app.route).

+
+
@app.get("/datie/")
+def _(d:parsed_date): return d
+date_str = "17th of May, 2024, 2p"
+print(cli.get(f'/datie/?d={date_str}').text)
+
+
2024-05-17 14:00:00
+
+
+

Handler functions can use date objects as parameter types. FastHTML uses dateutil.parser library to automatically parse a wide variety of date string formats into date objects.

+
+
@app.get("/ua")
+async def _(user_agent:str): return user_agent
+print(cli.get('/ua', headers={'User-Agent':'FastHTML'}).text)
+
+
FastHTML
+
+
+

Handler functions can access HTTP headers by using parameter names that match the header names. In this example, user_agent is used as a parameter name, which automatically captures the value of the ‘User-Agent’ header from the request.

+

The Client instance allows setting custom headers for test requests. Here, we set the ‘User-Agent’ header to ‘FastHTML’ in the test request.

+
+
@app.get("/hxtest")
+def _(htmx): return htmx.request
+print(cli.get('/hxtest', headers={'HX-Request':'1'}).text)
+
+@app.get("/hxtest2")
+def _(foo:HtmxHeaders, req): return foo.request
+print(cli.get('/hxtest2', headers={'HX-Request':'1'}).text)
+
+
1
+1
+
+
+

Handler functions can access HTMX-specific headers using either the special htmx parameter name, or a parameter annotated with HtmxHeaders. Both approaches provide access to HTMX-related information.

+

In these examples, the htmx.request attribute returns the value of the ‘HX-Request’ header.

+
+
app.chk = 'foo'
+@app.get("/app")
+def _(app): return app.chk
+print(cli.get('/app').text)
+
+
foo
+
+
+

Handler functions can access the FastHTML application instance using the special app parameter name. This allows handlers to access application-level attributes and methods.

+

In this example, we set a custom attribute chk on the application instance. The handler function then uses the app parameter to access this attribute and return its value.

+
+
@app.get("/app2")
+def _(foo:FastHTML): return foo.chk,HttpHeader("mykey", "myval")
+r = cli.get('/app2', **hxhdr)
+print(r.text)
+print(r.headers)
+
+
foo
+Headers({'mykey': 'myval', 'content-length': '3', 'content-type': 'text/html; charset=utf-8'})
+
+
+

Handler functions can access the FastHTML application instance using a parameter annotated with FastHTML. This allows handlers to access application-level attributes and methods, just like using the special app parameter name.

+

Handlers can return tuples containing both content and HttpHeader objects. HttpHeader allows setting custom HTTP headers in the response.

+

In this example:

+
    +
  • We define a handler that returns both the chk attribute from the application and a custom header.
  • +
  • The HttpHeader("mykey", "myval") sets a custom header in the response.
  • +
  • We use the test client to make a request and examine both the response text and headers.
  • +
  • The response includes the custom header “mykey” along with standard headers like content-length and content-type.
  • +
+
+
@app.get("/app3")
+def _(foo:FastHTML): return HtmxResponseHeaders(location="http://example.org")
+r = cli.get('/app3')
+print(r.headers)
+
+
Headers({'hx-location': 'http://example.org', 'content-length': '0', 'content-type': 'text/html; charset=utf-8'})
+
+
+

Handler functions can return HtmxResponseHeaders objects to set HTMX-specific response headers. This is useful for HTMX-specific behaviors like client-side redirects.

+

In this example we define a handler that returns an HtmxResponseHeaders object with a location parameter, which sets the HX-Location header in the response. HTMX uses this for client-side redirects.

+
+
@app.get("/app4")
+def _(foo:FastHTML): return Redirect("http://example.org")
+cli.get('/app4', follow_redirects=False)
+
+
<Response [303 See Other]>
+
+
+

Handler functions can return Redirect objects to perform HTTP redirects. This is useful for redirecting users to different pages or external URLs.

+

In this example:

+
    +
  • We define a handler that returns a Redirect object with the URL “http://example.org”.
  • +
  • The cli.get('/app4', follow_redirects=False) call simulates a GET request to the ‘/app4’ route without following redirects.
  • +
  • The response has a 303 See Other status code, indicating a redirect.
  • +
+

The follow_redirects=False parameter is used to prevent the test client from automatically following the redirect, allowing us to inspect the redirect response itself.

+
+
Redirect.__response__
+
+
<function fasthtml.core.Redirect.__response__(self, req)>
+
+
+

The Redirect class in FastHTML implements a __response__ method, which is a special method recognized by the framework. When a handler returns a Redirect object, FastHTML internally calls this __response__ method to replace the original response.

+

The __response__ method takes a req parameter, which represents the incoming request. This allows the method to access request information if needed when constructing the redirect response.

+
+
@rt
+def meta(): 
+    return ((Title('hi'),H1('hi')),
+        (Meta(property='image'), Meta(property='site_name')))
+
+print(cli.post('/meta').text)
+
+
 <!doctype html>
+ <html>
+   <head>
+     <title>hi</title>
+     <meta property="image">
+     <meta property="site_name">
+     <meta charset="utf-8">
+     <meta name="viewport" content="width=device-width, initial-scale=1, viewport-fit=cover">
+<script src="https://unpkg.com/htmx.org@next/dist/htmx.min.js"></script><script src="https://cdn.jsdelivr.net/gh/answerdotai/fasthtml-js@1.0.4/fasthtml.js"></script><script src="https://cdn.jsdelivr.net/gh/answerdotai/surreal@main/surreal.js"></script><script src="https://cdn.jsdelivr.net/gh/gnat/css-scope-inline@main/script.js"></script><script>
+    function sendmsg() {
+        window.parent.postMessage({height: document.documentElement.offsetHeight}, '*');
+    }
+    window.onload = function() {
+        sendmsg();
+        document.body.addEventListener('htmx:afterSettle',    sendmsg);
+        document.body.addEventListener('htmx:wsAfterMessage', sendmsg);
+    };</script>   </head>
+   <body>
+     <h1>hi</h1>
+   </body>
+ </html>
+
+
+
+

FastHTML automatically identifies elements typically placed in the <head> (like Title and Meta) and positions them accordingly, while other elements go in the <body>.

+

In this example: - (Title('hi'), H1('hi')) defines the title and main heading. The title is placed in the head, and the H1 in the body. - (Meta(property='image'), Meta(property='site_name')) defines two meta tags, which are both placed in the head.

+
+
+

APIRouter

+

APIRouter is useful when you want to split your application routes across multiple .py files that are part of a single FastHTMl application. It accepts an optional prefix argument that will be applied to all routes within that instance of APIRouter.

+

Below we define several hypothetical product related routes in a products.py and then demonstrate how they can seamlessly be incorporated into a FastHTML app instance.

+
+
# products.py
+ar = APIRouter(prefix="/products")
+
+@ar("/all")
+def all_products(req):
+    return Div(
+        "Welcome to the Products Page! Click the button below to look at the details for product 42",
+        Div(
+            Button(
+                "Details",
+                hx_get=req.url_for("details", pid=42),
+                hx_target="#products_list",
+                hx_swap="outerHTML",
+            ),
+        ),
+        id="products_list",
+    )
+
+
+@ar.get("/{pid}", name="details")
+def details(pid: int):
+    return f"Here are the product details for ID: {pid}"
+
+

Since we specified the prefix=/products in our hypothetical products.py file, all routes defined in that file will be found under /products.

+
+
print(str(ar.rt_funcs.all_products))
+print(str(ar.rt_funcs.details))
+
+
/products/all
+/products/{pid}
+
+
+
+
# main.py
+# from products import ar
+
+app, rt = fast_app()
+ar.to_app(app)
+
+@rt
+def index():
+    return Div(
+        "Click me for a look at our products",
+        hx_get=ar.rt_funcs.all_products,
+        hx_swap="outerHTML",
+    )
+
+

Note how you can reference our python route functions via APIRouter.rt_funcs in your hx_{http_method} calls like normal.

+
+
+

Form Data and JSON Handling

+
+
@app.post('/profile/me')
+def profile_update(username: str): return username
+
+print(cli.post('/profile/me', data={'username' : 'Alexis'}).text)
+r = cli.post('/profile/me', data={})
+print(r.text)
+r
+
+
404 Not Found
+404 Not Found
+
+
+
<Response [404 Not Found]>
+
+
+

Handler functions can accept form data parameters, without needing to manually extract it from the request. In this example, username is expected to be sent as form data.

+

If required form data is missing, FastHTML automatically returns a 400 Bad Request response with an error message.

+

The data parameter in the cli.post() method simulates sending form data in the request.

+
+
@app.post('/pet/dog')
+def pet_dog(dogname: str = None): return dogname or 'unknown name'
+print(cli.post('/pet/dog', data={}).text)
+
+
404 Not Found
+
+
+

Handlers can have optional form data parameters with default values. In this example, dogname is an optional parameter with a default value of None.

+

Here, if the form data doesn’t include the dogname field, the function uses the default value. The function returns either the provided dogname or ‘unknown name’ if dogname is None.

+
+
@dataclass
+class Bodie: a:int;b:str
+
+@rt("/bodie/{nm}")
+def post(nm:str, data:Bodie):
+    res = asdict(data)
+    res['nm'] = nm
+    return res
+
+print(cli.post('/bodie/me', data=dict(a=1, b='foo', nm='me')).text)
+
+
404 Not Found
+
+
+

You can use dataclasses to define structured form data. In this example, Bodie is a dataclass with a (int) and b (str) fields.

+

FastHTML automatically converts the incoming form data to a Bodie instance where attribute names match parameter names. Other form data elements are matched with parameters with the same names (in this case, nm).

+

Handler functions can return dictionaries, which FastHTML automatically JSON-encodes.

+
+
@app.post("/bodied/")
+def bodied(data:dict): return data
+
+d = dict(a=1, b='foo')
+print(cli.post('/bodied/', data=d).text)
+
+
404 Not Found
+
+
+

dict parameters capture all form data as a dictionary. In this example, the data parameter is annotated with dict, so FastHTML automatically converts all incoming form data into a dictionary.

+

Note that when form data is converted to a dictionary, all values become strings, even if they were originally numbers. This is why the ‘a’ key in the response has a string value “1” instead of the integer 1.

+
+
nt = namedtuple('Bodient', ['a','b'])
+
+@app.post("/bodient/")
+def bodient(data:nt): return asdict(data)
+print(cli.post('/bodient/', data=d).text)
+
+
404 Not Found
+
+
+

Handler functions can use named tuples to define structured form data. In this example, Bodient is a named tuple with a and b fields.

+

FastHTML automatically converts the incoming form data to a Bodient instance where field names match parameter names. As with the previous example, all form data values are converted to strings in the process.

+
+
class BodieTD(TypedDict): a:int;b:str='foo'
+
+@app.post("/bodietd/")
+def bodient(data:BodieTD): return data
+print(cli.post('/bodietd/', data=d).text)
+
+
404 Not Found
+
+
+

You can use TypedDict to define structured form data with type hints. In this example, BodieTD is a TypedDict with a (int) and b (str) fields, where b has a default value of ‘foo’.

+

FastHTML automatically converts the incoming form data to a BodieTD instance where keys match the defined fields. Unlike with regular dictionaries or named tuples, FastHTML respects the type hints in TypedDict, converting values to the specified types when possible (e.g., converting ‘1’ to the integer 1 for the ‘a’ field).

+
+
class Bodie2:
+    a:int|None; b:str
+    def __init__(self, a, b='foo'): store_attr()
+
+@app.post("/bodie2/")
+def bodie(d:Bodie2): return f"a: {d.a}; b: {d.b}"
+print(cli.post('/bodie2/', data={'a':1}).text)
+
+
404 Not Found
+
+
+

Custom classes can be used to define structured form data. Here, Bodie2 is a custom class with a (int|None) and b (str) attributes, where b has a default value of ‘foo’. The store_attr() function (from fastcore) automatically assigns constructor parameters to instance attributes.

+

FastHTML automatically converts the incoming form data to a Bodie2 instance, matching form fields to constructor parameters. It respects type hints and default values.

+
+
@app.post("/b")
+def index(it: Bodie): return Titled("It worked!", P(f"{it.a}, {it.b}"))
+
+s = json.dumps({"b": "Lorem", "a": 15})
+print(cli.post('/b', headers={"Content-Type": "application/json", 'hx-request':"1"}, data=s).text)
+
+
404 Not Found
+
+
+

Handler functions can accept JSON data as input, which is automatically parsed into the specified type. In this example, it is of type Bodie, and FastHTML converts the incoming JSON data to a Bodie instance.

+

The Titled component is used to create a page with a title and main content. It automatically generates an <h1> with the provided title, wraps the content in a <main> tag with a “container” class, and adds a title to the head.

+

When making a request with JSON data: - Set the “Content-Type” header to “application/json” - Provide the JSON data as a string in the data parameter of the request

+
+
+

Cookies, Sessions, File Uploads, and more

+
+
@rt("/setcookie")
+def get(): return cookie('now', datetime.now())
+
+@rt("/getcookie")
+def get(now:parsed_date): return f'Cookie was set at time {now.time()}'
+
+print(cli.get('/setcookie').text)
+time.sleep(0.01)
+cli.get('/getcookie').text
+
+
404 Not Found
+
+
+
'404 Not Found'
+
+
+

Handler functions can set and retrieve cookies. In this example:

+
    +
  • The /setcookie route sets a cookie named ‘now’ with the current datetime.
  • +
  • The /getcookie route retrieves the ‘now’ cookie and returns its value.
  • +
+

The cookie() function is used to create a cookie response. FastHTML automatically converts the datetime object to a string when setting the cookie, and parses it back to a date object when retrieving it.

+
+
cookie('now', datetime.now())
+
+
HttpHeader(k='set-cookie', v='now="2024-12-04 13:45:24.154187"; Path=/; SameSite=lax')
+
+
+

The cookie() function returns an HttpHeader object with the ‘set-cookie’ key. You can return it in a tuple along with FT elements, along with anything else FastHTML supports in responses.

+
+
app = FastHTML(secret_key='soopersecret')
+cli = Client(app)
+rt = app.route
+
+
+
@rt("/setsess")
+def get(sess, foo:str=''):
+    now = datetime.now()
+    sess['auth'] = str(now)
+    return f'Set to {now}'
+
+@rt("/getsess")
+def get(sess): return f'Session time: {sess["auth"]}'
+
+print(cli.get('/setsess').text)
+time.sleep(0.01)
+
+cli.get('/getsess').text
+
+
Set to 2024-12-04 13:45:24.159764
+
+
+
'Session time: 2024-12-04 13:45:24.159764'
+
+
+

Sessions store and retrieve data across requests. To use sessions, you should to initialize the FastHTML application with a secret_key. This is used to cryptographically sign the cookie used by the session.

+

The sess parameter in handler functions provides access to the session data. You can set and get session variables using dictionary-style access.

+
+
@rt("/upload")
+async def post(uf:UploadFile): return (await uf.read()).decode()
+
+with open('../../CHANGELOG.md', 'rb') as f:
+    print(cli.post('/upload', files={'uf':f}, data={'msg':'Hello'}).text[:15])
+
+
# Release notes
+
+
+

Handler functions can accept file uploads using Starlette’s UploadFile type. In this example:

+
    +
  • The /upload route accepts a file upload named uf.
  • +
  • The UploadFile object provides an asynchronous read() method to access the file contents.
  • +
  • We use await to read the file content asynchronously and decode it to a string.
  • +
+

We added async to the handler function because it uses await to read the file content asynchronously. In Python, any function that uses await must be declared as async. This allows the function to be run asynchronously, potentially improving performance by not blocking other operations while waiting for the file to be read.

+
+
app.static_route('.md', static_path='../..')
+print(cli.get('/README.md').text[:10])
+
+
# FastHTML
+
+
+

The static_route method of the FastHTML application allows serving static files with specified extensions from a given directory. In this example:

+
    +
  • .md files are served from the ../.. directory (two levels up from the current directory).
  • +
  • Accessing /README.md returns the contents of the README.md file from that directory.
  • +
+
+
help(app.static_route_exts)
+
+
Help on method static_route_exts in module fasthtml.core:
+
+static_route_exts(prefix='/', static_path='.', exts='static') method of fasthtml.core.FastHTML instance
+    Add a static route at URL path `prefix` with files from `static_path` and `exts` defined by `reg_re_param()`
+
+
+
+
+
app.static_route_exts()
+print(cli.get('/README.txt').text[:50])
+
+
404 Not Found
+
+
+

The static_route_exts method of the FastHTML application allows serving static files with specified extensions from a given directory. By default:

+
    +
  • It serves files from the current directory (‘.’).
  • +
  • It uses the ‘static’ regex, which includes common static file extensions like ‘ico’, ‘gif’, ‘jpg’, ‘css’, ‘js’, etc.
  • +
  • The URL prefix is set to ‘/’.
  • +
+

The ‘static’ regex is defined by FastHTML using this code:

+
reg_re_param("static", "ico|gif|jpg|jpeg|webm|css|js|woff|png|svg|mp4|webp|ttf|otf|eot|woff2|txt|html|map")
+
+
@rt("/form-submit/{list_id}")
+def options(list_id: str):
+    headers = {
+        'Access-Control-Allow-Origin': '*',
+        'Access-Control-Allow-Methods': 'POST',
+        'Access-Control-Allow-Headers': '*',
+    }
+    return Response(status_code=200, headers=headers)
+
+print(cli.options('/form-submit/2').headers)
+
+
Headers({'access-control-allow-origin': '*', 'access-control-allow-methods': 'POST', 'access-control-allow-headers': '*', 'content-length': '0', 'set-cookie': 'session_=eyJhdXRoIjogIjIwMjQtMTItMDQgMTM6NDU6MjQuMTU5NzY0In0=.Z1DNdA.GV-NVoOnJeambm9_uE3crhoGH34; path=/; Max-Age=31536000; httponly; samesite=lax'})
+
+
+

FastHTML handlers can handle OPTIONS requests and set custom headers. In this example:

+
    +
  • The /form-submit/{list_id} route handles OPTIONS requests.
  • +
  • Custom headers are set to allow cross-origin requests (CORS).
  • +
  • The function returns a Starlette Response object with a 200 status code and the custom headers.
  • +
+

You can return any Starlette Response type from a handler function, giving you full control over the response when needed.

+
+
def _not_found(req, exc): return Div('nope')
+
+app = FastHTML(exception_handlers={404:_not_found})
+cli = Client(app)
+rt = app.route
+
+r = cli.get('/')
+print(r.text)
+
+
 <!doctype html>
+ <html>
+   <head>
+     <title>FastHTML page</title>
+     <meta charset="utf-8">
+     <meta name="viewport" content="width=device-width, initial-scale=1, viewport-fit=cover">
+<script src="https://unpkg.com/htmx.org@next/dist/htmx.min.js"></script><script src="https://cdn.jsdelivr.net/gh/answerdotai/fasthtml-js@1.0.4/fasthtml.js"></script><script src="https://cdn.jsdelivr.net/gh/answerdotai/surreal@main/surreal.js"></script><script src="https://cdn.jsdelivr.net/gh/gnat/css-scope-inline@main/script.js"></script><script>
+    function sendmsg() {
+        window.parent.postMessage({height: document.documentElement.offsetHeight}, '*');
+    }
+    window.onload = function() {
+        sendmsg();
+        document.body.addEventListener('htmx:afterSettle',    sendmsg);
+        document.body.addEventListener('htmx:wsAfterMessage', sendmsg);
+    };</script>   </head>
+   <body>
+     <div>nope</div>
+   </body>
+ </html>
+
+
+
+

FastHTML allows you to define custom exception handlers – in this case, a custom 404 (Not Found) handler function _not_found, which returns a Div component with the text ‘nope’.

+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/ref/handlers.html.md b/ref/handlers.html.md new file mode 100644 index 00000000..ce7645e0 --- /dev/null +++ b/ref/handlers.html.md @@ -0,0 +1,1207 @@ +# Handling handlers + + + + +``` python +from fasthtml.common import * +from collections import namedtuple +from typing import TypedDict +from datetime import datetime +import json,time +``` + +``` python +app = FastHTML() +``` + +The +[`FastHTML`](https://AnswerDotAI.github.io/fasthtml/api/core.html#fasthtml) +class is the main application class for FastHTML apps. + +``` python +rt = app.route +``` + +`app.route` is used to register route handlers. It is a decorator, which +means we place it before a function that is used as a handler. Because +it’s used frequently in most FastHTML applications, we often alias it as +`rt`, as we do here. + +## Basic Route Handling + +``` python +@rt("/hi") +def get(): return 'Hi there' +``` + +Handler functions can return strings directly. These strings are sent as +the response body to the client. + +``` python +cli = Client(app) +``` + +[`Client`](https://AnswerDotAI.github.io/fasthtml/api/core.html#client) +is a test client for FastHTML applications. It allows you to simulate +requests to your app without running a server. + +``` python +cli.get('/hi').text +``` + + 'Hi there' + +The `get` method on a +[`Client`](https://AnswerDotAI.github.io/fasthtml/api/core.html#client) +instance simulates GET requests to the app. It returns a response object +that has a `.text` attribute, which you can use to access the body of +the response. It calls `httpx.get` internally – all httpx HTTP verbs are +supported. + +``` python +@rt("/hi") +def post(): return 'Postal' +cli.post('/hi').text +``` + + 'Postal' + +Handler functions can be defined for different HTTP methods on the same +route. Here, we define a `post` handler for the `/hi` route. The +[`Client`](https://AnswerDotAI.github.io/fasthtml/api/core.html#client) +instance can simulate different HTTP methods, including POST requests. + +## Request and Response Objects + +``` python +@app.get("/hostie") +def show_host(req): return req.headers['host'] +cli.get('/hostie').text +``` + + 'testserver' + +Handler functions can accept a `req` (or `request`) parameter, which +represents the incoming request. This object contains information about +the request, including headers. In this example, we return the `host` +header from the request. The test client uses ‘testserver’ as the +default host. + +In this example, we use `@app.get("/hostie")` instead of +`@rt("/hostie")`. The `@app.get()` decorator explicitly specifies the +HTTP method (GET) for the route, while `@rt()` by default handles both +GET and POST requests. + +``` python +@rt +def yoyo(): return 'a yoyo' +cli.post('/yoyo').text +``` + + 'a yoyo' + +If the `@rt` decorator is used without arguments, it uses the function +name as the route path. Here, the `yoyo` function becomes the handler +for the `/yoyo` route. This handler responds to GET and POST methods, +since a specific method wasn’t provided. + +``` python +@rt +def ft1(): return Html(Div('Text.')) +print(cli.get('/ft1').text) +``` + + +
Text.
+ + +Handler functions can return +[`FT`](https://docs.fastht.ml/explains/explaining_xt_components.html) +objects, which are automatically converted to HTML strings. The `FT` +class can take other `FT` components as arguments, such as `Div`. This +allows for easy composition of HTML elements in your responses. + +``` python +@app.get +def autopost(): return Html(Div('Text.', hx_post=yoyo.to())) +print(cli.get('/autopost').text) +``` + + +
Text.
+ + +The `rt` decorator modifies the `yoyo` function by adding an `rt()` +method. This method returns the route path associated with the handler. +It’s a convenient way to reference the route of a handler function +dynamically. + +In the example, `yoyo.to()` is used as the value for `hx_post`. This +means when the div is clicked, it will trigger an HTMX POST request to +the route of the `yoyo` handler. This approach allows for flexible, DRY +code by avoiding hardcoded route strings and automatically updating if +the route changes. + +This pattern is particularly useful in larger applications where routes +might change, or when building reusable components that need to +reference their own routes dynamically. + +``` python +@app.get +def autoget(): return Html(Body(Div('Text.', cls='px-2', hx_post=show_host.to(a='b')))) +print(cli.get('/autoget').text) +``` + + + +
Text.
+ + + +The `rt()` method of handler functions can also accept parameters. When +called with parameters, it returns the route path with a query string +appended. In this example, `show_host.to(a='b')` generates the path +`/hostie?a=b`. + +The `Body` component is used here to demonstrate nesting of FT +components. `Div` is nested inside `Body`, showcasing how you can create +more complex HTML structures. + +The `cls` parameter is used to add a CSS class to the `Div`. This +translates to the `class` attribute in the rendered HTML. (`class` can’t +be used as a parameter name directly in Python since it’s a reserved +word.) + +``` python +@rt('/ft2') +def get(): return Title('Foo'),H1('bar') +print(cli.get('/ft2').text) +``` + + + + + Foo + + + + +

bar

+ + + +Handler functions can return multiple `FT` objects as a tuple. The first +item is treated as the `Title`, and the rest are added to the `Body`. +When the request is not an HTMX request, FastHTML automatically adds +necessary HTML boilerplate, including default `head` content with +required scripts. + +When using `app.route` (or `rt`), if the function name matches an HTTP +verb (e.g., `get`, `post`, `put`, `delete`), that HTTP method is +automatically used for the route. In this case, a path must be +explicitly provided as an argument to the decorator. + +``` python +hxhdr = {'headers':{'hx-request':"1"}} +print(cli.get('/ft2', **hxhdr).text) +``` + + Foo +

bar

+ +For HTMX requests (indicated by the `hx-request` header), FastHTML +returns only the specified components without the full HTML structure. +This allows for efficient partial page updates in HTMX applications. + +``` python +@rt('/ft3') +def get(): return H1('bar') +print(cli.get('/ft3', **hxhdr).text) +``` + +

bar

+ +When a handler function returns a single `FT` object for an HTMX +request, it’s rendered as a single HTML partial. + +``` python +@rt('/ft4') +def get(): return Html(Head(Title('hi')), Body(P('there'))) + +print(cli.get('/ft4').text) +``` + + + + hi + + +

there

+ + + +Handler functions can return a complete `Html` structure, including +`Head` and `Body` components. When a full HTML structure is returned, +FastHTML doesn’t add any additional boilerplate. This gives you full +control over the HTML output when needed. + +``` python +@rt +def index(): return "welcome!" +print(cli.get('/').text) +``` + + welcome! + +The `index` function is a special handler in FastHTML. When defined +without arguments to the `@rt` decorator, it automatically becomes the +handler for the root path (`'/'`). This is a convenient way to define +the main page or entry point of your application. + +## Path and Query Parameters + +``` python +@rt('/user/{nm}', name='gday') +def get(nm:str=''): return f"Good day to you, {nm}!" +cli.get('/user/Alexis').text +``` + + 'Good day to you, Alexis!' + +Handler functions can use path parameters, defined using curly braces in +the route – this is implemented by Starlette directly, so all Starlette +path parameters can be used. These parameters are passed as arguments to +the function. + +The `name` parameter in the decorator allows you to give the route a +name, which can be used for URL generation. + +In this example, `{nm}` in the route becomes the `nm` parameter in the +function. The function uses this parameter to create a personalized +greeting. + +``` python +@app.get +def autolink(): return Html(Div('Text.', link=uri('gday', nm='Alexis'))) +print(cli.get('/autolink').text) +``` + + +
Text.
+ + +The [`uri`](https://AnswerDotAI.github.io/fasthtml/api/core.html#uri) +function is used to generate URLs for named routes. It takes the route +name as its first argument, followed by any path or query parameters +needed for that route. + +In this example, `uri('gday', nm='Alexis')` generates the URL for the +route named ‘gday’ (which we defined earlier as ‘/user/{nm}’), with +‘Alexis’ as the value for the ‘nm’ parameter. + +The `link` parameter in FT components sets the `href` attribute of the +rendered HTML element. By using +[`uri()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#uri), we +can dynamically generate correct URLs even if the underlying route +structure changes. + +This approach promotes maintainable code by centralizing route +definitions and avoiding hardcoded URLs throughout the application. + +``` python +@rt('/link') +def get(req): return f"{req.url_for('gday', nm='Alexis')}; {req.url_for('show_host')}" + +cli.get('/link').text +``` + + 'http://testserver/user/Alexis; http://testserver/hostie' + +The `url_for` method of the request object can be used to generate URLs +for named routes. It takes the route name as its first argument, +followed by any path parameters needed for that route. + +In this example, `req.url_for('gday', nm='Alexis')` generates the full +URL for the route named ‘gday’, including the scheme and host. +Similarly, `req.url_for('show_host')` generates the URL for the +‘show_host’ route. + +This method is particularly useful when you need to generate absolute +URLs, such as for email links or API responses. It ensures that the +correct host and scheme are included, even if the application is +accessed through different domains or protocols. + +``` python +app.url_path_for('gday', nm='Jeremy') +``` + + '/user/Jeremy' + +The `url_path_for` method of the application can be used to generate URL +paths for named routes. Unlike `url_for`, it returns only the path +component of the URL, without the scheme or host. + +In this example, `app.url_path_for('gday', nm='Jeremy')` generates the +path ‘/user/Jeremy’ for the route named ‘gday’. + +This method is useful when you need relative URLs or just the path +component, such as for internal links or when constructing URLs in a +host-agnostic manner. + +``` python +@rt('/oops') +def get(nope): return nope +r = cli.get('/oops?nope=1') +print(r) +r.text +``` + + + + /Users/wgilliam/development/projects/aai/fasthtml/fasthtml/core.py:188: UserWarning: `nope has no type annotation and is not a recognised special name, so is ignored. + if arg!='resp': warn(f"`{arg} has no type annotation and is not a recognised special name, so is ignored.") + + '' + +Handler functions can include parameters, but they must be +type-annotated or have special names (like `req`) to be recognized. In +this example, the `nope` parameter is not annotated, so it’s ignored, +resulting in a warning. + +When a parameter is ignored, it doesn’t receive the value from the query +string. This can lead to unexpected behavior, as the function attempts +to return `nope`, which is undefined. + +The `cli.get('/oops?nope=1')` call succeeds with a 200 OK status because +the handler doesn’t raise an exception, but it returns an empty +response, rather than the intended value. + +To fix this, you should either add a type annotation to the parameter +(e.g., `def get(nope: str):`) or use a recognized special name like +`req`. + +``` python +@rt('/html/{idx}') +def get(idx:int): return Body(H4(f'Next is {idx+1}.')) +print(cli.get('/html/1', **hxhdr).text) +``` + + +

Next is 2.

+ + +Path parameters can be type-annotated, and FastHTML will automatically +convert them to the specified type if possible. In this example, `idx` +is annotated as `int`, so it’s converted from the string in the URL to +an integer. + +``` python +reg_re_param("imgext", "ico|gif|jpg|jpeg|webm") + +@rt(r'/static/{path:path}{fn}.{ext:imgext}') +def get(fn:str, path:str, ext:str): return f"Getting {fn}.{ext} from /{path}" + +print(cli.get('/static/foo/jph.ico').text) +``` + + Getting jph.ico from /foo/ + +The +[`reg_re_param`](https://AnswerDotAI.github.io/fasthtml/api/core.html#reg_re_param) +function is used to register custom path parameter types using regular +expressions. Here, we define a new path parameter type called “imgext” +that matches common image file extensions. + +Handler functions can use complex path patterns with multiple parameters +and custom types. In this example, the route pattern +`r'/static/{path:path}{fn}.{ext:imgext}'` uses three path parameters: + +1. `path`: A Starlette built-in type that matches any path segments +2. `fn`: The filename without extension +3. `ext`: Our custom “imgext” type that matches specific image + extensions + +``` python +ModelName = str_enum('ModelName', "alexnet", "resnet", "lenet") + +@rt("/models/{nm}") +def get(nm:ModelName): return nm + +print(cli.get('/models/alexnet').text) +``` + + alexnet + +We define `ModelName` as an enum with three possible values: “alexnet”, +“resnet”, and “lenet”. Handler functions can use these enum types as +parameter annotations. In this example, the `nm` parameter is annotated +with `ModelName`, which ensures that only valid model names are +accepted. + +When a request is made with a valid model name, the handler function +returns that name. This pattern is useful for creating type-safe APIs +with a predefined set of valid values. + +``` python +@rt("/files/{path}") +async def get(path: Path): return path.with_suffix('.txt') +print(cli.get('/files/foo').text) +``` + + foo.txt + +Handler functions can use +[`Path`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#path) +objects as parameter types. The +[`Path`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#path) type +is from Python’s standard library `pathlib` module, which provides an +object-oriented interface for working with file paths. In this example, +the `path` parameter is annotated with +[`Path`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#path), so +FastHTML automatically converts the string from the URL to a +[`Path`](https://AnswerDotAI.github.io/fasthtml/api/svg.html#path) +object. + +This approach is particularly useful when working with file-related +routes, as it provides a convenient and platform-independent way to +handle file paths. + +``` python +fake_db = [{"name": "Foo"}, {"name": "Bar"}] + +@rt("/items/") +def get(idx:int|None = 0): return fake_db[idx] +print(cli.get('/items/?idx=1').text) +``` + + {"name":"Bar"} + +Handler functions can use query parameters, which are automatically +parsed from the URL. In this example, `idx` is a query parameter with a +default value of 0. It’s annotated as `int|None`, allowing it to be +either an integer or None. + +The function uses this parameter to index into a fake database +(`fake_db`). When a request is made with a valid `idx` query parameter, +the handler returns the corresponding item from the database. + +``` python +print(cli.get('/items/').text) +``` + + {"name":"Foo"} + +When no `idx` query parameter is provided, the handler function uses the +default value of 0. This results in returning the first item from the +`fake_db` list, which is `{"name":"Foo"}`. + +This behavior demonstrates how default values for query parameters work +in FastHTML. They allow the API to have a sensible default behavior when +optional parameters are not provided. + +``` python +print(cli.get('/items/?idx=g')) +``` + + + +When an invalid value is provided for a typed query parameter, FastHTML +returns a 404 Not Found response. In this example, ‘g’ is not a valid +integer for the `idx` parameter, so the request fails with a 404 status. + +This behavior ensures type safety and prevents invalid inputs from +reaching the handler function. + +``` python +@app.get("/booly/") +def _(coming:bool=True): return 'Coming' if coming else 'Not coming' +print(cli.get('/booly/?coming=true').text) +print(cli.get('/booly/?coming=no').text) +``` + + Coming + Not coming + +Handler functions can use boolean query parameters. In this example, +`coming` is a boolean parameter with a default value of `True`. FastHTML +automatically converts string values like ‘true’, ‘false’, ‘1’, ‘0’, +‘on’, ‘off’, ‘yes’, and ‘no’ to their corresponding boolean values. + +The underscore `_` is used as the function name in this example to +indicate that the function’s name is not important or won’t be +referenced elsewhere. This is a common Python convention for throwaway +or unused variables, and it works here because FastHTML uses the route +decorator parameter, when provided, to determine the URL path, not the +function name. By default, both `get` and `post` methods can be used in +routes that don’t specify an http method (by either using `app.get`, +`def get`, or the `methods` parameter to `app.route`). + +``` python +@app.get("/datie/") +def _(d:parsed_date): return d +date_str = "17th of May, 2024, 2p" +print(cli.get(f'/datie/?d={date_str}').text) +``` + + 2024-05-17 14:00:00 + +Handler functions can use `date` objects as parameter types. FastHTML +uses `dateutil.parser` library to automatically parse a wide variety of +date string formats into `date` objects. + +``` python +@app.get("/ua") +async def _(user_agent:str): return user_agent +print(cli.get('/ua', headers={'User-Agent':'FastHTML'}).text) +``` + + FastHTML + +Handler functions can access HTTP headers by using parameter names that +match the header names. In this example, `user_agent` is used as a +parameter name, which automatically captures the value of the +‘User-Agent’ header from the request. + +The +[`Client`](https://AnswerDotAI.github.io/fasthtml/api/core.html#client) +instance allows setting custom headers for test requests. Here, we set +the ‘User-Agent’ header to ‘FastHTML’ in the test request. + +``` python +@app.get("/hxtest") +def _(htmx): return htmx.request +print(cli.get('/hxtest', headers={'HX-Request':'1'}).text) + +@app.get("/hxtest2") +def _(foo:HtmxHeaders, req): return foo.request +print(cli.get('/hxtest2', headers={'HX-Request':'1'}).text) +``` + + 1 + 1 + +Handler functions can access HTMX-specific headers using either the +special `htmx` parameter name, or a parameter annotated with +[`HtmxHeaders`](https://AnswerDotAI.github.io/fasthtml/api/core.html#htmxheaders). +Both approaches provide access to HTMX-related information. + +In these examples, the `htmx.request` attribute returns the value of the +‘HX-Request’ header. + +``` python +app.chk = 'foo' +@app.get("/app") +def _(app): return app.chk +print(cli.get('/app').text) +``` + + foo + +Handler functions can access the +[`FastHTML`](https://AnswerDotAI.github.io/fasthtml/api/core.html#fasthtml) +application instance using the special `app` parameter name. This allows +handlers to access application-level attributes and methods. + +In this example, we set a custom attribute `chk` on the application +instance. The handler function then uses the `app` parameter to access +this attribute and return its value. + +``` python +@app.get("/app2") +def _(foo:FastHTML): return foo.chk,HttpHeader("mykey", "myval") +r = cli.get('/app2', **hxhdr) +print(r.text) +print(r.headers) +``` + + foo + Headers({'mykey': 'myval', 'content-length': '3', 'content-type': 'text/html; charset=utf-8'}) + +Handler functions can access the +[`FastHTML`](https://AnswerDotAI.github.io/fasthtml/api/core.html#fasthtml) +application instance using a parameter annotated with +[`FastHTML`](https://AnswerDotAI.github.io/fasthtml/api/core.html#fasthtml). +This allows handlers to access application-level attributes and methods, +just like using the special `app` parameter name. + +Handlers can return tuples containing both content and +[`HttpHeader`](https://AnswerDotAI.github.io/fasthtml/api/core.html#httpheader) +objects. +[`HttpHeader`](https://AnswerDotAI.github.io/fasthtml/api/core.html#httpheader) +allows setting custom HTTP headers in the response. + +In this example: + +- We define a handler that returns both the `chk` attribute from the + application and a custom header. +- The `HttpHeader("mykey", "myval")` sets a custom header in the + response. +- We use the test client to make a request and examine both the response + text and headers. +- The response includes the custom header “mykey” along with standard + headers like content-length and content-type. + +``` python +@app.get("/app3") +def _(foo:FastHTML): return HtmxResponseHeaders(location="http://example.org") +r = cli.get('/app3') +print(r.headers) +``` + + Headers({'hx-location': 'http://example.org', 'content-length': '0', 'content-type': 'text/html; charset=utf-8'}) + +Handler functions can return +[`HtmxResponseHeaders`](https://AnswerDotAI.github.io/fasthtml/api/core.html#htmxresponseheaders) +objects to set HTMX-specific response headers. This is useful for +HTMX-specific behaviors like client-side redirects. + +In this example we define a handler that returns an +[`HtmxResponseHeaders`](https://AnswerDotAI.github.io/fasthtml/api/core.html#htmxresponseheaders) +object with a `location` parameter, which sets the `HX-Location` header +in the response. HTMX uses this for client-side redirects. + +``` python +@app.get("/app4") +def _(foo:FastHTML): return Redirect("http://example.org") +cli.get('/app4', follow_redirects=False) +``` + + + +Handler functions can return +[`Redirect`](https://AnswerDotAI.github.io/fasthtml/api/core.html#redirect) +objects to perform HTTP redirects. This is useful for redirecting users +to different pages or external URLs. + +In this example: + +- We define a handler that returns a + [`Redirect`](https://AnswerDotAI.github.io/fasthtml/api/core.html#redirect) + object with the URL “http://example.org”. +- The `cli.get('/app4', follow_redirects=False)` call simulates a GET + request to the ‘/app4’ route without following redirects. +- The response has a 303 See Other status code, indicating a redirect. + +The `follow_redirects=False` parameter is used to prevent the test +client from automatically following the redirect, allowing us to inspect +the redirect response itself. + +``` python +Redirect.__response__ +``` + + + +The +[`Redirect`](https://AnswerDotAI.github.io/fasthtml/api/core.html#redirect) +class in FastHTML implements a `__response__` method, which is a special +method recognized by the framework. When a handler returns a +[`Redirect`](https://AnswerDotAI.github.io/fasthtml/api/core.html#redirect) +object, FastHTML internally calls this `__response__` method to replace +the original response. + +The `__response__` method takes a `req` parameter, which represents the +incoming request. This allows the method to access request information +if needed when constructing the redirect response. + +``` python +@rt +def meta(): + return ((Title('hi'),H1('hi')), + (Meta(property='image'), Meta(property='site_name'))) + +print(cli.post('/meta').text) +``` + + + + + hi + + + + + + +

hi

+ + + +FastHTML automatically identifies elements typically placed in the +`` (like `Title` and `Meta`) and positions them accordingly, while +other elements go in the ``. + +In this example: - `(Title('hi'), H1('hi'))` defines the title and main +heading. The title is placed in the head, and the H1 in the body. - +`(Meta(property='image'), Meta(property='site_name'))` defines two meta +tags, which are both placed in the head. + +## APIRouter + +[`APIRouter`](https://AnswerDotAI.github.io/fasthtml/api/core.html#apirouter) +is useful when you want to split your application routes across multiple +`.py` files that are part of a single FastHTMl application. It accepts +an optional `prefix` argument that will be applied to all routes within +that instance of +[`APIRouter`](https://AnswerDotAI.github.io/fasthtml/api/core.html#apirouter). + +Below we define several hypothetical product related routes in a +`products.py` and then demonstrate how they can seamlessly be +incorporated into a FastHTML app instance. + +``` python +# products.py +ar = APIRouter(prefix="/products") + +@ar("/all") +def all_products(req): + return Div( + "Welcome to the Products Page! Click the button below to look at the details for product 42", + Div( + Button( + "Details", + hx_get=req.url_for("details", pid=42), + hx_target="#products_list", + hx_swap="outerHTML", + ), + ), + id="products_list", + ) + + +@ar.get("/{pid}", name="details") +def details(pid: int): + return f"Here are the product details for ID: {pid}" +``` + +Since we specified the `prefix=/products` in our hypothetical +`products.py` file, all routes defined in that file will be found under +`/products`. + +``` python +print(str(ar.rt_funcs.all_products)) +print(str(ar.rt_funcs.details)) +``` + + /products/all + /products/{pid} + +``` python +# main.py +# from products import ar + +app, rt = fast_app() +ar.to_app(app) + +@rt +def index(): + return Div( + "Click me for a look at our products", + hx_get=ar.rt_funcs.all_products, + hx_swap="outerHTML", + ) +``` + +Note how you can reference our python route functions via +`APIRouter.rt_funcs` in your `hx_{http_method}` calls like normal. + +## Form Data and JSON Handling + +``` python +@app.post('/profile/me') +def profile_update(username: str): return username + +print(cli.post('/profile/me', data={'username' : 'Alexis'}).text) +r = cli.post('/profile/me', data={}) +print(r.text) +r +``` + + 404 Not Found + 404 Not Found + + + +Handler functions can accept form data parameters, without needing to +manually extract it from the request. In this example, `username` is +expected to be sent as form data. + +If required form data is missing, FastHTML automatically returns a 400 +Bad Request response with an error message. + +The `data` parameter in the `cli.post()` method simulates sending form +data in the request. + +``` python +@app.post('/pet/dog') +def pet_dog(dogname: str = None): return dogname or 'unknown name' +print(cli.post('/pet/dog', data={}).text) +``` + + 404 Not Found + +Handlers can have optional form data parameters with default values. In +this example, `dogname` is an optional parameter with a default value of +`None`. + +Here, if the form data doesn’t include the `dogname` field, the function +uses the default value. The function returns either the provided +`dogname` or ‘unknown name’ if `dogname` is `None`. + +``` python +@dataclass +class Bodie: a:int;b:str + +@rt("/bodie/{nm}") +def post(nm:str, data:Bodie): + res = asdict(data) + res['nm'] = nm + return res + +print(cli.post('/bodie/me', data=dict(a=1, b='foo', nm='me')).text) +``` + + 404 Not Found + +You can use dataclasses to define structured form data. In this example, +`Bodie` is a dataclass with `a` (int) and `b` (str) fields. + +FastHTML automatically converts the incoming form data to a `Bodie` +instance where attribute names match parameter names. Other form data +elements are matched with parameters with the same names (in this case, +`nm`). + +Handler functions can return dictionaries, which FastHTML automatically +JSON-encodes. + +``` python +@app.post("/bodied/") +def bodied(data:dict): return data + +d = dict(a=1, b='foo') +print(cli.post('/bodied/', data=d).text) +``` + + 404 Not Found + +`dict` parameters capture all form data as a dictionary. In this +example, the `data` parameter is annotated with `dict`, so FastHTML +automatically converts all incoming form data into a dictionary. + +Note that when form data is converted to a dictionary, all values become +strings, even if they were originally numbers. This is why the ‘a’ key +in the response has a string value “1” instead of the integer 1. + +``` python +nt = namedtuple('Bodient', ['a','b']) + +@app.post("/bodient/") +def bodient(data:nt): return asdict(data) +print(cli.post('/bodient/', data=d).text) +``` + + 404 Not Found + +Handler functions can use named tuples to define structured form data. +In this example, `Bodient` is a named tuple with `a` and `b` fields. + +FastHTML automatically converts the incoming form data to a `Bodient` +instance where field names match parameter names. As with the previous +example, all form data values are converted to strings in the process. + +``` python +class BodieTD(TypedDict): a:int;b:str='foo' + +@app.post("/bodietd/") +def bodient(data:BodieTD): return data +print(cli.post('/bodietd/', data=d).text) +``` + + 404 Not Found + +You can use `TypedDict` to define structured form data with type hints. +In this example, `BodieTD` is a `TypedDict` with `a` (int) and `b` (str) +fields, where `b` has a default value of ‘foo’. + +FastHTML automatically converts the incoming form data to a `BodieTD` +instance where keys match the defined fields. Unlike with regular +dictionaries or named tuples, FastHTML respects the type hints in +`TypedDict`, converting values to the specified types when possible +(e.g., converting ‘1’ to the integer 1 for the ‘a’ field). + +``` python +class Bodie2: + a:int|None; b:str + def __init__(self, a, b='foo'): store_attr() + +@app.post("/bodie2/") +def bodie(d:Bodie2): return f"a: {d.a}; b: {d.b}" +print(cli.post('/bodie2/', data={'a':1}).text) +``` + + 404 Not Found + +Custom classes can be used to define structured form data. Here, +`Bodie2` is a custom class with `a` (int|None) and `b` (str) attributes, +where `b` has a default value of ‘foo’. The `store_attr()` function +(from fastcore) automatically assigns constructor parameters to instance +attributes. + +FastHTML automatically converts the incoming form data to a `Bodie2` +instance, matching form fields to constructor parameters. It respects +type hints and default values. + +``` python +@app.post("/b") +def index(it: Bodie): return Titled("It worked!", P(f"{it.a}, {it.b}")) + +s = json.dumps({"b": "Lorem", "a": 15}) +print(cli.post('/b', headers={"Content-Type": "application/json", 'hx-request':"1"}, data=s).text) +``` + + 404 Not Found + +Handler functions can accept JSON data as input, which is automatically +parsed into the specified type. In this example, `it` is of type +`Bodie`, and FastHTML converts the incoming JSON data to a `Bodie` +instance. + +The +[`Titled`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#titled) +component is used to create a page with a title and main content. It +automatically generates an `

` with the provided title, wraps the +content in a `
` tag with a “container” class, and adds a `title` +to the head. + +When making a request with JSON data: - Set the “Content-Type” header to +“application/json” - Provide the JSON data as a string in the `data` +parameter of the request + +## Cookies, Sessions, File Uploads, and more + +``` python +@rt("/setcookie") +def get(): return cookie('now', datetime.now()) + +@rt("/getcookie") +def get(now:parsed_date): return f'Cookie was set at time {now.time()}' + +print(cli.get('/setcookie').text) +time.sleep(0.01) +cli.get('/getcookie').text +``` + + 404 Not Found + + '404 Not Found' + +Handler functions can set and retrieve cookies. In this example: + +- The `/setcookie` route sets a cookie named ‘now’ with the current + datetime. +- The `/getcookie` route retrieves the ‘now’ cookie and returns its + value. + +The +[`cookie()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#cookie) +function is used to create a cookie response. FastHTML automatically +converts the datetime object to a string when setting the cookie, and +parses it back to a date object when retrieving it. + +``` python +cookie('now', datetime.now()) +``` + + HttpHeader(k='set-cookie', v='now="2024-12-04 13:45:24.154187"; Path=/; SameSite=lax') + +The +[`cookie()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#cookie) +function returns an +[`HttpHeader`](https://AnswerDotAI.github.io/fasthtml/api/core.html#httpheader) +object with the ‘set-cookie’ key. You can return it in a tuple along +with `FT` elements, along with anything else FastHTML supports in +responses. + +``` python +app = FastHTML(secret_key='soopersecret') +cli = Client(app) +rt = app.route +``` + +``` python +@rt("/setsess") +def get(sess, foo:str=''): + now = datetime.now() + sess['auth'] = str(now) + return f'Set to {now}' + +@rt("/getsess") +def get(sess): return f'Session time: {sess["auth"]}' + +print(cli.get('/setsess').text) +time.sleep(0.01) + +cli.get('/getsess').text +``` + + Set to 2024-12-04 13:45:24.159764 + + 'Session time: 2024-12-04 13:45:24.159764' + +Sessions store and retrieve data across requests. To use sessions, you +should to initialize the FastHTML application with a `secret_key`. This +is used to cryptographically sign the cookie used by the session. + +The `sess` parameter in handler functions provides access to the session +data. You can set and get session variables using dictionary-style +access. + +``` python +@rt("/upload") +async def post(uf:UploadFile): return (await uf.read()).decode() + +with open('../../CHANGELOG.md', 'rb') as f: + print(cli.post('/upload', files={'uf':f}, data={'msg':'Hello'}).text[:15]) +``` + + # Release notes + +Handler functions can accept file uploads using Starlette’s `UploadFile` +type. In this example: + +- The `/upload` route accepts a file upload named `uf`. +- The `UploadFile` object provides an asynchronous `read()` method to + access the file contents. +- We use `await` to read the file content asynchronously and decode it + to a string. + +We added `async` to the handler function because it uses `await` to read +the file content asynchronously. In Python, any function that uses +`await` must be declared as `async`. This allows the function to be run +asynchronously, potentially improving performance by not blocking other +operations while waiting for the file to be read. + +``` python +app.static_route('.md', static_path='../..') +print(cli.get('/README.md').text[:10]) +``` + + # FastHTML + +The `static_route` method of the FastHTML application allows serving +static files with specified extensions from a given directory. In this +example: + +- `.md` files are served from the `../..` directory (two levels up from + the current directory). +- Accessing `/README.md` returns the contents of the README.md file from + that directory. + +``` python +help(app.static_route_exts) +``` + + Help on method static_route_exts in module fasthtml.core: + + static_route_exts(prefix='/', static_path='.', exts='static') method of fasthtml.core.FastHTML instance + Add a static route at URL path `prefix` with files from `static_path` and `exts` defined by `reg_re_param()` + +``` python +app.static_route_exts() +print(cli.get('/README.txt').text[:50]) +``` + + 404 Not Found + +The `static_route_exts` method of the FastHTML application allows +serving static files with specified extensions from a given directory. +By default: + +- It serves files from the current directory (‘.’). +- It uses the ‘static’ regex, which includes common static file + extensions like ‘ico’, ‘gif’, ‘jpg’, ‘css’, ‘js’, etc. +- The URL prefix is set to ‘/’. + +The ‘static’ regex is defined by FastHTML using this code: + +``` python +reg_re_param("static", "ico|gif|jpg|jpeg|webm|css|js|woff|png|svg|mp4|webp|ttf|otf|eot|woff2|txt|html|map") +``` + +``` python +@rt("/form-submit/{list_id}") +def options(list_id: str): + headers = { + 'Access-Control-Allow-Origin': '*', + 'Access-Control-Allow-Methods': 'POST', + 'Access-Control-Allow-Headers': '*', + } + return Response(status_code=200, headers=headers) + +print(cli.options('/form-submit/2').headers) +``` + + Headers({'access-control-allow-origin': '*', 'access-control-allow-methods': 'POST', 'access-control-allow-headers': '*', 'content-length': '0', 'set-cookie': 'session_=eyJhdXRoIjogIjIwMjQtMTItMDQgMTM6NDU6MjQuMTU5NzY0In0=.Z1DNdA.GV-NVoOnJeambm9_uE3crhoGH34; path=/; Max-Age=31536000; httponly; samesite=lax'}) + +FastHTML handlers can handle OPTIONS requests and set custom headers. In +this example: + +- The `/form-submit/{list_id}` route handles OPTIONS requests. +- Custom headers are set to allow cross-origin requests (CORS). +- The function returns a Starlette `Response` object with a 200 status + code and the custom headers. + +You can return any Starlette Response type from a handler function, +giving you full control over the response when needed. + +``` python +def _not_found(req, exc): return Div('nope') + +app = FastHTML(exception_handlers={404:_not_found}) +cli = Client(app) +rt = app.route + +r = cli.get('/') +print(r.text) +``` + + + + + FastHTML page + + + + +
nope
+ + + +FastHTML allows you to define custom exception handlers – in this case, +a custom 404 (Not Found) handler function `_not_found`, which returns a +`Div` component with the text ‘nope’. diff --git a/ref/live_reload.html b/ref/live_reload.html new file mode 100644 index 00000000..bee22810 --- /dev/null +++ b/ref/live_reload.html @@ -0,0 +1,855 @@ + + + + + + + + + +Live Reloading – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Live Reloading

+
+ + + +
+ + + + +
+ + + +
+ + + +

When building your app it can be useful to view your changes in a web browser as you make them. FastHTML supports live reloading which means that it watches for any changes to your code and automatically refreshes the webpage in your browser.

+

To enable live reloading simply replace FastHTML in your app with FastHTMLWithLiveReload.

+
from fasthtml.common import *
+app = FastHTMLWithLiveReload()
+

Then in your terminal run uvicorn with reloading enabled.

+
uvicorn main:app --reload
+

⚠️ Gotchas - A reload is only triggered when you save your changes. - FastHTMLWithLiveReload should only be used during development. - If your app spans multiple directories you might need to use the --reload-dir flag to watch all files in each directory. See the uvicorn docs for more info. - The live reload script is only injected into the page when rendering ft components.

+
+

Live reloading with fast_app

+

In development the fast_app function provides the same functionality. It instantiates the FastHTMLWithLiveReload class if you pass live=True:

+
+
+
main.py
+
+
from fasthtml.common import *
+
+1app, rt = fast_app(live=True)
+
+2serve()
+
+
+
1
+
+fast_app() instantiates the FastHTMLWithLiveReload class. +
+
2
+
+serve() is a wrapper around a uvicorn call. +
+
+

To run main.py in live reload mode, just do python main.py. We recommend turning off live reload when deploying your app to production.

+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/ref/live_reload.html.md b/ref/live_reload.html.md new file mode 100644 index 00000000..d40b0f9c --- /dev/null +++ b/ref/live_reload.html.md @@ -0,0 +1,61 @@ +# Live Reloading + + + + +When building your app it can be useful to view your changes in a web +browser as you make them. FastHTML supports live reloading which means +that it watches for any changes to your code and automatically refreshes +the webpage in your browser. + +To enable live reloading simply replace +[`FastHTML`](https://AnswerDotAI.github.io/fasthtml/api/core.html#fasthtml) +in your app with `FastHTMLWithLiveReload`. + +``` python +from fasthtml.common import * +app = FastHTMLWithLiveReload() +``` + +Then in your terminal run `uvicorn` with reloading enabled. + + uvicorn main:app --reload + +**⚠️ Gotchas** - A reload is only triggered when you save your +changes. - `FastHTMLWithLiveReload` should only be used during +development. - If your app spans multiple directories you might need to +use the `--reload-dir` flag to watch all files in each directory. See +the uvicorn [docs](https://www.uvicorn.org/settings/#development) for +more info. - The live reload script is only injected into the page when +rendering [ft +components](https://docs.fastht.ml/explains/explaining_xt_components.html). + +## Live reloading with `fast_app` + +In development the `fast_app` function provides the same functionality. +It instantiates the `FastHTMLWithLiveReload` class if you pass +`live=True`: + +
+ +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app(live=True) + +serve() +``` + +
+ +Line 3 +`fast_app()` instantiates the `FastHTMLWithLiveReload` class. + +Line 5 +[`serve()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#serve) +is a wrapper around a `uvicorn` call. + +To run `main.py` in live reload mode, just do `python main.py`. We +recommend turning off live reload when deploying your app to production. diff --git a/robots.txt b/robots.txt new file mode 100644 index 00000000..4edc0aac --- /dev/null +++ b/robots.txt @@ -0,0 +1 @@ +Sitemap: https://AnswerDotAI.github.io/fasthtml/sitemap.xml diff --git a/search.json b/search.json new file mode 100644 index 00000000..ab005824 --- /dev/null +++ b/search.json @@ -0,0 +1,1528 @@ +[ + { + "objectID": "ref/live_reload.html", + "href": "ref/live_reload.html", + "title": "Live Reloading", + "section": "", + "text": "When building your app it can be useful to view your changes in a web browser as you make them. FastHTML supports live reloading which means that it watches for any changes to your code and automatically refreshes the webpage in your browser.\nTo enable live reloading simply replace FastHTML in your app with FastHTMLWithLiveReload.\nThen in your terminal run uvicorn with reloading enabled.\n⚠️ Gotchas - A reload is only triggered when you save your changes. - FastHTMLWithLiveReload should only be used during development. - If your app spans multiple directories you might need to use the --reload-dir flag to watch all files in each directory. See the uvicorn docs for more info. - The live reload script is only injected into the page when rendering ft components.", + "crumbs": [ + "Home", + "Reference", + "Live Reloading" + ] + }, + { + "objectID": "ref/live_reload.html#live-reloading-with-fast_app", + "href": "ref/live_reload.html#live-reloading-with-fast_app", + "title": "Live Reloading", + "section": "Live reloading with fast_app", + "text": "Live reloading with fast_app\nIn development the fast_app function provides the same functionality. It instantiates the FastHTMLWithLiveReload class if you pass live=True:\n\n\nmain.py\n\nfrom fasthtml.common import *\n\n1app, rt = fast_app(live=True)\n\n2serve()\n\n\n1\n\nfast_app() instantiates the FastHTMLWithLiveReload class.\n\n2\n\nserve() is a wrapper around a uvicorn call.\n\n\nTo run main.py in live reload mode, just do python main.py. We recommend turning off live reload when deploying your app to production.", + "crumbs": [ + "Home", + "Reference", + "Live Reloading" + ] + }, + { + "objectID": "ref/defining_xt_component.html", + "href": "ref/defining_xt_component.html", + "title": "Custom Components", + "section": "", + "text": "The majority of the time the default ft components are all you need (for example Div, P, H1, etc.).\nHowever, there are many situations where you need a custom ft component that creates a unique HTML tag (for example <zero-md></zero-md>). There are many options in FastHTML to do this, and this section will walk through them. Generally you want to use the highest level option that fits your needs.", + "crumbs": [ + "Home", + "Reference", + "Custom Components" + ] + }, + { + "objectID": "ref/defining_xt_component.html#notstr", + "href": "ref/defining_xt_component.html#notstr", + "title": "Custom Components", + "section": "NotStr", + "text": "NotStr\nThe first way is to use the NotStr class to use an HTML tag as a string. It works as a one-off but quickly becomes harder to work with as complexity grows. However we can see that you can genenrate the same xml using NotStr as the out-of-the-box components.\n\nfrom fasthtml.common import NotStr,Div, to_xml\n\n\ndiv_NotStr = NotStr('<div></div>') \nprint(div_NotStr)\n\n<div></div>", + "crumbs": [ + "Home", + "Reference", + "Custom Components" + ] + }, + { + "objectID": "ref/defining_xt_component.html#automatic-creation", + "href": "ref/defining_xt_component.html#automatic-creation", + "title": "Custom Components", + "section": "Automatic Creation", + "text": "Automatic Creation\nThe next (and better) approach is to let FastHTML generate the component function for you. As you can see in our assert this creates a function that creates the HTML just as we wanted. This works even though there is not a Some_never_before_used_tag function in the fasthtml.components source code (you can verify this yourself by looking at the source code).\n\n\n\n\n\n\nTip\n\n\n\nTypically these tags are needed because a CSS or Javascript library created a new XML tag that isn’t default HTML. For example the zero-md javascript library looks for a <zero-md></zero-md> tag to know what to run its javascript code on. Most CSS libraries work by creating styling based on the class attribute, but they can also apply styling to an arbitrary HTML tag that they made up.\n\n\n\nfrom fasthtml.components import Some_never_before_used_tag\n\nSome_never_before_used_tag()\n\n<some-never-before-used-tag></some-never-before-used-tag>", + "crumbs": [ + "Home", + "Reference", + "Custom Components" + ] + }, + { + "objectID": "ref/defining_xt_component.html#manual-creation", + "href": "ref/defining_xt_component.html#manual-creation", + "title": "Custom Components", + "section": "Manual Creation", + "text": "Manual Creation\nThe automatic creation isn’t magic. It’s just calling a python function __getattr__ and you can call it yourself to get the same result.\n\nimport fasthtml\n\nauto_called = fasthtml.components.Some_never_before_used_tag()\nmanual_called = fasthtml.components.__getattr__('Some_never_before_used_tag')()\n\n# Proving they generate the same xml\nassert to_xml(auto_called) == to_xml(manual_called)\n\nKnowing that, we know that it’s possible to create a different function that has different behavior than FastHTMLs default behavior by modifying how the ___getattr__ function creates the components! It’s only a few lines of code and reading that what it does is a great way to understand components more deeply.\n\n\n\n\n\n\nTip\n\n\n\nDunder methods and functions are special functions that have double underscores at the beginning and end of their name. They are called at specific times in python so you can use them to cause customized behavior that makes sense for your specific use case. They can appear magical if you don’t know how python works, but they are extremely commonly used to modify python’s default behavior (__init__ is probably the most common one).\nIn a module __getattr__ is called to get an attribute. In fasthtml.components, this is defined to create components automatically for you.\n\n\nFor example if you want a component that creates <path></path> that doesn’t conflict names with pathlib.Path you can do that. FastHTML automatically creates new components with a 1:1 mapping and a consistent name, which is almost always what you want. But in some cases you may want to customize that and you can use the ft_hx function to do that differently than the default.\n\nfrom fasthtml.common import ft_hx\n\ndef ft_path(*c, target_id=None, **kwargs): \n return ft_hx('path', *c, target_id=target_id, **kwargs)\n\nft_path()\n\n<path></path>\n\n\nWe can add any behavior in that function that we need to, so let’s go through some progressively complex examples that you may need in some of your projects.\n\nUnderscores in tags\nNow that we understand how FastHTML generates components, we can create our own in all kinds of ways. For example, maybe we need a weird HTML tag that uses underscores. FastHTML replaces _ with - in tags because underscores in tags are highly unusual and rarely what you want, though it does come up rarely.\n\ndef tag_with_underscores(*c, target_id=None, **kwargs): \n return ft_hx('tag_with_underscores', *c, target_id=target_id, **kwargs)\n\ntag_with_underscores()\n\n<tag_with_underscores></tag_with_underscores>\n\n\n\n\nSymbols (ie @) in tags\nSometimes you may need to use a tag that uses characters that are not allowed in function names in python (again, very unusual).\n\ndef tag_with_AtSymbol(*c, target_id=None, **kwargs): \n return ft_hx('tag-with-@symbol', *c, target_id=target_id, **kwargs)\n\ntag_with_AtSymbol()\n\n<tag-with-@symbol></tag-with-@symbol>\n\n\n\n\nSymbols (ie @) in tag attributes\nIt also may be that an argument in an HTML tag uses characters that can’t be used in python arguments. To handle these you can define those args using a dictionary.\n\nDiv(normal_arg='normal stuff',**{'notNormal:arg:with_varing@symbols!':'123'})\n\n<div normal-arg=\"normal stuff\" notnormal:arg:with_varing@symbols!=\"123\"></div>", + "crumbs": [ + "Home", + "Reference", + "Custom Components" + ] + }, + { + "objectID": "explains/minidataapi.html", + "href": "explains/minidataapi.html", + "title": "MiniDataAPI Spec", + "section": "", + "text": "The MiniDataAPI is a persistence API specification that designed to be small and relatively easy to implement across a wide range of datastores. While early implementations have been SQL-based, the specification can be quickly implemented in key/value stores, document databases, and more.", + "crumbs": [ + "Home", + "Explanations", + "MiniDataAPI Spec" + ] + }, + { + "objectID": "explains/minidataapi.html#why", + "href": "explains/minidataapi.html#why", + "title": "MiniDataAPI Spec", + "section": "Why?", + "text": "Why?\nThe MiniDataAPI specification allows us to use the same API for many different database engines. Any application using the MiniDataAPI spec for interacting with its database requires no modification beyond import and configuration changes to switch database engines. For example, to convert an application from Fastlite running SQLite to FastSQL running PostgreSQL, should require only changing these two lines:\n\n\nFastLite version\nfrom fastlite import *\ndb = database('test.db')\n\nFastSQL version\nfrom fastsql import *\ndb = Database('postgres:...')\n\n\nAs both libraries adhere to the MiniDataAPI specification, the rest of the code in the application should remain the same. The advantage of the MiniDataAPI spec is that it allows people to use whatever datastores they have access to or prefer.\n\n\n\n\n\n\nNote\n\n\n\nSwitching databases won’t migrate any existing data between databases.\n\n\n\nEasy to learn, quick to implement\nThe MiniDataAPI specification is designed to be easy-to-learn and quick to implement. It focuses on straightforward Create, Read, Update, and Delete (CRUD) operations.\nMiniDataAPI databases aren’t limited to just row-based systems. In fact, the specification is closer in design to a key/value store than a set of records. What’s exciting about this is we can write implementations for tools like Python dict stored as JSON, Redis, and even the venerable ZODB.\n\n\nLimitations of the MiniDataAPI Specification\n\n“Mini refers to the lightweightness of specification, not the data.”\n– Jeremy Howard\n\nThe advantages of the MiniDataAPI come at a cost. The MiniDataAPI specification focuses a very small set of features compared to what can be found in full-fledged ORMs and query languages. It intentionally avoids nuances or sophisticated features.\nThis means the specification does not include joins or formal foreign keys. Complex data stored over multiple tables that require joins isn’t handled well. For this kind of scenario it’s probably for the best to use more sophisticated ORMs or even direct database queries.\n\n\nSummary of the MiniDataAPI Design\n\nEasy-to-learn\nRelative quick to implement for new database engines\nAn API for CRUD operations\nFor many different types of databases including row- and key/value-based designs\nIntentionally small in terms of features: no joins, no foreign keys, no database specific features\nBest for simpler designs, complex architectures will need more sophisticated tools.", + "crumbs": [ + "Home", + "Explanations", + "MiniDataAPI Spec" + ] + }, + { + "objectID": "explains/minidataapi.html#connectconstruct-the-database", + "href": "explains/minidataapi.html#connectconstruct-the-database", + "title": "MiniDataAPI Spec", + "section": "Connect/construct the database", + "text": "Connect/construct the database\nWe connect or construct the database by passing in a string connecting to the database endpoint or a filepath representing the database’s location. While this example is for SQLite running in memory, other databases such as PostgreSQL, Redis, MongoDB, might instead use a URI pointing at the database’s filepath or endpoint. The method of connecting to a DB is not part of this API, but part of the underlying library. For instance, for fastlite:\n\ndb = database(':memory:')\n\nHere’s a complete list of the available methods in the API, all documented below (assuming db is a database and t is a table):\n\ndb.create\nt.insert\nt.delete\nt.update\nt[key]\nt(...)\nt.xtra", + "crumbs": [ + "Home", + "Explanations", + "MiniDataAPI Spec" + ] + }, + { + "objectID": "explains/minidataapi.html#tables", + "href": "explains/minidataapi.html#tables", + "title": "MiniDataAPI Spec", + "section": "Tables", + "text": "Tables\nFor the sake of expediency, this document uses a SQL example. However, tables can represent anything, not just the fundamental construct of a SQL databases. They might represent keys within a key/value structure or files on a hard-drive.\n\nCreating tables\nWe use a create() method attached to Database object (db in our example) to create the tables.\n\nclass User: name:str; email: str; year_started:int\nusers = db.create(User, pk='name')\nusers\n\n<Table user (name, email, year_started)>\n\n\n\nclass User: name:str; email: str; year_started:int\nusers = db.create(User, pk='name')\nusers\n\n<Table user (name, email, year_started)>\n\n\nIf no pk is provided, id is assumed to be the primary key. Regardless of whether you mark a class as a dataclass or not, it will be turned into one – specifically into a flexiclass.\n\n@dataclass\nclass Todo: id: int; title: str; detail: str; status: str; name: str\ntodos = db.create(Todo) \ntodos\n\n<Table todo (id, title, detail, status, name)>\n\n\n\n\nCompound primary keys\nThe MiniData API spec supports compound primary keys, where more than one column is used to identify records. We’ll also use this example to demonstrate creating a table using a dict of keyword arguments.\n\nclass Publication: authors: str; year: int; title: str\npublications = db.create(Publication, pk=('authors', 'year'))\n\n\n\nTransforming tables\nDepending on the database type, this method can include transforms - the ability to modify the tables. Let’s go ahead and add a password field for our table called pwd.\n\nclass User: name:str; email: str; year_started:int; pwd:str\nusers = db.create(User, pk='name', transform=True)\nusers\n\n<Table user (name, email, year_started, pwd)>", + "crumbs": [ + "Home", + "Explanations", + "MiniDataAPI Spec" + ] + }, + { + "objectID": "explains/minidataapi.html#manipulating-data", + "href": "explains/minidataapi.html#manipulating-data", + "title": "MiniDataAPI Spec", + "section": "Manipulating data", + "text": "Manipulating data\nThe specification is designed to provide as straightforward CRUD API (Create, Read, Update, and Delete) as possible. Additional features like joins are out of scope.\n\n.insert()\nAdd a new record to the database. We want to support as many types as possible, for now we have tests for Python classes, dataclasses, and dicts. Returns an instance of the new record.\nHere’s how to add a record using a Python class:\n\nusers.insert(User(name='Braden', email='b@example.com', year_started=2018))\n\nUser(name='Braden', email='b@example.com', year_started=2018, pwd=None)\n\n\nWe can also use keyword arguments directly:\n\nusers.insert(name='Alma', email='a@example.com', year_started=2019)\n\nUser(name='Alma', email='a@example.com', year_started=2019, pwd=None)\n\n\nAnd now Charlie gets added via a Python dict.\n\nusers.insert({'name': 'Charlie', 'email': 'c@example.com', 'year_started': 2018})\n\nUser(name='Charlie', email='c@example.com', year_started=2018, pwd=None)\n\n\nAnd now TODOs. Note that the inserted row is returned:\n\ntodos.insert(Todo(title='Write MiniDataAPI spec', status='open', name='Braden'))\ntodos.insert(title='Implement SSE in FastHTML', status='open', name='Alma')\ntodo = todos.insert(dict(title='Finish development of FastHTML', status='closed', name='Charlie'))\ntodo\n\nTodo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')\n\n\nLet’s do the same with the Publications table.\n\npublications.insert(Publication(authors='Alma', year=2019, title='FastHTML'))\npublications.insert(authors='Alma', year=2030, title='FastHTML and beyond')\npublication= publications.insert((dict(authors='Alma', year=2035, title='FastHTML, the early years')))\npublication\n\nPublication(authors='Alma', year=2035, title='FastHTML, the early years')\n\n\n\n\nSquare bracket search []\nGet a single record by entering a primary key into a table object within square brackets. Let’s see if we can find Alma.\n\nuser = users['Alma']\nuser\n\nUser(name='Alma', email='a@example.com', year_started=2019, pwd=None)\n\n\nIf no record is found, a NotFoundError error is raised. Here we look for David, who hasn’t yet been added to our users table.\n\ntry: users['David']\nexcept NotFoundError: print(f'User not found')\n\nUser not found\n\n\nHere’s a demonstration of a ticket search, demonstrating how this works with non-string primary keys.\n\ntodos[1]\n\nTodo(id=1, title='Write MiniDataAPI spec', detail=None, status='open', name='Braden')\n\n\nCompound primary keys can be supplied in lists or tuples, in the order they were defined. In this case it is the authors and year columns.\nHere’s a query by compound primary key done with a list:\n\npublications[['Alma', 2019]]\n\nPublication(authors='Alma', year=2019, title='FastHTML')\n\n\nHere’s the same query done directly with index args.\n\npublications['Alma', 2030]\n\nPublication(authors='Alma', year=2030, title='FastHTML and beyond')\n\n\n\n\nParentheses search ()\nGet zero to many records by entering values with parentheses searches. If nothing is in the parentheses, then everything is returned.\n\nusers()\n\n[User(name='Braden', email='b@example.com', year_started=2018, pwd=None),\n User(name='Alma', email='a@example.com', year_started=2019, pwd=None),\n User(name='Charlie', email='c@example.com', year_started=2018, pwd=None)]\n\n\nWe can order the results.\n\nusers(order_by='name')\n\n[User(name='Alma', email='a@example.com', year_started=2019, pwd=None),\n User(name='Braden', email='b@example.com', year_started=2018, pwd=None),\n User(name='Charlie', email='c@example.com', year_started=2018, pwd=None)]\n\n\nWe can filter on the results:\n\nusers(where=\"name='Alma'\")\n\n[User(name='Alma', email='a@example.com', year_started=2019, pwd=None)]\n\n\nGenerally you probably want to use placeholders, to avoid SQL injection attacks:\n\nusers(\"name=?\", ('Alma',))\n\n[User(name='Alma', email='a@example.com', year_started=2019, pwd=None)]\n\n\nWe can limit results with the limit keyword:\n\nusers(limit=1)\n\n[User(name='Braden', email='b@example.com', year_started=2018, pwd=None)]\n\n\nIf we’re using the limit keyword, we can also use the offset keyword to start the query later.\n\nusers(limit=5, offset=1)\n\n[User(name='Alma', email='a@example.com', year_started=2019, pwd=None),\n User(name='Charlie', email='c@example.com', year_started=2018, pwd=None)]\n\n\n\n\n.update()\nUpdate an existing record of the database. Must accept Python dict, dataclasses, and standard classes. Uses the primary key for identifying the record to be changed. Returns an instance of the updated record.\nHere’s with a normal Python class:\n\nuser\n\nUser(name='Alma', email='a@example.com', year_started=2019, pwd=None)\n\n\n\nuser.year_started = 2099\nusers.update(user)\n\nUser(name='Alma', email='a@example.com', year_started=2099, pwd=None)\n\n\nOr use a dict:\n\nusers.update(dict(name='Alma', year_started=2199, email='a@example.com'))\n\nUser(name='Alma', email='a@example.com', year_started=2199, pwd=None)\n\n\nOr use kwargs:\n\nusers.update(name='Alma', year_started=2149)\n\nUser(name='Alma', email='a@example.com', year_started=2149, pwd=None)\n\n\nIf the primary key doesn’t match a record, raise a NotFoundError.\nJohn hasn’t started with us yet so doesn’t get the chance yet to travel in time.\n\ntry: users.update(User(name='John', year_started=2024, email='j@example.com'))\nexcept NotFoundError: print('User not found')\n\nUser not found\n\n\n\n\n.delete()\nDelete a record of the database. Uses the primary key for identifying the record to be removed. Returns a table object.\nCharlie decides to not travel in time. He exits our little group.\n\nusers.delete('Charlie')\n\n<Table user (name, email, year_started, pwd)>\n\n\nIf the primary key value can’t be found, raises a NotFoundError.\n\ntry: users.delete('Charlies')\nexcept NotFoundError: print('User not found')\n\nUser not found\n\n\nIn John’s case, he isn’t time travelling with us yet so can’t be removed.\n\ntry: users.delete('John')\nexcept NotFoundError: print('User not found')\n\nUser not found\n\n\nDeleting records with compound primary keys requires providing the entire key.\n\npublications.delete(['Alma' , 2035])\n\n<Table publication (authors, year, title)>\n\n\n\n\nin keyword\nAre Alma and John contained in the Users table? Or, to be technically precise, is the item with the specified primary key value in this table?\n\n'Alma' in users, 'John' in users\n\n(True, False)\n\n\nAlso works with compound primary keys, as shown below. You’ll note that the operation can be done with either a list or tuple.\n\n['Alma', 2019] in publications\n\nTrue\n\n\nAnd now for a False result, where John has no publications.\n\n('John', 1967) in publications\n\nFalse\n\n\n\n\n.xtra()\nIf we set fields within the .xtra function to a particular value, then indexing is also filtered by those. This applies to every database method except for record creation. This makes it easier to limit users (or other objects) access to only things for which they have permission. This is a one-way operation, once set it can’t be undone for a particular table object.\nFor example, if we query all our records below without setting values via the .xtra function, we can see todos for everyone. Pay special attention to the id values of all three records, as we are about to filter most of them away.\n\ntodos()\n\n[Todo(id=1, title='Write MiniDataAPI spec', detail=None, status='open', name='Braden'),\n Todo(id=2, title='Implement SSE in FastHTML', detail=None, status='open', name='Alma'),\n Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')]\n\n\nLet’s use .xtra to constrain results just to Charlie. We set the name field in Todos, but it could be any field defined for this table.\n\ntodos.xtra(name='Charlie')\n\nWe’ve now set a field to a value with .xtra, if we loop over all the records again, only those assigned to records with a name of Charlie will be displayed.\n\ntodos()\n\n[Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')]\n\n\nThe in keyword is also affected. Only records with a name of Charlie will evaluate to be True. Let’s demonstrate by testing it with a Charlie record:\n\nct = todos[3]\nct\n\nTodo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')\n\n\nCharlie’s record has an ID of 3. Here we demonstrate that Charlie’s TODO can be found in the list of todos:\n\nct.id in todos\n\nTrue\n\n\nIf we try in with the other IDs the query fails because the filtering is now set to just records with a name of Charlie.\n\n1 in todos, 2 in todos\n\n(False, False)\n\n\n\ntry: todos[2]\nexcept NotFoundError: print('Record not found')\n\nRecord not found\n\n\nWe are also constrained by what records we can update. In the following example we try to update a TODO not named ‘Charlie’. Because the name is wrong, the .update function will raise a NotFoundError.\n\ntry: todos.update(Todo(id=1, title='Finish MiniDataAPI Spec', status='closed', name='Braden'))\nexcept NotFoundError as e: print('Record not updated')\n\nRecord not updated\n\n\nUnlike poor Braden, Charlie isn’t filtered out. Let’s update his TODO.\n\ntodos.update(Todo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie'))\n\nTodo(id=3, title='Finish development of FastHTML', detail=None, status='closed', name='Charlie')\n\n\nFinally, once constrained by .xtra, only records with Charlie as the name can be deleted.\n\ntry: todos.delete(1)\nexcept NotFoundError as e: print('Record not updated')\n\nRecord not updated\n\n\nCharlie’s TODO was to finish development of FastHTML. While the framework will stabilize, like any good project it will see new features added and the odd bug corrected for many years to come. Therefore, Charlie’s TODO is nonsensical. Let’s delete it.\n\ntodos.delete(ct.id)\n\n<Table todo (id, title, detail, status, name)>\n\n\nWhen a TODO is inserted, the xtra fields are automatically set. This ensures that we don’t accidentally, for instance, insert items for others users. Note that here we don’t set the name field, but it’s still included in the resultant row:\n\nct = todos.insert(Todo(title='Rewrite personal site in FastHTML', status='open'))\nct\n\nTodo(id=3, title='Rewrite personal site in FastHTML', detail=None, status='open', name='Charlie')\n\n\nIf we try to change the username to someone else, the change is ignored, due to xtra:\n\nct.name = 'Braden'\ntodos.update(ct)\n\nTodo(id=3, title='Rewrite personal site in FastHTML', detail=None, status='open', name='Charlie')", + "crumbs": [ + "Home", + "Explanations", + "MiniDataAPI Spec" + ] + }, + { + "objectID": "explains/minidataapi.html#sql-first-design", + "href": "explains/minidataapi.html#sql-first-design", + "title": "MiniDataAPI Spec", + "section": "SQL-first design", + "text": "SQL-first design\n\nusers = None\nUser = None\n\n\nusers = db.t.user\nusers\n\n<Table user (name, email, year_started, pwd)>\n\n\n(This section needs to be documented properly.)\nFrom the table objects we can extract a Dataclass version of our tables. Usually this is given an singular uppercase version of our table name, which in this case is User.\n\nUser = users.dataclass()\n\n\nUser(name='Braden', email='b@example.com', year_started=2018)\n\nUser(name='Braden', email='b@example.com', year_started=2018, pwd=UNSET)", + "crumbs": [ + "Home", + "Explanations", + "MiniDataAPI Spec" + ] + }, + { + "objectID": "explains/minidataapi.html#implementations", + "href": "explains/minidataapi.html#implementations", + "title": "MiniDataAPI Spec", + "section": "Implementations", + "text": "Implementations\n\nImplementing MiniDataAPI for a new datastore\nFor creating new implementations, the code examples in this specification are the test case for the API. New implementations should pass the tests in order to be compliant with the specification.\n\n\nImplementations\n\nfastlite - The original implementation, only for Sqlite\nfastsql - An SQL database agnostic implementation based on the excellent SQLAlchemy library.", + "crumbs": [ + "Home", + "Explanations", + "MiniDataAPI Spec" + ] + }, + { + "objectID": "explains/websockets.html", + "href": "explains/websockets.html", + "title": "WebSockets", + "section": "", + "text": "Websockets are a protocol for two-way, persistent communication between a client and server. This is different from HTTP, which uses a request/response model where the client sends a request and the server responds. With websockets, either party can send messages at any time, and the other party can respond.\nThis allows for different applications to be built, including things like chat apps, live-updating dashboards, and real-time collaborative tools, which would require constant polling of the server for updates with HTTP.\nIn FastHTML, you can create a websocket route using the @app.ws decorator. This decorator takes a route path, and optional conn and disconn parameters representing the on_connect and on_disconnect callbacks in websockets, respectively. The function decorated by @app.ws is the main function that is called when a message is received.\nHere’s an example of a basic websocket route:\nThe on_message function is the main function that is called when a message is received and can be named however you like. Similar to standard routes, the arguments to on_message are automatically parsed from the websocket payload for you, so you don’t need to manually parse the message content. However, certain argument names are reserved for special purposes. Here are the most important ones:\nFor example, we can send a message to the client that just connected like this:\nOr if we receive a message from the client, we can send a message back to them:\nOn the client side, we can use HTMX’s websocket extension to open a websocket connection and send/receive messages. For example:\nThis will create a websocket connection to the server on route /ws, and send any form submissions to the server via the websocket. The server will then respond by sending a message back to the client. The client will then update the message div with the message from the server using Out of Band Swaps, which means that the content is swapped with the same id without reloading the page.\nPutting it all together, the code for the client and server should look like this:\nThis is a fairly simple example and could be done just as easily with standard HTTP requests, but it illustrates the basic idea of how websockets work. Let’s look at a more complex example next.", + "crumbs": [ + "Home", + "Explanations", + "WebSockets" + ] + }, + { + "objectID": "explains/websockets.html#session-data-in-websockets", + "href": "explains/websockets.html#session-data-in-websockets", + "title": "WebSockets", + "section": "Session data in Websockets", + "text": "Session data in Websockets\nSession data is shared between standard HTTP routes and Websockets. This means you can access, for example, logged in user ID inside websocket handler:\nfrom fasthtml.common import *\n\napp = FastHTML(exts='ws')\nrt = app.route\n\n@rt('/login')\ndef get(session):\n session[\"person\"] = \"Bob\"\n return \"ok\"\n\n@app.ws('/ws')\nasync def ws(msg:str, send, session):\n await send(Div(f'Hello {session.get(\"person\")}' + msg, id='notifications'))\n\nserve()", + "crumbs": [ + "Home", + "Explanations", + "WebSockets" + ] + }, + { + "objectID": "explains/websockets.html#real-time-chat-app", + "href": "explains/websockets.html#real-time-chat-app", + "title": "WebSockets", + "section": "Real-Time Chat App", + "text": "Real-Time Chat App\nLet’s put our new websocket knowledge to use by building a simple chat app. We will create a chat app where multiple users can send and receive messages in real time.\nLet’s start by defining the app and the home page:\nfrom fasthtml.common import *\n\napp = FastHTML(exts='ws')\nrt = app.route\n\nmsgs = []\n@rt('/')\ndef home(): return Div(\n Div(Ul(*[Li(m) for m in msgs], id='msg-list')),\n Form(Input(id='msg'), id='form', ws_send=True),\n hx_ext='ws', ws_connect='/ws')\nNow, let’s handle the websocket connection. We’ll add a new route for this along with an on_conn and on_disconn function to keep track of the users currently connected to the websocket. Finally, we will handle the logic for sending messages to all connected users.\nusers = {}\ndef on_conn(ws, send): users[str(id(ws))] = send\ndef on_disconn(ws): users.pop(str(id(ws)), None)\n\n@app.ws('/ws', conn=on_conn, disconn=on_disconn)\nasync def ws(msg:str):\n msgs.append(msg)\n # Use associated `send` function to send message to each user\n for u in users.values(): await u(Ul(*[Li(m) for m in msgs], id='msg-list'))\n\nserve()\nWe can now run this app with python chat_ws.py and open multiple browser tabs to http://localhost:5001. You should be able to send messages in one tab and see them appear in the other tabs.\n\nA Work in Progress\nThis page (and Websocket support in FastHTML) is a work in progress. Questions, PRs, and feedback are welcome!", + "crumbs": [ + "Home", + "Explanations", + "WebSockets" + ] + }, + { + "objectID": "explains/oauth.html", + "href": "explains/oauth.html", + "title": "OAuth", + "section": "", + "text": "OAuth is an open standard for ‘access delegation’, commonly used as a way for Internet users to grant websites or applications access to their information on other websites but without giving them the passwords. It is the mechanism that enables “Log in with Google” on many sites, saving you from having to remember and manage yet another password. Like many auth-related topics, there’s a lot of depth and complexity to the OAuth standard, but once you understand the basic usage it can be a very convenient alternative to managing your own user accounts.\nOn this page you’ll see how to use OAuth with FastHTML to implement some common pieces of functionality.", + "crumbs": [ + "Home", + "Explanations", + "OAuth" + ] + }, + { + "objectID": "explains/oauth.html#creating-an-client", + "href": "explains/oauth.html#creating-an-client", + "title": "OAuth", + "section": "Creating an Client", + "text": "Creating an Client\nFastHTML has Client classes for managing settings and state for different OAuth providers. Currently implemented are: GoogleAppClient, GitHubAppClient, HuggingFaceClient and DiscordAppClient - see the source if you need to add other providers. You’ll need a client_id and client_secret from the provider (see the from-scratch example later in this page for an example of registering with GitHub) to create the client. We recommend storing these in environment variables, rather than hardcoding them in your code.\n\nimport os\nfrom fasthtml.oauth import GoogleAppClient\nclient = GoogleAppClient(os.getenv(\"AUTH_CLIENT_ID\"),\n os.getenv(\"AUTH_CLIENT_SECRET\"))\n\nThe client is used to obtain a login link and to manage communications between your app and the OAuth provider (client.login_link(redirect_uri=\"/redirect\")).", + "crumbs": [ + "Home", + "Explanations", + "OAuth" + ] + }, + { + "objectID": "explains/oauth.html#using-the-oauth-class", + "href": "explains/oauth.html#using-the-oauth-class", + "title": "OAuth", + "section": "Using the OAuth class", + "text": "Using the OAuth class\nOnce you’ve set up a client, adding OAuth to a FastHTML app can be as simple as:\n\nfrom fasthtml.oauth import OAuth\nfrom fasthtml.common import FastHTML, RedirectResponse\n\nclass Auth(OAuth):\n def get_auth(self, info, ident, session, state):\n email = info.email or ''\n if info.email_verified and email.split('@')[-1]=='answer.ai':\n return RedirectResponse('/', status_code=303)\n\napp = FastHTML()\noauth = Auth(app, client)\n\n@app.get('/')\ndef home(auth): return P('Logged in!'), A('Log out', href='/logout')\n\n@app.get('/login')\ndef login(req): return Div(P(\"Not logged in\"), A('Log in', href=oauth.login_link(req)))\n\nThere’s a fair bit going on here, so let’s unpack what’s happening in that code:\n\nOAuth (and by extension our custom Auth class) has a number of default arguments, including some key URLs: redir_path='/redirect', error_path='/error', logout_path='/logout', login_path='/login'. It will create and handle the redirect and logout paths, and it’s up to you to handle /login (where unsuccessful login attempts will be redirected) and /error (for oauth errors).\nWhen we run oauth = Auth(app, client) it adds the redirect and logout paths to the app and also adds some beforeware. This beforeware runs on any requests (apart from any specified with the skip parameter).\n\nThe added beforeware specifies some app behaviour:\n\nIf someone who isn’t logged in attempts to visit our homepage (/) here, they will be redirected to /login.\nIf they are logged in, it calls a check_invalid method. This defaults to False, which let’s the user continue to the page they requested. The behaviour can be modified by defining your own check_invalid method in the Auth class - for example, you could have this forcibly log out users who have recently been banned.\n\nSo how does someone log in? If they visit (or are redirected to) the login page at /login, we show them a login link. This sends them to the OAuth provider, where they’ll go through the steps of selecting their account, giving permissions etc. Once done they will be redirected back to /redirect. Behind the scenes a code that comes as part of their request gets turned into user info, which is then passed to the key function get_auth(self, info, ident, session, state). Here is where you’d handle looking up or adding a user in a database, checking for some condition (for example, this code checks if the email is an answer.ai email address) or choosing the destination based on state. The arguments are:\n\nself: the Auth object, which you can use to access the client (self.cli)\ninfo: the information provided by the OAuth provider, typically including a unique user id, email address, username and other metadata.\nident: a unique identifier for this user. What this looks like varies between providers. This is useful for managing a database of users, for example.\nsession: the current session, that you can store information in securely\nstate: you can optionally pass in some state when creating the login link. This persists and is returned after the user goes through the Oath steps, which is useful for returning them to the same page they left. It can also be used as added security against CSRF attacks.\n\nIn our example, we check the email in info (we use a GoogleAppClient, not all providers will include an email). If we aren’t happy, and get_auth returns False or nothing (as in the case here for non-answerai people) then the user is redirected back to the login page. But if everything looks good we return a redirect to the homepage, and an auth key is added to the session and the scope containing the users identity ident. So, for example, in the homepage route we could use auth to look up this particular user’s profile info and customize the page accordingly. This auth will persist in their session until they clear the browser cache, so by default they’ll stay logged in. To log them out, remove it ( session.pop('auth', None)) or send them to /logout which will do that for you.", + "crumbs": [ + "Home", + "Explanations", + "OAuth" + ] + }, + { + "objectID": "explains/oauth.html#explaining-oauth-with-a-from-scratch-implementation", + "href": "explains/oauth.html#explaining-oauth-with-a-from-scratch-implementation", + "title": "OAuth", + "section": "Explaining OAuth with a from-scratch implementation", + "text": "Explaining OAuth with a from-scratch implementation\nHopefully the example above is enough to get you started. You can also check out the (fairly minimal) source code where this is implemented, and the examples here.\nIf you’re wanting to learn more about how this works, and to see where you might add additional functionality, the rest of this page will walk through some examples without the OAuth convenience class, to illustrate the concepts. This was writted before said OAuth class was available, and is kep here for educational purposes - we recommend you stick with the new approach shown above in most cases.", + "crumbs": [ + "Home", + "Explanations", + "OAuth" + ] + }, + { + "objectID": "explains/oauth.html#a-minimal-login-flow-github", + "href": "explains/oauth.html#a-minimal-login-flow-github", + "title": "OAuth", + "section": "A Minimal Login Flow (GitHub)", + "text": "A Minimal Login Flow (GitHub)\nLet’s begin by building a minimal ‘Sign in with GitHub’ flow. This will demonstrate the basic steps of OAuth.\nOAuth requires a “provider” (in this case, GitHub) to authenticate the user. So the first step when setting up our app is to register with GitHub to set things up.\nGo to https://github.com/settings/developers and click “New OAuth App”. Fill in the form with the following values, then click ‘Register application’.\n\nApplication name: Your app name\nHomepage URL: http://localhost:8000 (or whatever URL you’re using - you can change this later)\nAuthorization callback URL: http://localhost:8000/auth_redirect (you can modify this later too)\n\n\n\n\nAfter you register, you’ll see a screen where you can view the client ID and generate a client secret. Store these values in a safe place. You’ll use them to create a GitHubAppClient object in FastHTML.\nThis client object is responsible for handling the parts of the OAuth flow which depend on direct communication between your app and GitHub, as opposed to interactions which go through the user’s browser via redirects.\nHere is how to setup the client object:\nclient = GitHubAppClient(\n client_id=\"your_client_id\",\n client_secret=\"your_client_secret\"\n)\nYou should also save the path component of the authorization callback URL which you provided on registration.\nThis route is where GitHub will redirect the user’s browser in order to send an authorization code to your app. You should save only the URL’s path component rather than the entire URL because you want your code to work automatically in deployment, when the host and port part of the URL change from localhost:8000 to your real DNS name.\nSave the special authorization callback path under an obvious name:\nauth_callback_path = \"/auth_redirect\"\n\n\n\n\n\n\nNote\n\n\n\nIt’s recommended to store the client ID, and secret, in environment variables, rather than hardcoding them in your code.\n\n\nWhen the user visit a normal page of your app, if they are not already logged in, then you’ll want to redirect them to your app’s login page, which will live at the /login path. We accomplish that by using this piece of “beforeware”, which defines logic which runs before other work for all routes except ones we specify to be skipped:\ndef before(req, session):\n auth = req.scope['auth'] = session.get('user_id', None)\n if not auth: return RedirectResponse('/login', status_code=303)\n counts.xtra(name=auth)\nbware = Beforeware(before, skip=['/login', auth_callback_path])\nWe configure the beforeware to skip /login because that’s where the user goes to login, and we also skip the special authorization callback path because that is used by OAuth itself to receive information from GitHub.\nIt’s only at your login page that we start the OAuth flow. To start the OAuth flow, you need to give the user a link to GitHub’s login for your app. You’ll need the client object to generate that link, and the client object will in turn need the full authorization callback URL, which we need to build from the authorization callback path, so it is a multi-step process to produce this GitHub login link.\nHere is an implementation of your own /login route handler. It generates the GitHub login link and presents it to the user:\n@app.get('/login')\ndef login(request)\n redir = redir_url(request,auth_callback_path)\n login_link = client.login_link(redir)\n return P(A('Login with GitHub', href=login_link)) \nOnce the user follows that link, GitHub will ask them to grant permission to your app to access their GitHub account. If they agree, GitHub will redirect them back to your app’s authorization callback URL, carrying an authorization code which your app can use to generate an access token. To receive this code, you need to set up a route in FastHTML that listens for requests at the authorization callback path. For example:\n@app.get(auth_callback_path)\ndef auth_redirect(code:str):\n return P(f\"code: {code}\")\nThis authorization code is temporary, and is used by your app to directly ask the provider for user information like an access token.\nTo recap, you can think of the exchange so far as:\n\nUser to us: “I want to log in with you, app.”\nUs to User: “Okay but first, here’s a special link to log in with GitHub”\nUser to GitHub: “I want to log in with you, GitHub, to use this app.”\nGitHub to User: “OK, redirecting you back to the app’s URL (with an auth code)”\nUser to Us: “Hi again, app. Here’s the GitHub auth code you need to ask GitHub for info about me” (delivered via /auth_redirect?code=...)\n\nThe final steps we need to implement are as follows:\n\nUs to GitHUb: “A user just gave me this auth code. May I have the user info (e.g., an access token)?”\nGitHub to us: “Since you have an auth code, here’s the user info”\n\nIt’s critical for us to derive the user info from the auth code immediately in the authorization callback, because the auth code may be used only once. So we use it that once in order to get information like an access token, which will remain valid for longer.\nTo go from the auth code to user info, you use info = client.retr_info(code,redirect_uri). From the user info, you can extract the user_id, which is a unique identifier for the user:\n@app.get(auth_callback_path)\ndef auth_redirect(code:str, request):\n redir = redir_url(request, auth_callback_path)\n user_info = client.retr_info(code, redir)\n user_id = info[client.id_key]\n return P(f\"User id: {user_id}\")\nBut we want the user ID not to print it but to remember the user.\nSo let us store it in the session object, to remember who is logged in:\n@app.get(auth_callback_path)\ndef auth_redirect(code:str, request, session):\n redir = redir_url(request, auth_callback_path)\n user_info = client.retr_info(code, redir)\n user_id = user_info[client.id_key] # get their ID\n session['user_id'] = user_id # save ID in the session\n return RedirectResponse('/', status_code=303)\nThe session object is derived from values visible to the user’s browser, but it is cryptographically signed so the user can’t read it themselves. This makes it safe to store even information we don’t want to expose to the user.\nFor larger quantities of data, we’d want to save that information in a database and use the session to hold keys to lookup information from that database.\nHere’s a minimal app that puts all these pieces together. It uses the user info to get the user_id. It stores that in the session object. It then uses the user_id as a key into a database, which tracks how frequently every user has hit an increment button.\nimport os\nfrom fasthtml.common import *\nfrom fasthtml.oauth import GitHubAppClient, redir_url\n\ndb = database('data/counts.db')\ncounts = db.t.counts\nif counts not in db.t: counts.create(dict(name=str, count=int), pk='name')\nCount = counts.dataclass()\n\n# Auth client setup for GitHub\nclient = GitHubAppClient(os.getenv(\"AUTH_CLIENT_ID\"), \n os.getenv(\"AUTH_CLIENT_SECRET\"))\nauth_callback_path = \"/auth_redirect\"\n\ndef before(req, session):\n # if not logged in, we send them to our login page\n # logged in means:\n # - 'user_id' in the session object, \n # - 'auth' in the request object\n auth = req.scope['auth'] = session.get('user_id', None)\n if not auth: return RedirectResponse('/login', status_code=303)\n counts.xtra(name=auth)\nbware = Beforeware(before, skip=['/login', auth_callback_path])\n\napp = FastHTML(before=bware)\n\n# User asks us to Login\n@app.get('/login')\ndef login(request):\n redir = redir_url(request,auth_callback_path)\n login_link = client.login_link(redir)\n # we tell user to login at github\n return P(A('Login with GitHub', href=login_link)) \n\n# User comes back to us with an auth code from Github\n@app.get(auth_callback_path)\ndef auth_redirect(code:str, request, session):\n redir = redir_url(request, auth_callback_path)\n user_info = client.retr_info(code, redir)\n user_id = user_info[client.id_key] # get their ID\n session['user_id'] = user_id # save ID in the session\n # create a db entry for the user\n if user_id not in counts: counts.insert(name=user_id, count=0)\n return RedirectResponse('/', status_code=303)\n\n@app.get('/')\ndef home(auth):\n return Div(\n P(\"Count demo\"),\n P(f\"Count: \", Span(counts[auth].count, id='count')),\n Button('Increment', hx_get='/increment', hx_target='#count'),\n P(A('Logout', href='/logout'))\n )\n\n@app.get('/increment')\ndef increment(auth):\n c = counts[auth]\n c.count += 1\n return counts.upsert(c).count\n\n@app.get('/logout')\ndef logout(session):\n session.pop('user_id', None)\n return RedirectResponse('/login', status_code=303)\n\nserve()\nSome things to note:\n\nThe before function is used to check if the user is authenticated. If not, they are redirected to the login page.\nTo log the user out, we remove the user ID from the session.\nCalling counts.xtra(name=auth) ensures that only the row corresponding to the current user is accessible when responding to a request. This is often nicer than trying to remember to filter the data in every route, and lowers the risk of accidentally leaking data.\nIn the auth_redirect route, we store the user ID in the session and create a new row in the user_counts table if it doesn’t already exist.\n\nYou can find more heavily-commented version of this code in the oauth directory in fasthtml-example, along with an even more minimal example. More examples may be added in the future.\n\nRevoking Tokens (Google)\nWhen the user in the example above logs out, we remove their user ID from the session. However, the user is still logged in to GitHub. If they click ‘Login with GitHub’ again, they’ll be redirected back to our site without having to log in again. This is because GitHub remembers that they’ve already granted our app permission to access their account. Most of the time this is convenient, but for testing or security purposes you may want a way to revoke this permission.\nAs a user, you can usually revoke access to an app from the provider’s website (for example, https://github.com/settings/applications). But as a developer, you can also revoke access programmatically - at least with some providers. This requires keeping track of the access token (stored in client.token[\"access_token\"] after you call retr_info), and sending a request to the provider’s revoke URL:\nauth_revoke_url = \"https://accounts.google.com/o/oauth2/revoke\"\ndef revoke_token(token):\n response = requests.post(auth_revoke_url, params={\"token\": token})\n return response.status_code == 200 # True if successful\nNot all providers support token revocation, and it is not built into FastHTML clients at the moment.\n\n\nUsing State (Hugging Face)\nImagine a user (not logged in) comes to your AI image editing site, starts testing things out, and then realizes they need to sign in before they can click “Run (Pro)” on the edit they’re working on. They click “Sign in with Hugging Face”, log in, and are redirected back to your site. But now they’ve lost their in-progress edit and are left just looking at the homepage! This is an example of a case where you might want to keep track of some additional state. Another strong use case for being able to pass some uniqie state through the OAuth flow is to prevent something called a CSRF attack. To add a state string to the OAuth flow, you can use client.login_link_with_state(state) instead of client.login_link(), like so:\n# in login page:\nlink = A('Login with GitHub', href=client.login_link_with_state(state='current_prompt: add a unicorn'))\n\n# in auth_redirect:\n@app.get('/auth_redirect')\ndef auth_redirect(code:str, session, state:str=None):\n print(f\"state: {state}\") # Use as needed\n ...\nThe state string is passed through the OAuth flow and back to your site.\n\n\nA Work in Progress\nThis page (and OAuth support in FastHTML) is a work in progress. Questions, PRs, and feedback are welcome!", + "crumbs": [ + "Home", + "Explanations", + "OAuth" + ] + }, + { + "objectID": "api/components.html", + "href": "api/components.html", + "title": "Components", + "section": "", + "text": "from lxml import html as lx\nfrom pprint import pprint", + "crumbs": [ + "Home", + "Source", + "Components" + ] + }, + { + "objectID": "api/components.html#tests", + "href": "api/components.html#tests", + "title": "Components", + "section": "Tests", + "text": "Tests\n\ntest_html2ft('<input value=\"Profit\" name=\"title\" id=\"title\" class=\"char\">', attr1st=True)\ntest_html2ft('<input value=\"Profit\" name=\"title\" id=\"title\" class=\"char\">')\ntest_html2ft('<div id=\"foo\"></div>')\ntest_html2ft('<div id=\"foo\">hi</div>')\ntest_html2ft('<div x-show=\"open\" x-transition:enter=\"transition duration-300\" x-transition:enter-start=\"opacity-0 scale-90\">Hello 👋</div>')\ntest_html2ft('<div x-transition:enter.scale.80 x-transition:leave.scale.90>hello</div>')\n\n\nassert html2ft('<div id=\"foo\">hi</div>', attr1st=True) == \"Div(id='foo')('hi')\"\nassert html2ft(\"\"\"\n <div x-show=\"open\" x-transition:enter=\"transition duration-300\" x-transition:enter-start=\"opacity-0 scale-90\">Hello 👋</div>\n\"\"\") == \"Div('Hello 👋', x_show='open', **{'x-transition:enter': 'transition duration-300', 'x-transition:enter-start': 'opacity-0 scale-90'})\"\nassert html2ft('<div x-transition:enter.scale.80 x-transition:leave.scale.90>hello</div>') == \"Div('hello', **{'x-transition:enter.scale.80': True, 'x-transition:leave.scale.90': True})\"\nassert html2ft(\"<img alt=' ' />\") == \"Img(alt=' ')\"", + "crumbs": [ + "Home", + "Source", + "Components" + ] + }, + { + "objectID": "api/jupyter.html", + "href": "api/jupyter.html", + "title": "Jupyter compatibility", + "section": "", + "text": "from httpx import get, AsyncClient", + "crumbs": [ + "Home", + "Source", + "Jupyter compatibility" + ] + }, + { + "objectID": "api/jupyter.html#helper-functions", + "href": "api/jupyter.html#helper-functions", + "title": "Jupyter compatibility", + "section": "Helper functions", + "text": "Helper functions\n\nsource\n\nnb_serve\n\n nb_serve (app, log_level='error', port=8000, host='0.0.0.0', **kwargs)\n\nStart a Jupyter compatible uvicorn server with ASGI app on port with log_level\n\nsource\n\n\nnb_serve_async\n\n nb_serve_async (app, log_level='error', port=8000, host='0.0.0.0',\n **kwargs)\n\nAsync version of nb_serve\n\nsource\n\n\nis_port_free\n\n is_port_free (port, host='localhost')\n\nCheck if port is free on host\n\nsource\n\n\nwait_port_free\n\n wait_port_free (port, host='localhost', max_wait=3)\n\nWait for port to be free on host", + "crumbs": [ + "Home", + "Source", + "Jupyter compatibility" + ] + }, + { + "objectID": "api/jupyter.html#using-fasthtml-in-jupyter", + "href": "api/jupyter.html#using-fasthtml-in-jupyter", + "title": "Jupyter compatibility", + "section": "Using FastHTML in Jupyter", + "text": "Using FastHTML in Jupyter\n\nsource\n\nshow\n\n show (*s)\n\nSame as fasthtml.components.show, but also adds htmx.process()\n\nsource\n\n\nrender_ft\n\n render_ft ()\n\n\nsource\n\n\nhtmx_config_port\n\n htmx_config_port (port=8000)\n\n\nsource\n\n\nJupyUvi\n\n JupyUvi (app, log_level='error', host='0.0.0.0', port=8000, start=True,\n **kwargs)\n\nStart and stop a Jupyter compatible uvicorn server with ASGI app on port with log_level\nCreating an object of this class also starts the Uvicorn server. It runs in a separate thread, so you can use normal HTTP client functions in a notebook.\n\napp = FastHTML()\nrt = app.route\n\n@app.route\ndef index(): return 'hi'\n\nport = 8000\nserver = JupyUvi(app, port=port)\n\n\n\n\n\n\nget(f'http://localhost:{port}').text\n\n'hi'\n\n\nYou can stop the server, modify routes, and start the server again without restarting the notebook or recreating the server or application.\n\n\nUsing a notebook as a web app\nYou can also run an HTMX web app directly in a notebook. To make this work, you have to add the default FastHTML headers to the DOM of the notebook with show(*def_hdrs()). Additionally, you might find it convenient to use auto_id mode, in which the ID of an FT object is automatically generated if not provided.\n\nfh_cfg['auto_id' ]=True\n\nAfter importing fasthtml.jupyter and calling render_ft(), FT components render directly in the notebook.\n\n(c := Div('Cogito ergo sum'))\n\n\n\nCogito ergo sum\n\n\n\n\n\nHandlers are written just like a regular web app:\n\n@rt\ndef hoho(): return P('loaded!'), Div('hee hee', id=c, hx_swap_oob='true')\n\nAll the usual hx_* attributes can be used:\n\nP('not loaded', hx_get=hoho, hx_trigger='load')\n\n\n\nnot loaded\n\n\n\n\n\nFT components can be used directly both as id values and as hx_target values.\n\n(c := Div(''))\n\n\n\n\n\n\n\n\n\n\n@rt\ndef foo(): return Div('foo bar')\nP('hi', hx_get=foo, hx_trigger='load', hx_target=c)\n\n\n\nhi\n\n\n\n\n\n\nserver.stop()\n\n\n\nRunning apps in an IFrame\nUsing an IFrame can be a good idea to get complete isolation of the styles and scripts in an app. The HTMX function creates an auto-sizing IFrame for a web app.\n\nsource\n\n\nHTMX\n\n HTMX (path='', app=None, host='localhost', port=8000, height='auto',\n link=False, iframe=True)\n\nAn iframe which displays the HTMX application in a notebook.\n\n@rt\ndef index():\n return Div(\n P(A('Click me', hx_get=update, hx_target='#result')),\n P(A('No me!', hx_get=update, hx_target='#result')),\n Div(id='result'))\n\n@rt\ndef update(): return Div(P('Hi!'),P('There!'))\n\n\nserver.start()\n\n\n# Run the notebook locally to see the HTMX iframe in action\nHTMX()\n\n \n\n\n\nserver.stop()\n\n\nsource\n\n\nws_client\n\n ws_client (app, nm='', host='localhost', port=8000, ws_connect='/ws',\n frame=True, link=True, **kwargs)", + "crumbs": [ + "Home", + "Source", + "Jupyter compatibility" + ] + }, + { + "objectID": "api/cli.html", + "href": "api/cli.html", + "title": "Command Line Tools", + "section": "", + "text": "source\n\nrailway_link\n\n railway_link ()\n\nLink the current directory to the current project’s Railway service\n\nsource\n\n\nrailway_deploy\n\n railway_deploy (name:str, mount:<function bool_arg>=True)\n\nDeploy a FastHTML app to Railway\n\n\n\n\nType\nDefault\nDetails\n\n\n\n\nname\nstr\n\nThe project name to deploy\n\n\nmount\nbool_arg\nTrue\nCreate a mounted volume at /app/data?", + "crumbs": [ + "Home", + "Source", + "Command Line Tools" + ] + }, + { + "objectID": "api/core.html", + "href": "api/core.html", + "title": "Core", + "section": "", + "text": "This is the source code to fasthtml. You won’t need to read this unless you want to understand how things are built behind the scenes, or need full details of a particular API. The notebook is converted to the Python module fasthtml/core.py using nbdev.", + "crumbs": [ + "Home", + "Source", + "Core" + ] + }, + { + "objectID": "api/core.html#imports-and-utils", + "href": "api/core.html#imports-and-utils", + "title": "Core", + "section": "Imports and utils", + "text": "Imports and utils\n\nimport time\n\nfrom IPython import display\nfrom enum import Enum\nfrom pprint import pprint\n\nfrom fastcore.test import *\nfrom starlette.testclient import TestClient\nfrom starlette.requests import Headers\nfrom starlette.datastructures import UploadFile\n\nWe write source code first, and then tests come after. The tests serve as both a means to confirm that the code works and also serves as working examples. The first exported function, parsed_date, is an example of this pattern.\n\nsource\n\nparsed_date\n\n parsed_date (s:str)\n\nConvert s to a datetime\n\nparsed_date('2pm')\n\ndatetime.datetime(2025, 1, 12, 14, 0)\n\n\n\nisinstance(date.fromtimestamp(0), date)\n\nTrue\n\n\n\nsource\n\n\nsnake2hyphens\n\n snake2hyphens (s:str)\n\nConvert s from snake case to hyphenated and capitalised\n\nsnake2hyphens(\"snake_case\")\n\n'Snake-Case'\n\n\n\nsource\n\n\nHtmxHeaders\n\n HtmxHeaders (boosted:str|None=None, current_url:str|None=None,\n history_restore_request:str|None=None, prompt:str|None=None,\n request:str|None=None, target:str|None=None,\n trigger_name:str|None=None, trigger:str|None=None)\n\n\ndef test_request(url: str='/', headers: dict={}, method: str='get') -> Request:\n scope = {\n 'type': 'http',\n 'method': method,\n 'path': url,\n 'headers': Headers(headers).raw,\n 'query_string': b'',\n 'scheme': 'http',\n 'client': ('127.0.0.1', 8000),\n 'server': ('127.0.0.1', 8000),\n }\n receive = lambda: {\"body\": b\"\", \"more_body\": False}\n return Request(scope, receive)\n\n\nh = test_request(headers=Headers({'HX-Request':'1'}))\n_get_htmx(h.headers)\n\nHtmxHeaders(boosted=None, current_url=None, history_restore_request=None, prompt=None, request='1', target=None, trigger_name=None, trigger=None)", + "crumbs": [ + "Home", + "Source", + "Core" + ] + }, + { + "objectID": "api/core.html#request-and-response", + "href": "api/core.html#request-and-response", + "title": "Core", + "section": "Request and response", + "text": "Request and response\n\ntest_eq(_fix_anno(Union[str,None], 'a'), 'a')\ntest_eq(_fix_anno(float, 0.9), 0.9)\ntest_eq(_fix_anno(int, '1'), 1)\ntest_eq(_fix_anno(int, ['1','2']), 2)\ntest_eq(_fix_anno(list[int], ['1','2']), [1,2])\ntest_eq(_fix_anno(list[int], '1'), [1])\n\n\nd = dict(k=int, l=List[int])\ntest_eq(_form_arg('k', \"1\", d), 1)\ntest_eq(_form_arg('l', \"1\", d), [1])\ntest_eq(_form_arg('l', [\"1\",\"2\"], d), [1,2])\n\n\nsource\n\nHttpHeader\n\n HttpHeader (k:str, v:str)\n\n\n_to_htmx_header('trigger_after_settle')\n\n'HX-Trigger-After-Settle'\n\n\n\nsource\n\n\nHtmxResponseHeaders\n\n HtmxResponseHeaders (location=None, push_url=None, redirect=None,\n refresh=None, replace_url=None, reswap=None,\n retarget=None, reselect=None, trigger=None,\n trigger_after_settle=None, trigger_after_swap=None)\n\nHTMX response headers\n\nHtmxResponseHeaders(trigger_after_settle='hi')\n\nHttpHeader(k='HX-Trigger-After-Settle', v='hi')\n\n\n\nsource\n\n\nform2dict\n\n form2dict (form:starlette.datastructures.FormData)\n\nConvert starlette form data to a dict\n\nd = [('a',1),('a',2),('b',0)]\nfd = FormData(d)\nres = form2dict(fd)\ntest_eq(res['a'], [1,2])\ntest_eq(res['b'], 0)\n\n\nsource\n\n\nparse_form\n\n parse_form (req:starlette.requests.Request)\n\nStarlette errors on empty multipart forms, so this checks for that situation\n\nasync def f(req):\n def _f(p:HttpHeader): ...\n p = first(_params(_f).values())\n result = await _from_body(req, p)\n return JSONResponse(result.__dict__)\n\nclient = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])]))\n\nd = dict(k='value1',v=['value2','value3'])\nresponse = client.post('/', data=d)\nprint(response.json())\n\n{'k': 'value1', 'v': 'value3'}\n\n\n\nasync def f(req): return Response(str(req.query_params.getlist('x')))\nclient = TestClient(Starlette(routes=[Route('/', f, methods=['GET'])]))\nclient.get('/?x=1&x=2').text\n\n\"['1', '2']\"\n\n\n\ndef g(req, this:Starlette, a:str, b:HttpHeader): ...\n\nasync def f(req):\n a = await _wrap_req(req, _params(g))\n return Response(str(a))\n\nclient = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])]))\nresponse = client.post('/?a=1', data=d)\nprint(response.text)\n\n[<starlette.requests.Request object>, <starlette.applications.Starlette object>, '1', HttpHeader(k='value1', v='value3')]\n\n\n\ndef g(req, this:Starlette, a:str, b:HttpHeader): ...\n\nasync def f(req):\n a = await _wrap_req(req, _params(g))\n return Response(str(a))\n\nclient = TestClient(Starlette(routes=[Route('/', f, methods=['POST'])]))\nresponse = client.post('/?a=1', data=d)\nprint(response.text)\n\n[<starlette.requests.Request object>, <starlette.applications.Starlette object>, '1', HttpHeader(k='value1', v='value3')]\n\n\n\nsource\n\n\nflat_xt\n\n flat_xt (lst)\n\nFlatten lists\n\nx = ft('a',1)\ntest_eq(flat_xt([x, x, [x,x]]), (x,)*4)\ntest_eq(flat_xt(x), (x,))\n\n\nsource\n\n\nBeforeware\n\n Beforeware (f, skip=None)\n\nInitialize self. See help(type(self)) for accurate signature.", + "crumbs": [ + "Home", + "Source", + "Core" + ] + }, + { + "objectID": "api/core.html#websockets-sse", + "href": "api/core.html#websockets-sse", + "title": "Core", + "section": "Websockets / SSE", + "text": "Websockets / SSE\n\ndef on_receive(self, msg:str): return f\"Message text was: {msg}\"\nc = _ws_endp(on_receive)\ncli = TestClient(Starlette(routes=[WebSocketRoute('/', _ws_endp(on_receive))]))\nwith cli.websocket_connect('/') as ws:\n ws.send_text('{\"msg\":\"Hi!\"}')\n data = ws.receive_text()\n assert data == 'Message text was: Hi!'\n\n\nsource\n\nEventStream\n\n EventStream (s)\n\nCreate a text/event-stream response from s\n\nsource\n\n\nsignal_shutdown\n\n signal_shutdown ()", + "crumbs": [ + "Home", + "Source", + "Core" + ] + }, + { + "objectID": "api/core.html#routing-and-application", + "href": "api/core.html#routing-and-application", + "title": "Core", + "section": "Routing and application", + "text": "Routing and application\n\nsource\n\nuri\n\n uri (_arg, **kwargs)\n\n\nsource\n\n\ndecode_uri\n\n decode_uri (s)\n\n\nsource\n\n\nStringConvertor.to_string\n\n StringConvertor.to_string (value:str)\n\n\nsource\n\n\nHTTPConnection.url_path_for\n\n HTTPConnection.url_path_for (name:str, **path_params)\n\n\nsource\n\n\nflat_tuple\n\n flat_tuple (o)\n\nFlatten lists\n\nsource\n\n\nnoop_body\n\n noop_body (c, req)\n\nDefault Body wrap function which just returns the content\n\nsource\n\n\nrespond\n\n respond (req, heads, bdy)\n\nDefault FT response creation function\n\nsource\n\n\nRedirect\n\n Redirect (loc)\n\nUse HTMX or Starlette RedirectResponse as required to redirect to loc\n\nsource\n\n\nget_key\n\n get_key (key=None, fname='.sesskey')\n\n\nget_key()\n\n'5a5e5544-5ee8-46f2-836e-924976ce8b58'\n\n\n\nsource\n\n\nqp\n\n qp (p:str, **kw)\n\nAdd query parameters to path p\n\nqp('/foo', a=None, b=False, c=[1,2], d='bar')\n\n'/foo?a=&b=&c=1&c=2&d=bar'\n\n\n\nsource\n\n\ndef_hdrs\n\n def_hdrs (htmx=True, surreal=True)\n\nDefault headers for a FastHTML app\n\nsource\n\n\nFastHTML\n\n FastHTML (debug=False, routes=None, middleware=None, title:str='FastHTML\n page', exception_handlers=None, on_startup=None,\n on_shutdown=None, lifespan=None, hdrs=None, ftrs=None,\n exts=None, before=None, after=None, surreal=True, htmx=True,\n default_hdrs=True, sess_cls=<class\n 'starlette.middleware.sessions.SessionMiddleware'>,\n secret_key=None, session_cookie='session_', max_age=31536000,\n sess_path='/', same_site='lax', sess_https_only=False,\n sess_domain=None, key_fname='.sesskey', body_wrap=<function\n noop_body>, htmlkw=None, nb_hdrs=False, **bodykw)\n\nCreates an Starlette application.\n\nsource\n\n\nFastHTML.ws\n\n FastHTML.ws (path:str, conn=None, disconn=None, name=None,\n middleware=None)\n\nAdd a websocket route at path\n\nsource\n\n\nnested_name\n\n nested_name (f)\n\n*Get name of function f using ’_’ to join nested function names*\n\ndef f():\n def g(): ...\n return g\n\n\nfunc = f()\nnested_name(func)\n\n'f_g'\n\n\n\nsource\n\n\nFastHTML.route\n\n FastHTML.route (path:str=None, methods=None, name=None,\n include_in_schema=True, body_wrap=None)\n\nAdd a route at path\n\napp = FastHTML()\n@app.get\ndef foo(a:str, b:list[int]): ...\n\nprint(app.routes)\nfoo.to(a='bar', b=[1,2])\n\n[Route(path='/foo', name='foo', methods=['GET', 'HEAD'])]\n\n\n'/foo?a=bar&b=1&b=2'\n\n\n\nsource\n\n\nserve\n\n serve (appname=None, app='app', host='0.0.0.0', port=None, reload=True,\n reload_includes:list[str]|str|None=None,\n reload_excludes:list[str]|str|None=None)\n\nRun the app in an async server, with live reload set as the default.\n\n\n\n\n\n\n\n\n\n\nType\nDefault\nDetails\n\n\n\n\nappname\nNoneType\nNone\nName of the module\n\n\napp\nstr\napp\nApp instance to be served\n\n\nhost\nstr\n0.0.0.0\nIf host is 0.0.0.0 will convert to localhost\n\n\nport\nNoneType\nNone\nIf port is None it will default to 5001 or the PORT environment variable\n\n\nreload\nbool\nTrue\nDefault is to reload the app upon code changes\n\n\nreload_includes\nlist[str] | str | None\nNone\nAdditional files to watch for changes\n\n\nreload_excludes\nlist[str] | str | None\nNone\nFiles to ignore for changes\n\n\n\n\nsource\n\n\nClient\n\n Client (app, url='http://testserver')\n\nA simple httpx ASGI client that doesn’t require async\n\napp = FastHTML(routes=[Route('/', lambda _: Response('test'))])\ncli = Client(app)\n\ncli.get('/').text\n\n'test'\n\n\nNote that you can also use Starlette’s TestClient instead of FastHTML’s Client. They should be largely interchangable.", + "crumbs": [ + "Home", + "Source", + "Core" + ] + }, + { + "objectID": "api/core.html#fasthtml-tests", + "href": "api/core.html#fasthtml-tests", + "title": "Core", + "section": "FastHTML Tests", + "text": "FastHTML Tests\n\ndef get_cli(app): return app,TestClient(app),app.route\n\n\napp,cli,rt = get_cli(FastHTML(secret_key='soopersecret'))\n\n\napp,cli,rt = get_cli(FastHTML(title=\"My Custom Title\"))\n@app.get\ndef foo(): return Div(\"Hello World\")\n\nprint(app.routes)\n\nresponse = cli.get('/foo')\nassert '<title>My Custom Title</title>' in response.text\n\nfoo.to(param='value')\n\n[Route(path='/foo', name='foo', methods=['GET', 'HEAD'])]\n\n\n'/foo?param=value'\n\n\n\napp,cli,rt = get_cli(FastHTML())\n\n@rt('/xt2')\ndef get(): return H1('bar')\n\ntxt = cli.get('/xt2').text\nassert '<title>FastHTML page</title>' in txt and '<h1>bar</h1>' in txt and '<html>' in txt\n\n\n@rt(\"/hi\")\ndef get(): return 'Hi there'\n\nr = cli.get('/hi')\nr.text\n\n'Hi there'\n\n\n\n@rt(\"/hi\")\ndef post(): return 'Postal'\n\ncli.post('/hi').text\n\n'Postal'\n\n\n\n@app.get(\"/hostie\")\ndef show_host(req): return req.headers['host']\n\ncli.get('/hostie').text\n\n'testserver'\n\n\n\n@app.get(\"/setsess\")\ndef set_sess(session):\n session['foo'] = 'bar'\n return 'ok'\n\n@app.ws(\"/ws\")\ndef ws(self, msg:str, ws:WebSocket, session): return f\"Message text was: {msg} with session {session.get('foo')}, from client: {ws.client}\"\n\ncli.get('/setsess')\nwith cli.websocket_connect('/ws') as ws:\n ws.send_text('{\"msg\":\"Hi!\"}')\n data = ws.receive_text()\nassert 'Message text was: Hi! with session bar' in data\nprint(data)\n\nMessage text was: Hi! with session bar, from client: Address(host='testclient', port=50000)\n\n\n\n@rt\ndef yoyo(): return 'a yoyo'\n\ncli.post('/yoyo').text\n\n'a yoyo'\n\n\n\n@app.get\ndef autopost(): return Html(Div('Text.', hx_post=yoyo()))\nprint(cli.get('/autopost').text)\n\n <!doctype html>\n <html>\n <div hx-post=\"a yoyo\">Text.</div>\n </html>\n\n\n\n\n@app.get\ndef autopost2(): return Html(Body(Div('Text.', cls='px-2', hx_post=show_host.to(a='b'))))\nprint(cli.get('/autopost2').text)\n\n <!doctype html>\n <html>\n <body>\n <div class=\"px-2\" hx-post=\"/hostie?a=b\">Text.</div>\n </body>\n </html>\n\n\n\n\n@app.get\ndef autoget2(): return Html(Div('Text.', hx_get=show_host))\nprint(cli.get('/autoget2').text)\n\n <!doctype html>\n <html>\n <div hx-get=\"/hostie\">Text.</div>\n </html>\n\n\n\n\n@rt('/user/{nm}', name='gday')\ndef get(nm:str=''): return f\"Good day to you, {nm}!\"\ncli.get('/user/Alexis').text\n\n'Good day to you, Alexis!'\n\n\n\n@app.get\ndef autolink(): return Html(Div('Text.', link=uri('gday', nm='Alexis')))\nprint(cli.get('/autolink').text)\n\n <!doctype html>\n <html>\n <div href=\"/user/Alexis\">Text.</div>\n </html>\n\n\n\n\n@rt('/link')\ndef get(req): return f\"{req.url_for('gday', nm='Alexis')}; {req.url_for('show_host')}\"\n\ncli.get('/link').text\n\n'http://testserver/user/Alexis; http://testserver/hostie'\n\n\n\n@app.get(\"/background\")\nasync def background_task(request):\n async def long_running_task():\n await asyncio.sleep(0.1)\n print(\"Background task completed!\")\n return P(\"Task started\"), BackgroundTask(long_running_task)\n\nresponse = cli.get(\"/background\")\n\nBackground task completed!\n\n\n\ntest_eq(app.router.url_path_for('gday', nm='Jeremy'), '/user/Jeremy')\n\n\nhxhdr = {'headers':{'hx-request':\"1\"}}\n\n@rt('/ft')\ndef get(): return Title('Foo'),H1('bar')\n\ntxt = cli.get('/ft').text\nassert '<title>Foo</title>' in txt and '<h1>bar</h1>' in txt and '<html>' in txt\n\n@rt('/xt2')\ndef get(): return H1('bar')\n\ntxt = cli.get('/xt2').text\nassert '<title>FastHTML page</title>' in txt and '<h1>bar</h1>' in txt and '<html>' in txt\n\nassert cli.get('/xt2', **hxhdr).text.strip() == '<h1>bar</h1>'\n\n@rt('/xt3')\ndef get(): return Html(Head(Title('hi')), Body(P('there')))\n\ntxt = cli.get('/xt3').text\nassert '<title>FastHTML page</title>' not in txt and '<title>hi</title>' in txt and '<p>there</p>' in txt\n\n\n@rt('/oops')\ndef get(nope): return nope\ntest_warns(lambda: cli.get('/oops?nope=1'))\n\n\ndef test_r(cli, path, exp, meth='get', hx=False, **kwargs):\n if hx: kwargs['headers'] = {'hx-request':\"1\"}\n test_eq(getattr(cli, meth)(path, **kwargs).text, exp)\n\nModelName = str_enum('ModelName', \"alexnet\", \"resnet\", \"lenet\")\nfake_db = [{\"name\": \"Foo\"}, {\"name\": \"Bar\"}]\n\n\n@rt('/html/{idx}')\nasync def get(idx:int): return Body(H4(f'Next is {idx+1}.'))\n\n\n@rt(\"/models/{nm}\")\ndef get(nm:ModelName): return nm\n\n@rt(\"/files/{path}\")\nasync def get(path: Path): return path.with_suffix('.txt')\n\n@rt(\"/items/\")\ndef get(idx:int|None = 0): return fake_db[idx]\n\n@rt(\"/idxl/\")\ndef get(idx:list[int]): return str(idx)\n\n\nr = cli.get('/html/1', headers={'hx-request':\"1\"})\nassert '<h4>Next is 2.</h4>' in r.text\ntest_r(cli, '/models/alexnet', 'alexnet')\ntest_r(cli, '/files/foo', 'foo.txt')\ntest_r(cli, '/items/?idx=1', '{\"name\":\"Bar\"}')\ntest_r(cli, '/items/', '{\"name\":\"Foo\"}')\nassert cli.get('/items/?idx=g').text=='404 Not Found'\nassert cli.get('/items/?idx=g').status_code == 404\ntest_r(cli, '/idxl/?idx=1&idx=2', '[1, 2]')\nassert cli.get('/idxl/?idx=1&idx=g').status_code == 404\n\n\napp = FastHTML()\nrt = app.route\ncli = TestClient(app)\n@app.route(r'/static/{path:path}.jpg')\ndef index(path:str): return f'got {path}'\ncli.get('/static/sub/a.b.jpg').text\n\n'got sub/a.b'\n\n\n\napp.chk = 'foo'\n\n\n@app.get(\"/booly/\")\ndef _(coming:bool=True): return 'Coming' if coming else 'Not coming'\n\n@app.get(\"/datie/\")\ndef _(d:parsed_date): return d\n\n@app.get(\"/ua\")\nasync def _(user_agent:str): return user_agent\n\n@app.get(\"/hxtest\")\ndef _(htmx): return htmx.request\n\n@app.get(\"/hxtest2\")\ndef _(foo:HtmxHeaders, req): return foo.request\n\n@app.get(\"/app\")\ndef _(app): return app.chk\n\n@app.get(\"/app2\")\ndef _(foo:FastHTML): return foo.chk,HttpHeader(\"mykey\", \"myval\")\n\n@app.get(\"/app3\")\ndef _(foo:FastHTML): return HtmxResponseHeaders(location=\"http://example.org\")\n\n@app.get(\"/app4\")\ndef _(foo:FastHTML): return Redirect(\"http://example.org\")\n\n\ntest_r(cli, '/booly/?coming=true', 'Coming')\ntest_r(cli, '/booly/?coming=no', 'Not coming')\ndate_str = \"17th of May, 2024, 2p\"\ntest_r(cli, f'/datie/?d={date_str}', '2024-05-17 14:00:00')\ntest_r(cli, '/ua', 'FastHTML', headers={'User-Agent':'FastHTML'})\ntest_r(cli, '/hxtest' , '1', headers={'HX-Request':'1'})\ntest_r(cli, '/hxtest2', '1', headers={'HX-Request':'1'})\ntest_r(cli, '/app' , 'foo')\n\n\nr = cli.get('/app2', **hxhdr)\ntest_eq(r.text, 'foo')\ntest_eq(r.headers['mykey'], 'myval')\n\n\nr = cli.get('/app3')\ntest_eq(r.headers['HX-Location'], 'http://example.org')\n\n\nr = cli.get('/app4', follow_redirects=False)\ntest_eq(r.status_code, 303)\n\n\nr = cli.get('/app4', headers={'HX-Request':'1'})\ntest_eq(r.headers['HX-Redirect'], 'http://example.org')\n\n\n@rt\ndef meta():\n return ((Title('hi'),H1('hi')),\n (Meta(property='image'), Meta(property='site_name'))\n )\n\nt = cli.post('/meta').text\nassert re.search(r'<body>\\s*<h1>hi</h1>\\s*</body>', t)\nassert '<meta' in t\n\n\n@app.post('/profile/me')\ndef profile_update(username: str): return username\n\ntest_r(cli, '/profile/me', 'Alexis', 'post', data={'username' : 'Alexis'})\ntest_r(cli, '/profile/me', 'Missing required field: username', 'post', data={})\n\n\n# Example post request with parameter that has a default value\n@app.post('/pet/dog')\ndef pet_dog(dogname: str = None): return dogname\n\n# Working post request with optional parameter\ntest_r(cli, '/pet/dog', '', 'post', data={})\n\n\n@dataclass\nclass Bodie: a:int;b:str\n\n@rt(\"/bodie/{nm}\")\ndef post(nm:str, data:Bodie):\n res = asdict(data)\n res['nm'] = nm\n return res\n\n@app.post(\"/bodied/\")\ndef bodied(data:dict): return data\n\nnt = namedtuple('Bodient', ['a','b'])\n\n@app.post(\"/bodient/\")\ndef bodient(data:nt): return asdict(data)\n\nclass BodieTD(TypedDict): a:int;b:str='foo'\n\n@app.post(\"/bodietd/\")\ndef bodient(data:BodieTD): return data\n\nclass Bodie2:\n a:int|None; b:str\n def __init__(self, a, b='foo'): store_attr()\n\n@rt(\"/bodie2/\", methods=['get','post'])\ndef bodie(d:Bodie2): return f\"a: {d.a}; b: {d.b}\"\n\n\nfrom fasthtml.xtend import Titled\n\n\nd = dict(a=1, b='foo')\n\ntest_r(cli, '/bodie/me', '{\"a\":1,\"b\":\"foo\",\"nm\":\"me\"}', 'post', data=dict(a=1, b='foo', nm='me'))\ntest_r(cli, '/bodied/', '{\"a\":\"1\",\"b\":\"foo\"}', 'post', data=d)\ntest_r(cli, '/bodie2/', 'a: 1; b: foo', 'post', data={'a':1})\ntest_r(cli, '/bodie2/?a=1&b=foo&nm=me', 'a: 1; b: foo')\ntest_r(cli, '/bodient/', '{\"a\":\"1\",\"b\":\"foo\"}', 'post', data=d)\ntest_r(cli, '/bodietd/', '{\"a\":1,\"b\":\"foo\"}', 'post', data=d)\n\n\n# Testing POST with Content-Type: application/json\n@app.post(\"/\")\ndef index(it: Bodie): return Titled(\"It worked!\", P(f\"{it.a}, {it.b}\"))\n\ns = json.dumps({\"b\": \"Lorem\", \"a\": 15})\nresponse = cli.post('/', headers={\"Content-Type\": \"application/json\"}, data=s).text\nassert \"<title>It worked!</title>\" in response and \"<p>15, Lorem</p>\" in response\n\n\n# Testing POST with Content-Type: application/json\n@app.post(\"/bodytext\")\ndef index(body): return body\n\nresponse = cli.post('/bodytext', headers={\"Content-Type\": \"application/json\"}, data=s).text\ntest_eq(response, '{\"b\": \"Lorem\", \"a\": 15}')\n\n\nfiles = [ ('files', ('file1.txt', b'content1')),\n ('files', ('file2.txt', b'content2')) ]\n\n\n@rt(\"/uploads\")\nasync def post(files:list[UploadFile]):\n return ','.join([(await file.read()).decode() for file in files])\n\nres = cli.post('/uploads', files=files)\nprint(res.status_code)\nprint(res.text)\n\n200\ncontent1,content2\n\n\n\nres = cli.post('/uploads', files=[files[0]])\nprint(res.status_code)\nprint(res.text)\n\n200\ncontent1\n\n\n\n@rt(\"/setsess\")\ndef get(sess, foo:str=''):\n now = datetime.now()\n sess['auth'] = str(now)\n return f'Set to {now}'\n\n@rt(\"/getsess\")\ndef get(sess): return f'Session time: {sess[\"auth\"]}'\n\nprint(cli.get('/setsess').text)\ntime.sleep(0.01)\n\ncli.get('/getsess').text\n\nSet to 2025-01-12 14:12:46.576323\n\n\n'Session time: 2025-01-12 14:12:46.576323'\n\n\n\n@rt(\"/sess-first\")\ndef post(sess, name: str):\n sess[\"name\"] = name\n return str(sess)\n\ncli.post('/sess-first', data={'name': 2})\n\n@rt(\"/getsess-all\")\ndef get(sess): return sess['name']\n\ntest_eq(cli.get('/getsess-all').text, '2')\n\n\n@rt(\"/upload\")\nasync def post(uf:UploadFile): return (await uf.read()).decode()\n\nwith open('../../CHANGELOG.md', 'rb') as f:\n print(cli.post('/upload', files={'uf':f}, data={'msg':'Hello'}).text[:15])\n\n# Release notes\n\n\n\n@rt(\"/form-submit/{list_id}\")\ndef options(list_id: str):\n headers = {\n 'Access-Control-Allow-Origin': '*',\n 'Access-Control-Allow-Methods': 'POST',\n 'Access-Control-Allow-Headers': '*',\n }\n return Response(status_code=200, headers=headers)\n\n\nh = cli.options('/form-submit/2').headers\ntest_eq(h['Access-Control-Allow-Methods'], 'POST')\n\n\nfrom fasthtml.authmw import user_pwd_auth\n\n\ndef _not_found(req, exc): return Div('nope')\n\napp,cli,rt = get_cli(FastHTML(exception_handlers={404:_not_found}))\n\ntxt = cli.get('/').text\nassert '<div>nope</div>' in txt\nassert '<!doctype html>' in txt\n\n\napp,cli,rt = get_cli(FastHTML())\n\n@rt(\"/{name}/{age}\")\ndef get(name: str, age: int):\n return Titled(f\"Hello {name.title()}, age {age}\")\n\nassert '<title>Hello Uma, age 5</title>' in cli.get('/uma/5').text\nassert '404 Not Found' in cli.get('/uma/five').text\n\n\nauth = user_pwd_auth(testuser='spycraft')\napp,cli,rt = get_cli(FastHTML(middleware=[auth]))\n\n@rt(\"/locked\")\ndef get(auth): return 'Hello, ' + auth\n\ntest_eq(cli.get('/locked').text, 'not authenticated')\ntest_eq(cli.get('/locked', auth=(\"testuser\",\"spycraft\")).text, 'Hello, testuser')\n\n\nauth = user_pwd_auth(testuser='spycraft')\napp,cli,rt = get_cli(FastHTML(middleware=[auth]))\n\n@rt(\"/locked\")\ndef get(auth): return 'Hello, ' + auth\n\ntest_eq(cli.get('/locked').text, 'not authenticated')\ntest_eq(cli.get('/locked', auth=(\"testuser\",\"spycraft\")).text, 'Hello, testuser')", + "crumbs": [ + "Home", + "Source", + "Core" + ] + }, + { + "objectID": "api/core.html#apirouter", + "href": "api/core.html#apirouter", + "title": "Core", + "section": "APIRouter", + "text": "APIRouter\n\nsource\n\nRouteFuncs\n\n RouteFuncs ()\n\nInitialize self. See help(type(self)) for accurate signature.\n\nsource\n\n\nAPIRouter\n\n APIRouter (prefix:str|None=None, body_wrap=<function noop_body>)\n\nAdd routes to an app\n\nar = APIRouter()\n\n\n@ar(\"/hi\")\ndef get(): return 'Hi there'\n@ar(\"/hi\")\ndef post(): return 'Postal'\n@ar\ndef ho(): return 'Ho ho'\n@ar(\"/hostie\")\ndef show_host(req): return req.headers['host']\n@ar\ndef yoyo(): return 'a yoyo'\n@ar\ndef index(): return \"home page\"\n\n@ar.ws(\"/ws\")\ndef ws(self, msg:str): return f\"Message text was: {msg}\"\n\n\napp,cli,_ = get_cli(FastHTML())\nar.to_app(app)\n\n\nassert str(yoyo) == '/yoyo'\n# ensure route functions are properly discoverable on `APIRouter` and `APIRouter.rt_funcs`\nassert ar.prefix == ''\nassert str(ar.rt_funcs.index) == '/'\nassert str(ar.index) == '/'\nwith ExceptionExpected(): ar.blah()\nwith ExceptionExpected(): ar.rt_funcs.blah()\n# ensure any route functions named using an HTTPMethod are not discoverable via `rt_funcs`\nassert \"get\" not in ar.rt_funcs._funcs.keys()\n\n\ntest_eq(cli.get('/hi').text, 'Hi there')\ntest_eq(cli.post('/hi').text, 'Postal')\ntest_eq(cli.get('/hostie').text, 'testserver')\ntest_eq(cli.post('/yoyo').text, 'a yoyo')\n\ntest_eq(cli.get('/ho').text, 'Ho ho')\ntest_eq(cli.post('/ho').text, 'Ho ho')\n\n\nwith cli.websocket_connect('/ws') as ws:\n ws.send_text('{\"msg\":\"Hi!\"}')\n data = ws.receive_text()\n assert data == 'Message text was: Hi!'\n\n\nar2 = APIRouter(\"/products\")\n\n\n@ar2(\"/hi\")\ndef get(): return 'Hi there'\n@ar2(\"/hi\")\ndef post(): return 'Postal'\n@ar2\ndef ho(): return 'Ho ho'\n@ar2(\"/hostie\")\ndef show_host(req): return req.headers['host']\n@ar2\ndef yoyo(): return 'a yoyo'\n@ar2\ndef index(): return \"home page\"\n\n@ar2.ws(\"/ws\")\ndef ws(self, msg:str): return f\"Message text was: {msg}\"\n\n\napp,cli,_ = get_cli(FastHTML())\nar2.to_app(app)\n\n\nassert str(yoyo) == '/products/yoyo'\nassert ar2.prefix == '/products'\nassert str(ar2.rt_funcs.index) == '/products/'\nassert str(ar2.index) == '/products/'\nassert str(ar.index) == '/'\nwith ExceptionExpected(): ar2.blah()\nwith ExceptionExpected(): ar2.rt_funcs.blah()\nassert \"get\" not in ar2.rt_funcs._funcs.keys()\n\n\ntest_eq(cli.get('/products/hi').text, 'Hi there')\ntest_eq(cli.post('/products/hi').text, 'Postal')\ntest_eq(cli.get('/products/hostie').text, 'testserver')\ntest_eq(cli.post('/products/yoyo').text, 'a yoyo')\n\ntest_eq(cli.get('/products/ho').text, 'Ho ho')\ntest_eq(cli.post('/products/ho').text, 'Ho ho')\n\n\nwith cli.websocket_connect('/products/ws') as ws:\n ws.send_text('{\"msg\":\"Hi!\"}')\n data = ws.receive_text()\n assert data == 'Message text was: Hi!'\n\n\n@ar.get\ndef hi2(): return 'Hi there'\n@ar.get(\"/hi3\")\ndef _(): return 'Hi there'\n@ar.post(\"/post2\")\ndef _(): return 'Postal'\n\n@ar2.get\ndef hi2(): return 'Hi there'\n@ar2.get(\"/hi3\")\ndef _(): return 'Hi there'\n@ar2.post(\"/post2\")\ndef _(): return 'Postal'", + "crumbs": [ + "Home", + "Source", + "Core" + ] + }, + { + "objectID": "api/core.html#extras", + "href": "api/core.html#extras", + "title": "Core", + "section": "Extras", + "text": "Extras\n\napp,cli,rt = get_cli(FastHTML(secret_key='soopersecret'))\n\n\nsource\n\ncookie\n\n cookie (key:str, value='', max_age=None, expires=None, path='/',\n domain=None, secure=False, httponly=False, samesite='lax')\n\nCreate a ‘set-cookie’ HttpHeader\n\n@rt(\"/setcookie\")\ndef get(req): return cookie('now', datetime.now())\n\n@rt(\"/getcookie\")\ndef get(now:parsed_date): return f'Cookie was set at time {now.time()}'\n\nprint(cli.get('/setcookie').text)\ntime.sleep(0.01)\ncli.get('/getcookie').text\n\n\n\n\n'Cookie was set at time 14:12:47.159530'\n\n\n\nsource\n\n\nreg_re_param\n\n reg_re_param (m, s)\n\n\nsource\n\n\nFastHTML.static_route_exts\n\n FastHTML.static_route_exts (prefix='/', static_path='.', exts='static')\n\nAdd a static route at URL path prefix with files from static_path and exts defined by reg_re_param()\n\nreg_re_param(\"imgext\", \"ico|gif|jpg|jpeg|webm|pdf\")\n\n@rt(r'/static/{path:path}{fn}.{ext:imgext}')\ndef get(fn:str, path:str, ext:str): return f\"Getting {fn}.{ext} from /{path}\"\n\ntest_r(cli, '/static/foo/jph.me.ico', 'Getting jph.me.ico from /foo/')\n\n\napp.static_route_exts()\nassert 'These are the source notebooks for FastHTML' in cli.get('/README.txt').text\n\n\nsource\n\n\nFastHTML.static_route\n\n FastHTML.static_route (ext='', prefix='/', static_path='.')\n\nAdd a static route at URL path prefix with files from static_path and single ext (including the ‘.’)\n\napp.static_route('.md', static_path='../..')\nassert 'THIS FILE WAS AUTOGENERATED' in cli.get('/README.md').text\n\n\nsource\n\n\nMiddlewareBase\n\n MiddlewareBase ()\n\nInitialize self. See help(type(self)) for accurate signature.\n\nsource\n\n\nFtResponse\n\n FtResponse (content, status_code:int=200, headers=None, cls=<class\n 'starlette.responses.HTMLResponse'>,\n media_type:str|None=None)\n\nWrap an FT response with any Starlette Response\n\n@rt('/ftr')\ndef get():\n cts = Title('Foo'),H1('bar')\n return FtResponse(cts, status_code=201, headers={'Location':'/foo/1'})\n\nr = cli.get('/ftr')\n\ntest_eq(r.status_code, 201)\ntest_eq(r.headers['location'], '/foo/1')\ntxt = r.text\nassert '<title>Foo</title>' in txt and '<h1>bar</h1>' in txt and '<html>' in txt\n\n\nsource\n\n\nunqid\n\n unqid ()\n\n\nsource\n\n\nsetup_ws\n\n setup_ws (app, f=<function noop>)", + "crumbs": [ + "Home", + "Source", + "Core" + ] + }, + { + "objectID": "unpublished/tutorial_for_web_devs.html", + "href": "unpublished/tutorial_for_web_devs.html", + "title": "BYO Blog", + "section": "", + "text": "Caution\n\n\n\nThis document is a work in progress.\nIn this tutorial we’re going to write a blog by example. Blogs are a good way to learn a web framework as they start simple yet can get surprisingly sophistated. The wikipedia definition of a blog is “an informational website consisting of discrete, often informal diary-style text entries (posts) informal diary-style text entries (posts)”, which means we need to provide these basic features:\nWe’ll also add in these features, so the blog can become a working site:" + }, + { + "objectID": "unpublished/tutorial_for_web_devs.html#how-to-best-use-this-tutorial", + "href": "unpublished/tutorial_for_web_devs.html#how-to-best-use-this-tutorial", + "title": "BYO Blog", + "section": "How to best use this tutorial", + "text": "How to best use this tutorial\nWe could copy/paste every code example in sequence and have a finished blog at the end. However, it’s debatable how much we will learn through the copy/paste method. We’re not saying its impossible to learn through copy/paste, we’re just saying it’s not that of an efficient way to learn. It’s analogous to learning how to play a musical instrument or sport or video game by watching other people do it - you can learn some but its not the same as doing.\nA better approach is to type out every line of code in this tutorial. This forces us to run the code through our brains, giving us actual practice in how to write FastHTML and Pythoncode and forcing us to debug our own mistakes. In some cases we’ll repeat similar tasks - a key component in achieving mastery in anything. Coming back to the instrument/sport/video game analogy, it’s exactly like actually practicing an instrument, sport, or video game. Through practice and repetition we eventually achieve mastery." + }, + { + "objectID": "unpublished/tutorial_for_web_devs.html#installing-fasthtml", + "href": "unpublished/tutorial_for_web_devs.html#installing-fasthtml", + "title": "BYO Blog", + "section": "Installing FastHTML", + "text": "Installing FastHTML\nFastHTML is just Python. Installation is often done with pip:\npip install python-fasthtml" + }, + { + "objectID": "unpublished/tutorial_for_web_devs.html#a-minimal-fasthtml-app", + "href": "unpublished/tutorial_for_web_devs.html#a-minimal-fasthtml-app", + "title": "BYO Blog", + "section": "A minimal FastHTML app", + "text": "A minimal FastHTML app\nFirst, create the directory for our project using Python’s pathlib module:\nimport pathlib\npathlib.Path('blog-system').mkdir()\nNow that we have our directory, let’s create a minimal FastHTML site in it.\n\n\nblog-system/minimal.py\n\nfrom fasthtml.common import * \n\napp, rt = fast_app() \n\n@rt(\"/\") \ndef get():\n return Titled(\"FastHTML\", P(\"Let's do this!\")) \n\nserve()\n\nRun that with python minimal.py and you should get something like this:\npython minimal.py \nLink: http://localhost:5001\nINFO: Will watch for changes in these directories: ['/Users/pydanny/projects/blog-system']\nINFO: Uvicorn running on http://0.0.0.0:5001 (Press CTRL+C to quit)\nINFO: Started reloader process [46572] using WatchFiles\nINFO: Started server process [46576]\nINFO: Waiting for application startup.\nINFO: Application startup complete.\nConfirm FastHTML is running by opening your web browser to 127.0.0.1:5001. You should see something like the image below:\n\n\n\n\n\n\n\nWhat about the import *?\n\n\n\nFor those worried about the use of import * rather than a PEP8-style declared namespace, understand that __all__ is defined in FastHTML’s common module. That means that only the symbols (functions, classes, and other things) the framework wants us to have will be brought into our own code via import *. Read importing from a package) for more information.\nNevertheless, if we want to use a defined namespace we can do so. Here’s an example:\nfrom fasthtml import common as fh\n\n\napp, rt = fh.fast_app() \n\n@rt(\"/\") \ndef get():\n return fh.Titled(\"FastHTML\", fh.P(\"Let's do this!\")) \n\nfh.serve()" + }, + { + "objectID": "unpublished/tutorial_for_web_devs.html#looking-more-closely-at-our-app", + "href": "unpublished/tutorial_for_web_devs.html#looking-more-closely-at-our-app", + "title": "BYO Blog", + "section": "Looking more closely at our app", + "text": "Looking more closely at our app\nLet’s look more closely at our application. Every line is packed with powerful features of FastHTML:\n\n\nblog-system/minimal.py\n\n1from fasthtml.common import *\n\n2app, rt = fast_app()\n\n3@rt(\"/\")\n4def get():\n5 return Titled(\"FastHTML\", P(\"Let's do this!\"))\n\n6serve()\n\n\n1\n\nThe top level namespace of Fast HTML (fasthtml.common) contains everything we need from FastHTML to build applications. A carefully-curated set of FastHTML functions and other Python objects is brought into our global namespace for convenience.\n\n2\n\nWe instantiate a FastHTML app with the fast_app() utility function. This provides a number of really useful defaults that we’ll modify or take advantage of later in the tutorial.\n\n3\n\nWe use the rt() decorator to tell FastHTML what to return when a user visits / in their browser.\n\n4\n\nWe connect this route to HTTP GET requests by defining a view function called get().\n\n5\n\nA tree of Python function calls that return all the HTML required to write a properly formed web page. You’ll soon see the power of this approach.\n\n6\n\nThe serve() utility configures and runs FastHTML using a library called uvicorn. Any changes to this module will be reloaded into the browser." + }, + { + "objectID": "unpublished/tutorial_for_web_devs.html#adding-dynamic-content-to-our-minimal-app", + "href": "unpublished/tutorial_for_web_devs.html#adding-dynamic-content-to-our-minimal-app", + "title": "BYO Blog", + "section": "Adding dynamic content to our minimal app", + "text": "Adding dynamic content to our minimal app\nOur page is great, but we’ll make it better. Let’s add a randomized list of letters to the page. Every time the page reloads, a new list of varying length will be generated.\n\n\nblog-system/random_letters.py\n\nfrom fasthtml.common import *\n1import string, random\n\napp, rt = fast_app()\n\n@rt(\"/\")\ndef get():\n2 letters = random.choices(string.ascii_uppercase, k=random.randint(5, 20))\n3 items = [Li(c) for c in letters]\n return Titled(\"Random lists of letters\",\n4 Ul(*items)\n ) \n\nserve()\n\n\n1\n\nThe string and random libraries are part of Python’s standard library\n\n2\n\nWe use these libraries to generate a random length list of random letters called letters\n\n3\n\nUsing letters as the base we use list comprehension to generate a list of Li ft display components, each with their own letter and save that to the variable items\n\n4\n\nInside a call to the Ul() ft component we use Python’s *args special syntax on the items variable. Therefore *list is treated not as one argument but rather a set of them.\n\n\nWhen this is run, it will generate something like this with a different random list of letters for each page load:" + }, + { + "objectID": "unpublished/tutorial_for_web_devs.html#storing-the-articles", + "href": "unpublished/tutorial_for_web_devs.html#storing-the-articles", + "title": "BYO Blog", + "section": "Storing the articles", + "text": "Storing the articles\nThe most basic component of a blog is a series of articles sorted by date authored. Rather than a database we’re going to use our computer’s harddrive to store a set of markdown files in a directory within our blog called posts. First, let’s create the directory and some test files we can use to search for:\n\nfrom fastcore.utils import *\n\n\n# Create some dummy posts\nposts = Path(\"posts\")\nposts.mkdir(exist_ok=True)\nfor i in range(10): (posts/f\"article_{i}.md\").write_text(f\"This is article {i}\")\n\nSearching for these files can be done with pathlib.\n\nimport pathlib\nposts.ls()\n\n(#10) [Path('posts/article_5.md'),Path('posts/article_1.md'),Path('posts/article_0.md'),Path('posts/article_4.md'),Path('posts/article_3.md'),Path('posts/article_7.md'),Path('posts/article_6.md'),Path('posts/article_2.md'),Path('posts/article_9.md'),Path('posts/article_8.md')]\n\n\n\n\n\n\n\n\nTip\n\n\n\nPython’s pathlib library is quite useful and makes file search and manipulation much easier. There’s many uses for it and is compatible across operating systems." + }, + { + "objectID": "unpublished/tutorial_for_web_devs.html#creating-the-blog-home-page", + "href": "unpublished/tutorial_for_web_devs.html#creating-the-blog-home-page", + "title": "BYO Blog", + "section": "Creating the blog home page", + "text": "Creating the blog home page\nWe now have enough tools that we can create the home page. Let’s create a new Python file and write out our simple view to list the articles in our blog.\n\n\nblog-system/main.py\n\nfrom fasthtml.common import *\nimport pathlib\n\napp, rt = fast_app()\n\n@rt(\"/\")\ndef get():\n fnames = pathlib.Path(\"posts\").rglob(\"*.md\")\n items = [Li(A(fname, href=fname)) for fname in fnames] \n return Titled(\"My Blog\",\n Ul(*items)\n ) \n\nserve()\n\n\nfor p in posts.ls(): p.unlink()" + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html", + "href": "tutorials/quickstart_for_web_devs.html", + "title": "Web Devs Quickstart", + "section": "", + "text": "pip install python-fasthtml", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#installation", + "href": "tutorials/quickstart_for_web_devs.html#installation", + "title": "Web Devs Quickstart", + "section": "", + "text": "pip install python-fasthtml", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#a-minimal-application", + "href": "tutorials/quickstart_for_web_devs.html#a-minimal-application", + "title": "Web Devs Quickstart", + "section": "A Minimal Application", + "text": "A Minimal Application\nA minimal FastHTML application looks something like this:\n\n\nmain.py\n\n1from fasthtml.common import *\n\n2app, rt = fast_app()\n\n3@rt(\"/\")\n4def get():\n5 return Titled(\"FastHTML\", P(\"Let's do this!\"))\n\n6serve()\n\n\n1\n\nWe import what we need for rapid development! A carefully-curated set of FastHTML functions and other Python objects is brought into our global namespace for convenience.\n\n2\n\nWe instantiate a FastHTML app with the fast_app() utility function. This provides a number of really useful defaults that we’ll take advantage of later in the tutorial.\n\n3\n\nWe use the rt() decorator to tell FastHTML what to return when a user visits / in their browser.\n\n4\n\nWe connect this route to HTTP GET requests by defining a view function called get().\n\n5\n\nA tree of Python function calls that return all the HTML required to write a properly formed web page. You’ll soon see the power of this approach.\n\n6\n\nThe serve() utility configures and runs FastHTML using a library called uvicorn.\n\n\nRun the code:\npython main.py\nThe terminal will look like this:\nINFO: Uvicorn running on http://0.0.0.0:5001 (Press CTRL+C to quit)\nINFO: Started reloader process [58058] using WatchFiles\nINFO: Started server process [58060]\nINFO: Waiting for application startup.\nINFO: Application startup complete.\nConfirm FastHTML is running by opening your web browser to 127.0.0.1:5001. You should see something like the image below:\n\n\n\n\n\n\n\nNote\n\n\n\nWhile some linters and developers will complain about the wildcard import, it is by design here and perfectly safe. FastHTML is very deliberate about the objects it exports in fasthtml.common. If it bothers you, you can import the objects you need individually, though it will make the code more verbose and less readable.\nIf you want to learn more about how FastHTML handles imports, we cover that here.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#a-minimal-charting-application", + "href": "tutorials/quickstart_for_web_devs.html#a-minimal-charting-application", + "title": "Web Devs Quickstart", + "section": "A Minimal Charting Application", + "text": "A Minimal Charting Application\nThe Script function allows you to include JavaScript. You can use Python to generate parts of your JS or JSON like this:\nimport json\nfrom fasthtml.common import * \n\napp, rt = fast_app(hdrs=(Script(src=\"https://cdn.plot.ly/plotly-2.32.0.min.js\"),))\n\ndata = json.dumps({\n \"data\": [{\"x\": [1, 2, 3, 4],\"type\": \"scatter\"},\n {\"x\": [1, 2, 3, 4],\"y\": [16, 5, 11, 9],\"type\": \"scatter\"}],\n \"title\": \"Plotly chart in FastHTML \",\n \"description\": \"This is a demo dashboard\",\n \"type\": \"scatter\"\n})\n\n\n@rt(\"/\")\ndef get():\n return Titled(\"Chart Demo\", Div(id=\"myDiv\"),\n Script(f\"var data = {data}; Plotly.newPlot('myDiv', data);\"))\n\nserve()", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#debug-mode", + "href": "tutorials/quickstart_for_web_devs.html#debug-mode", + "title": "Web Devs Quickstart", + "section": "Debug Mode", + "text": "Debug Mode\nWhen we can’t figure out a bug in FastHTML, we can run it in DEBUG mode. When an error is thrown, the error screen is displayed in the browser. This error setting should never be used in a deployed app.\nfrom fasthtml.common import *\n\n1app, rt = fast_app(debug=True)\n\n@rt(\"/\")\ndef get():\n2 1/0\n return Titled(\"FastHTML Error!\", P(\"Let's error!\"))\n\nserve()\n\n1\n\ndebug=True sets debug mode on.\n\n2\n\nPython throws an error when it tries to divide an integer by zero.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#routing", + "href": "tutorials/quickstart_for_web_devs.html#routing", + "title": "Web Devs Quickstart", + "section": "Routing", + "text": "Routing\nFastHTML builds upon FastAPI’s friendly decorator pattern for specifying URLs, with extra features:\n\n\nmain.py\n\nfrom fasthtml.common import * \n\napp, rt = fast_app()\n\n1@rt(\"/\")\ndef get():\n return Titled(\"FastHTML\", P(\"Let's do this!\"))\n\n2@rt(\"/hello\")\ndef get():\n return Titled(\"Hello, world!\")\n\nserve()\n\n\n1\n\nThe “/” URL on line 5 is the home of a project. This would be accessed at 127.0.0.1:5001.\n\n2\n\n“/hello” URL on line 9 will be found by the project if the user visits 127.0.0.1:5001/hello.\n\n\n\n\n\n\n\n\nTip\n\n\n\nIt looks like get() is being defined twice, but that’s not the case. Each function decorated with rt is totally separate, and is injected into the router. We’re not calling them in the module’s namespace (locals()). Rather, we’re loading them into the routing mechanism using the rt decorator.\n\n\nYou can do more! Read on to learn what we can do to make parts of the URL dynamic.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#variables-in-urls", + "href": "tutorials/quickstart_for_web_devs.html#variables-in-urls", + "title": "Web Devs Quickstart", + "section": "Variables in URLs", + "text": "Variables in URLs\nYou can add variable sections to a URL by marking them with {variable_name}. Your function then receives the {variable_name} as a keyword argument, but only if it is the correct type. Here’s an example:\n\n\nmain.py\n\nfrom fasthtml.common import * \n\napp, rt = fast_app()\n\n1@rt(\"/{name}/{age}\")\n2def get(name: str, age: int):\n3 return Titled(f\"Hello {name.title()}, age {age}\")\n\nserve()\n\n\n1\n\nWe specify two variable names, name and age.\n\n2\n\nWe define two function arguments named identically to the variables. You will note that we specify the Python types to be passed.\n\n3\n\nWe use these functions in our project.\n\n\nTry it out by going to this address: 127.0.0.1:5001/uma/5. You should get a page that says,\n\n“Hello Uma, age 5”.\n\n\nWhat happens if we enter incorrect data?\nThe 127.0.0.1:5001/uma/5 URL works because 5 is an integer. If we enter something that is not, such as 127.0.0.1:5001/uma/five, then FastHTML will return an error instead of a web page.\n\n\n\n\n\n\nFastHTML URL routing supports more complex types\n\n\n\nThe two examples we provide here use Python’s built-in str and int types, but you can use your own types, including more complex ones such as those defined by libraries like attrs, pydantic, and even sqlmodel.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#http-methods", + "href": "tutorials/quickstart_for_web_devs.html#http-methods", + "title": "Web Devs Quickstart", + "section": "HTTP Methods", + "text": "HTTP Methods\nFastHTML matches function names to HTTP methods. So far the URL routes we’ve defined have been for HTTP GET methods, the most common method for web pages.\nForm submissions often are sent as HTTP POST. When dealing with more dynamic web page designs, also known as Single Page Apps (SPA for short), the need can arise for other methods such as HTTP PUT and HTTP DELETE. The way FastHTML handles this is by changing the function name.\n\n\nmain.py\n\nfrom fasthtml.common import * \n\napp, rt = fast_app()\n\n@rt(\"/\") \n1def get():\n return Titled(\"HTTP GET\", P(\"Handle GET\"))\n\n@rt(\"/\") \n2def post():\n return Titled(\"HTTP POST\", P(\"Handle POST\"))\n\nserve()\n\n\n1\n\nOn line 6 because the get() function name is used, this will handle HTTP GETs going to the / URI.\n\n2\n\nOn line 10 because the post() function name is used, this will handle HTTP POSTs going to the / URI.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#css-files-and-inline-styles", + "href": "tutorials/quickstart_for_web_devs.html#css-files-and-inline-styles", + "title": "Web Devs Quickstart", + "section": "CSS Files and Inline Styles", + "text": "CSS Files and Inline Styles\nHere we modify default headers to demonstrate how to use the Sakura CSS microframework instead of FastHTML’s default of Pico CSS.\n\n\nmain.py\n\nfrom fasthtml.common import * \n\napp, rt = fast_app(\n1 pico=False,\n hdrs=(\n Link(rel='stylesheet', href='assets/normalize.min.css', type='text/css'),\n2 Link(rel='stylesheet', href='assets/sakura.css', type='text/css'),\n3 Style(\"p {color: red;}\")\n))\n\n@app.get(\"/\")\ndef home():\n return Titled(\"FastHTML\",\n P(\"Let's do this!\"),\n )\n\nserve()\n\n\n1\n\nBy setting pico to False, FastHTML will not include pico.min.css.\n\n2\n\nThis will generate an HTML <link> tag for sourcing the css for Sakura.\n\n3\n\nIf you want an inline styles, the Style() function will put the result into the HTML.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#other-static-media-file-locations", + "href": "tutorials/quickstart_for_web_devs.html#other-static-media-file-locations", + "title": "Web Devs Quickstart", + "section": "Other Static Media File Locations", + "text": "Other Static Media File Locations\nAs you saw, Script and Link are specific to the most common static media use cases in web apps: including JavaScript, CSS, and images. But it also works with videos and other static media files. The default behavior is to look for these files in the root directory - typically we don’t do anything special to include them. We can change the default directory that is looked in for files by adding the static_path parameter to the fast_app function.\napp, rt = fast_app(static_path='public')\nFastHTML also allows us to define a route that uses FileResponse to serve the file at a specified path. This is useful for serving images, videos, and other media files from a different directory without having to change the paths of many files. So if we move the directory containing the media files, we only need to change the path in one place. In the example below, we call images from a directory called public.\n@rt(\"/{fname:path}.{ext:static}\")\nasync def get(fname:str, ext:str): \n return FileResponse(f'public/{fname}.{ext}')", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#rendering-markdown", + "href": "tutorials/quickstart_for_web_devs.html#rendering-markdown", + "title": "Web Devs Quickstart", + "section": "Rendering Markdown", + "text": "Rendering Markdown\nfrom fasthtml.common import *\n\nhdrs = (MarkdownJS(), HighlightJS(langs=['python', 'javascript', 'html', 'css']), )\n\napp, rt = fast_app(hdrs=hdrs)\n\ncontent = \"\"\"\nHere are some _markdown_ elements.\n\n- This is a list item\n- This is another list item\n- And this is a third list item\n\n**Fenced code blocks work here.**\n\"\"\"\n\n@rt('/')\ndef get(req):\n return Titled(\"Markdown rendering example\", Div(content,cls=\"marked\"))\n\nserve()", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#code-highlighting", + "href": "tutorials/quickstart_for_web_devs.html#code-highlighting", + "title": "Web Devs Quickstart", + "section": "Code highlighting", + "text": "Code highlighting\nHere’s how to highlight code without any markdown configuration.\nfrom fasthtml.common import *\n\n# Add the HighlightJS built-in header\nhdrs = (HighlightJS(langs=['python', 'javascript', 'html', 'css']),)\n\napp, rt = fast_app(hdrs=hdrs)\n\ncode_example = \"\"\"\nimport datetime\nimport time\n\nfor i in range(10):\n print(f\"{datetime.datetime.now()}\")\n time.sleep(1)\n\"\"\"\n\n@rt('/')\ndef get(req):\n return Titled(\"Markdown rendering example\",\n Div(\n # The code example needs to be surrounded by\n # Pre & Code elements\n Pre(Code(code_example))\n ))\n\nserve()", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#defining-new-ft-components", + "href": "tutorials/quickstart_for_web_devs.html#defining-new-ft-components", + "title": "Web Devs Quickstart", + "section": "Defining new ft components", + "text": "Defining new ft components\nWe can build our own ft components and combine them with other components. The simplest method is defining them as a function.\n\nfrom fasthtml.common import *\n\n\ndef hero(title, statement):\n return Div(H1(title),P(statement), cls=\"hero\")\n\n# usage example\nMain(\n hero(\"Hello World\", \"This is a hero statement\")\n)\n\n<main> <div class=\"hero\">\n <h1>Hello World</h1>\n <p>This is a hero statement</p>\n </div>\n</main>\n\n\n\nPass through components\nFor when we need to define a new component that allows zero-to-many components to be nested within them, we lean on Python’s *args and **kwargs mechanism. Useful for creating page layout controls.\n\ndef layout(*args, **kwargs):\n \"\"\"Dashboard layout for all our dashboard views\"\"\"\n return Main(\n H1(\"Dashboard\"),\n Div(*args, **kwargs),\n cls=\"dashboard\",\n )\n\n# usage example\nlayout(\n Ul(*[Li(o) for o in range(3)]),\n P(\"Some content\", cls=\"description\"),\n)\n\n<main class=\"dashboard\"> <h1>Dashboard</h1>\n <div>\n <ul>\n <li>0</li>\n <li>1</li>\n <li>2</li>\n </ul>\n <p class=\"description\">Some content</p>\n </div>\n</main>\n\n\n\n\nDataclasses as ft components\nWhile functions are easy to read, for more complex components some might find it easier to use a dataclass.\n\nfrom dataclasses import dataclass\n\n@dataclass\nclass Hero:\n title: str\n statement: str\n \n def __ft__(self):\n \"\"\" The __ft__ method renders the dataclass at runtime.\"\"\"\n return Div(H1(self.title),P(self.statement), cls=\"hero\")\n \n# usage example\nMain(\n Hero(\"Hello World\", \"This is a hero statement\")\n)\n\n<main> <div class=\"hero\">\n <h1>Hello World</h1>\n <p>This is a hero statement</p>\n </div>\n</main>", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#testing-views-in-notebooks", + "href": "tutorials/quickstart_for_web_devs.html#testing-views-in-notebooks", + "title": "Web Devs Quickstart", + "section": "Testing views in notebooks", + "text": "Testing views in notebooks\nBecause of the ASGI event loop it is currently impossible to run FastHTML inside a notebook. However, we can still test the output of our views. To do this, we leverage Starlette, an ASGI toolkit that FastHTML uses.\n\n# First we instantiate our app, in this case we remove the\n# default headers to reduce the size of the output.\napp, rt = fast_app(default_hdrs=False)\n\n# Setting up the Starlette test client\nfrom starlette.testclient import TestClient\nclient = TestClient(app)\n\n# Usage example\n@rt(\"/\")\ndef get():\n return Titled(\"FastHTML is awesome\", \n P(\"The fastest way to create web apps in Python\"))\n\nprint(client.get(\"/\").text)\n\n <!doctype html>\n <html>\n <head>\n<title>FastHTML is awesome</title> </head>\n <body>\n<main class=\"container\"> <h1>FastHTML is awesome</h1>\n <p>The fastest way to create web apps in Python</p>\n</main> </body>\n </html>", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#forms", + "href": "tutorials/quickstart_for_web_devs.html#forms", + "title": "Web Devs Quickstart", + "section": "Forms", + "text": "Forms\nTo validate data coming from users, first define a dataclass representing the data you want to check. Here’s an example representing a signup form.\n\nfrom dataclasses import dataclass\n\n@dataclass\nclass Profile: email:str; phone:str; age:int\n\nCreate an FT component representing an empty version of that form. Don’t pass in any value to fill the form, that gets handled later.\n\nprofile_form = Form(method=\"post\", action=\"/profile\")(\n Fieldset(\n Label('Email', Input(name=\"email\")),\n Label(\"Phone\", Input(name=\"phone\")),\n Label(\"Age\", Input(name=\"age\")),\n ),\n Button(\"Save\", type=\"submit\"),\n )\nprofile_form\n\n<form enctype=\"multipart/form-data\" method=\"post\" action=\"/profile\"><fieldset><label>Email <input name=\"email\">\n</label><label>Phone <input name=\"phone\">\n</label><label>Age <input name=\"age\">\n</label></fieldset><button type=\"submit\">Save</button></form>\n\n\nOnce the dataclass and form function are completed, we can add data to the form. To do that, instantiate the profile dataclass:\n\nprofile = Profile(email='john@example.com', phone='123456789', age=5)\nprofile\n\nProfile(email='john@example.com', phone='123456789', age=5)\n\n\nThen add that data to the profile_form using FastHTML’s fill_form class:\n\nfill_form(profile_form, profile)\n\n<form enctype=\"multipart/form-data\" method=\"post\" action=\"/profile\"><fieldset><label>Email <input name=\"email\" value=\"john@example.com\">\n</label><label>Phone <input name=\"phone\" value=\"123456789\">\n</label><label>Age <input name=\"age\" value=\"5\">\n</label></fieldset><button type=\"submit\">Save</button></form>\n\n\n\nForms with views\nThe usefulness of FastHTML forms becomes more apparent when they are combined with FastHTML views. We’ll show how this works by using the test client from above. First, let’s create a SQlite database:\n\ndb = database(\"profiles.db\")\nprofiles = db.create(Profile, pk=\"email\")\n\nNow we insert a record into the database:\n\nprofiles.insert(profile)\n\nProfile(email='john@example.com', phone='123456789', age=5)\n\n\nAnd we can then demonstrate in the code that form is filled and displayed to the user.\n\n@rt(\"/profile/{email}\")\ndef profile(email:str):\n1 profile = profiles[email]\n2 filled_profile_form = fill_form(profile_form, profile)\n return Titled(f'Profile for {profile.email}', filled_profile_form)\n\nprint(client.get(f\"/profile/john@example.com\").text)\n\n\n1\n\nFetch the profile using the profile table’s email primary key\n\n2\n\nFill the form for display.\n\n\n\n\n <!doctype html>\n <html>\n <head>\n<title>Profile for john@example.com</title> </head>\n <body>\n<main class=\"container\"> <h1>Profile for john@example.com</h1>\n<form enctype=\"multipart/form-data\" method=\"post\" action=\"/profile\"><fieldset><label>Email <input name=\"email\" value=\"john@example.com\">\n</label><label>Phone <input name=\"phone\" value=\"123456789\">\n</label><label>Age <input name=\"age\" value=\"5\">\n</label></fieldset><button type=\"submit\">Save</button></form></main> </body>\n </html>\n\n\n\nAnd now let’s demonstrate making a change to the data.\n\n@rt(\"/profile\")\n1def post(profile: Profile):\n2 profiles.update(profile)\n3 return RedirectResponse(url=f\"/profile/{profile.email}\")\n\nnew_data = dict(email='john@example.com', phone='7654321', age=25)\n4print(client.post(\"/profile\", data=new_data).text)\n\n\n1\n\nWe use the Profile dataclass definition to set the type for the incoming profile content. This validates the field types for the incoming data\n\n2\n\nTaking our validated data, we updated the profiles table\n\n3\n\nWe redirect the user back to their profile view\n\n4\n\nThe display is of the profile form view showing the changes in data.\n\n\n\n\n <!doctype html>\n <html>\n <head>\n<title>Profile for john@example.com</title> </head>\n <body>\n<main class=\"container\"> <h1>Profile for john@example.com</h1>\n<form enctype=\"multipart/form-data\" method=\"post\" action=\"/profile\"><fieldset><label>Email <input name=\"email\" value=\"john@example.com\">\n</label><label>Phone <input name=\"phone\" value=\"7654321\">\n</label><label>Age <input name=\"age\" value=\"25\">\n</label></fieldset><button type=\"submit\">Save</button></form></main> </body>\n </html>", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#strings-and-conversion-order", + "href": "tutorials/quickstart_for_web_devs.html#strings-and-conversion-order", + "title": "Web Devs Quickstart", + "section": "Strings and conversion order", + "text": "Strings and conversion order\nThe general rules for rendering are: - __ft__ method will be called (for default components like P, H2, etc. or if you define your own components) - If you pass a string, it will be escaped - On other python objects, str() will be called\nAs a consequence, if you want to include plain HTML tags directly into e.g. a Div() they will get escaped by default (as a security measure to avoid code injections). This can be avoided by using NotStr(), a convenient way to reuse python code that returns already HTML. If you use pandas, you can use pandas.DataFrame.to_html() to get a nice table. To include the output a FastHTML, wrap it in NotStr(), like Div(NotStr(df.to_html())).\nAbove we saw how a dataclass behaves with the __ft__ method defined. On a plain dataclass, str() will be called (but not escaped).\n\nfrom dataclasses import dataclass\n\n@dataclass\nclass Hero:\n title: str\n statement: str\n \n# rendering the dataclass with the default method\nMain(\n Hero(\"<h1>Hello World</h1>\", \"This is a hero statement\")\n)\n\n<main>Hero(title='<h1>Hello World</h1>', statement='This is a hero statement')</main>\n\n\n\n# This will display the HTML as text on your page\nDiv(\"Let's include some HTML here: <div>Some HTML</div>\")\n\n<div>Let's include some HTML here: <div>Some HTML</div></div>\n\n\n\n# Keep the string untouched, will be rendered on the page\nDiv(NotStr(\"<div><h1>Some HTML</h1></div>\"))\n\n<div><div><h1>Some HTML</h1></div></div>", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#custom-exception-handlers", + "href": "tutorials/quickstart_for_web_devs.html#custom-exception-handlers", + "title": "Web Devs Quickstart", + "section": "Custom exception handlers", + "text": "Custom exception handlers\nFastHTML allows customization of exception handlers, but does so gracefully. What this means is by default it includes all the <html> tags needed to display attractive content. Try it out!\nfrom fasthtml.common import *\n\ndef not_found(req, exc): return Titled(\"404: I don't exist!\")\n\nexception_handlers = {404: not_found}\n\napp, rt = fast_app(exception_handlers=exception_handlers)\n\n@rt('/')\ndef get():\n return (Titled(\"Home page\", P(A(href=\"/oops\")(\"Click to generate 404 error\"))))\n\nserve()\nWe can also use lambda to make things more terse:\nfrom fasthtml.common import *\n\nexception_handlers={\n 404: lambda req, exc: Titled(\"404: I don't exist!\"),\n 418: lambda req, exc: Titled(\"418: I'm a teapot!\")\n}\n\napp, rt = fast_app(exception_handlers=exception_handlers)\n\n@rt('/')\ndef get():\n return (Titled(\"Home page\", P(A(href=\"/oops\")(\"Click to generate 404 error\"))))\n\nserve()", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#cookies", + "href": "tutorials/quickstart_for_web_devs.html#cookies", + "title": "Web Devs Quickstart", + "section": "Cookies", + "text": "Cookies\nWe can set cookies using the cookie() function. In our example, we’ll create a timestamp cookie.\n\nfrom datetime import datetime\nfrom IPython.display import HTML\n\n\n@rt(\"/settimestamp\")\ndef get(req):\n now = datetime.now()\n return P(f'Set to {now}'), cookie('now', datetime.now())\n\nHTML(client.get('/settimestamp').text)\n\n \n \n \nFastHTML page \n \n Set to 2024-09-26 15:33:48.141869\n \n \n\n\nNow let’s get it back using the same name for our parameter as the cookie name.\n\n@rt('/gettimestamp')\ndef get(now:parsed_date): return f'Cookie was set at time {now.time()}'\n\nclient.get('/gettimestamp').text\n\n'Cookie was set at time 15:33:48.141903'", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#sessions", + "href": "tutorials/quickstart_for_web_devs.html#sessions", + "title": "Web Devs Quickstart", + "section": "Sessions", + "text": "Sessions\nFor convenience and security, FastHTML has a mechanism for storing small amounts of data in the user’s browser. We can do this by adding a session argument to routes. FastHTML sessions are Python dictionaries, and we can leverage to our benefit. The example below shows how to concisely set and get sessions.\n\n@rt('/adder/{num}')\ndef get(session, num: int):\n session.setdefault('sum', 0)\n session['sum'] = session.get('sum') + num\n return Response(f'The sum is {session[\"sum\"]}.')", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#toasts-also-known-as-messages", + "href": "tutorials/quickstart_for_web_devs.html#toasts-also-known-as-messages", + "title": "Web Devs Quickstart", + "section": "Toasts (also known as Messages)", + "text": "Toasts (also known as Messages)\nToasts, sometimes called “Messages” are small notifications usually in colored boxes used to notify users that something has happened. Toasts can be of four types:\n\ninfo\nsuccess\nwarning\nerror\n\nExamples toasts might include:\n\n“Payment accepted”\n“Data submitted”\n“Request approved”\n\nToasts require the use of the setup_toasts() function plus every view needs these two features:\n\nThe session argument\nMust return FT components\n\n1setup_toasts(app)\n\n@rt('/toasting')\n2def get(session):\n # Normally one toast is enough, this allows us to see\n # different toast types in action.\n add_toast(session, f\"Toast is being cooked\", \"info\")\n add_toast(session, f\"Toast is ready\", \"success\")\n add_toast(session, f\"Toast is getting a bit crispy\", \"warning\")\n add_toast(session, f\"Toast is burning!\", \"error\")\n3 return Titled(\"I like toast\")\n\n1\n\nsetup_toasts is a helper function that adds toast dependencies. Usually this would be declared right after fast_app()\n\n2\n\nToasts require sessions\n\n3\n\nViews with Toasts must return FT or FtResponse components.\n\n\n💡 setup_toasts takes a duration input that allows you to specify how long a toast will be visible before disappearing. For example setup_toasts(duration=5) sets the toasts duration to 5 seconds. By default toasts disappear after 10 seconds.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#authentication-and-authorization", + "href": "tutorials/quickstart_for_web_devs.html#authentication-and-authorization", + "title": "Web Devs Quickstart", + "section": "Authentication and authorization", + "text": "Authentication and authorization\nIn FastHTML the tasks of authentication and authorization are handled with Beforeware. Beforeware are functions that run before the route handler is called. They are useful for global tasks like ensuring users are authenticated or have permissions to access a view.\nFirst, we write a function that accepts a request and session arguments:\n\n# Status code 303 is a redirect that can change POST to GET,\n# so it's appropriate for a login page.\nlogin_redir = RedirectResponse('/login', status_code=303)\n\ndef user_auth_before(req, sess):\n # The `auth` key in the request scope is automatically provided\n # to any handler which requests it, and can not be injected\n # by the user using query params, cookies, etc, so it should\n # be secure to use. \n auth = req.scope['auth'] = sess.get('auth', None)\n # If the session key is not there, it redirects to the login page.\n if not auth: return login_redir\n\nNow we pass our user_auth_before function as the first argument into a Beforeware class. We also pass a list of regular expressions to the skip argument, designed to allow users to still get to the home and login pages.\n\nbeforeware = Beforeware(\n user_auth_before,\n skip=[r'/favicon\\.ico', r'/static/.*', r'.*\\.css', r'.*\\.js', '/login', '/']\n)\n\napp, rt = fast_app(before=beforeware)", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#server-sent-events-sse", + "href": "tutorials/quickstart_for_web_devs.html#server-sent-events-sse", + "title": "Web Devs Quickstart", + "section": "Server-sent events (SSE)", + "text": "Server-sent events (SSE)\nWith server-sent events, it’s possible for a server to send new data to a web page at any time, by pushing messages to the web page. Unlike WebSockets, SSE can only go in one direction: server to client. SSE is also part of the HTTP specification unlike WebSockets which uses its own specification.\nFastHTML introduces several tools for working with SSE which are covered in the example below. While concise, there’s a lot going on in this function so we’ve annotated it quite a bit.\n\nimport random\nfrom asyncio import sleep\nfrom fasthtml.common import *\n\n1hdrs=(Script(src=\"https://unpkg.com/htmx-ext-sse@2.2.1/sse.js\"),)\napp,rt = fast_app(hdrs=hdrs)\n\n@rt\ndef index():\n return Titled(\"SSE Random Number Generator\",\n P(\"Generate pairs of random numbers, as the list grows scroll downwards.\"),\n2 Div(hx_ext=\"sse\",\n3 sse_connect=\"/number-stream\",\n4 hx_swap=\"beforeend show:bottom\",\n5 sse_swap=\"message\"))\n\n6shutdown_event = signal_shutdown()\n\n7async def number_generator():\n8 while not shutdown_event.is_set():\n data = Article(random.randint(1, 100))\n9 yield sse_message(data)\n await sleep(1)\n\n@rt(\"/number-stream\")\n10async def get(): return EventStream(number_generator())\n\n\n1\n\nImport the HTMX SSE extension\n\n2\n\nTell HTMX to load the SSE extension\n\n3\n\nLook at the /number-stream endpoint for SSE content\n\n4\n\nWhen new items come in from the SSE endpoint, add them at the end of the current content within the div. If they go beyond the screen, scroll downwards\n\n5\n\nSpecify the name of the event. FastHTML’s default event name is “message”. Only change if you have more than one call to SSE endpoints within a view\n\n6\n\nSet up the asyncio event loop\n\n7\n\nDon’t forget to make this an async function!\n\n8\n\nIterate through the asyncio event loop\n\n9\n\nWe yield the data. Data ideally should be comprised of FT components as that plugs nicely into HTMX in the browser\n\n10\n\nThe endpoint view needs to be an async function that returns a EventStream", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#websockets", + "href": "tutorials/quickstart_for_web_devs.html#websockets", + "title": "Web Devs Quickstart", + "section": "Websockets", + "text": "Websockets\nWith websockets we can have bi-directional communications between a browser and client. Websockets are useful for things like chat and certain types of games. While websockets can be used for single direction messages from the server (i.e. telling users that a process is finished), that task is arguably better suited for SSE.\nFastHTML provides useful tools for adding websockets to your pages.\n\nfrom fasthtml.common import *\nfrom asyncio import sleep\n\n1app, rt = fast_app(exts='ws')\n\n2def mk_inp(): return Input(id='msg', autofocus=True)\n\n@rt('/')\nasync def get(request):\n cts = Div(\n Div(id='notifications'),\n3 Form(mk_inp(), id='form', ws_send=True),\n4 hx_ext='ws', ws_connect='/ws')\n return Titled('Websocket Test', cts)\n\n5async def on_connect(send):\n print('Connected!')\n6 await send(Div('Hello, you have connected', id=\"notifications\"))\n\n7async def on_disconnect(ws):\n print('Disconnected!')\n\n8@app.ws('/ws', conn=on_connect, disconn=on_disconnect)\n9async def ws(msg:str, send):\n10 await send(Div('Hello ' + msg, id=\"notifications\"))\n await sleep(2)\n11 return Div('Goodbye ' + msg, id=\"notifications\"), mk_inp()\n\n\n1\n\nTo use websockets in FastHTML, you must instantiate the app with exts set to ‘ws’\n\n2\n\nAs we want to use websockets to reset the form, we define the mk_input function that can be called from multiple locations\n\n3\n\nWe create the form and mark it with the ws_send attribute, which is documented here in the HTMX websocket specification. This tells HTMX to send a message to the nearest websocket based on the trigger for the form element, which for forms is pressing the enter key, an action considered to be a form submission\n\n4\n\nThis is where the HTMX extension is loaded (hx_ext='ws') and the nearest websocket is defined (ws_connect='/ws')\n\n5\n\nWhen a websocket first connects we can optionally have it call a function that accepts a send argument. The send argument will push a message to the browser.\n\n6\n\nHere we use the send function that was passed into the on_connect function to send a Div with an id of notifications that HTMX assigns to the element in the page that already has an id of notifications\n\n7\n\nWhen a websocket disconnects we can call a function which takes no arguments. Typically the role of this function is to notify the server to take an action. In this case, we print a simple message to the console\n\n8\n\nWe use the app.ws decorator to mark that /ws is the route for our websocket. We also pass in the two optional conn and disconn parameters to this decorator. As a fun experiment, remove the conn and disconn arguments and see what happens\n\n9\n\nDefine the ws function as async. This is necessary for ASGI to be able to serve websockets. The function accepts two arguments, a msg that is user input from the browser, and a send function for pushing data back to the browser\n\n10\n\nThe send function is used here to send HTML back to the page. As the HTML has an id of notifications, HTMX will overwrite what is already on the page with the same ID\n\n11\n\nThe websocket function can also be used to return a value. In this case, it is a tuple of two HTML elements. HTMX will take the elements and replace them where appropriate. As both have id specified (notifications and msg respectively), they will replace their predecessor on the page.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/quickstart_for_web_devs.html#file-uploads", + "href": "tutorials/quickstart_for_web_devs.html#file-uploads", + "title": "Web Devs Quickstart", + "section": "File Uploads", + "text": "File Uploads\nA common task in web development is uploading files. The examples below are for uploading files to the hosting server, with information about the uploaded file presented to the user.\n\n\n\n\n\n\nFile uploads in production can be dangerous\n\n\n\nFile uploads can be the target of abuse, accidental or intentional. That means users may attempt to upload files that are too large or present a security risk. This is especially of concern for public facing apps. File upload security is outside the scope of this tutorial, for now we suggest reading the OWASP File Upload Cheat Sheet.\n\n\n\nSingle File Uploads\nfrom fasthtml.common import *\nfrom pathlib import Path\n\napp, rt = fast_app()\n\nupload_dir = Path(\"filez\")\nupload_dir.mkdir(exist_ok=True)\n\n@rt('/')\ndef get():\n return Titled(\"File Upload Demo\",\n Article(\n1 Form(hx_post=upload, hx_target=\"#result-one\")(\n2 Input(type=\"file\", name=\"file\"),\n Button(\"Upload\", type=\"submit\", cls='secondary'),\n ),\n Div(id=\"result-one\")\n )\n )\n\ndef FileMetaDataCard(file):\n return Article(\n Header(H3(file.filename)),\n Ul(\n Li('Size: ', file.size), \n Li('Content Type: ', file.content_type),\n Li('Headers: ', file.headers),\n )\n ) \n\n@rt\n3async def upload(file: UploadFile):\n4 card = FileMetaDataCard(file)\n5 filebuffer = await file.read()\n6 (upload_dir / file.filename).write_bytes(filebuffer)\n return card\n\nserve()\n\n1\n\nEvery form rendered with the Form FT component defaults to enctype=\"multipart/form-data\"\n\n2\n\nDon’t forget to set the Input FT Component’s type to file\n\n3\n\nThe upload view should receive a Starlette UploadFile type. You can add other form variables\n\n4\n\nWe can access the metadata of the card (filename, size, content_type, headers), a quick and safe process. We set that to the card variable\n\n5\n\nIn order to access the contents contained within a file we use the await method to read() it. As files may be quite large or contain bad data, this is a seperate step from accessing metadata\n\n6\n\nThis step shows how to use Python’s built-in pathlib.Path library to write the file to disk.\n\n\n\n\nMultiple File Uploads\nfrom fasthtml.common import *\nfrom pathlib import Path\n\napp, rt = fast_app()\n\nupload_dir = Path(\"filez\")\nupload_dir.mkdir(exist_ok=True)\n\n@rt('/')\ndef get():\n return Titled(\"Multiple File Upload Demo\",\n Article(\n1 Form(hx_post=upload_many, hx_target=\"#result-many\")(\n2 Input(type=\"file\", name=\"files\", multiple=True),\n Button(\"Upload\", type=\"submit\", cls='secondary'),\n ),\n Div(id=\"result-many\")\n )\n )\n\ndef FileMetaDataCard(file):\n return Article(\n Header(H3(file.filename)),\n Ul(\n Li('Size: ', file.size), \n Li('Content Type: ', file.content_type),\n Li('Headers: ', file.headers),\n )\n ) \n\n@rt\n3async def upload_many(files: list[UploadFile]):\n cards = []\n4 for file in files:\n5 cards.append(FileMetaDataCard(file))\n6 filebuffer = await file.read()\n7 (upload_dir / file.filename).write_bytes(filebuffer)\n return cards\n\nserve()\n\n1\n\nEvery form rendered with the Form FT component defaults to enctype=\"multipart/form-data\"\n\n2\n\nDon’t forget to set the Input FT Component’s type to file and assign the multiple attribute to True\n\n3\n\nThe upload view should receive a list containing the Starlette UploadFile type. You can add other form variables\n\n4\n\nIterate through the files\n\n5\n\nWe can access the metadata of the card (filename, size, content_type, headers), a quick and safe process. We add that to the cards variable\n\n6\n\nIn order to access the contents contained within a file we use the await method to read() it. As files may be quite large or contain bad data, this is a seperate step from accessing metadata\n\n7\n\nThis step shows how to use Python’s built-in pathlib.Path library to write the file to disk.", + "crumbs": [ + "Home", + "Tutorials", + "Web Devs Quickstart" + ] + }, + { + "objectID": "tutorials/index.html", + "href": "tutorials/index.html", + "title": "Tutorials", + "section": "", + "text": "Click through to any of these tutorials to get started with FastHTML’s features.\n\n\n\n\n\n\n\n\n\nTitle\n\n\nDescription\n\n\n\n\n\n\nFastHTML By Example\n\n\nAn introduction to FastHTML from the ground up, with four complete examples\n\n\n\n\nWeb Devs Quickstart\n\n\nA fast introduction to FastHTML for experienced web developers.\n\n\n\n\nJS App Walkthrough\n\n\nHow to build a website with custom JavaScript in FastHTML step-by-step\n\n\n\n\nUsing Jupyter to write FastHTML\n\n\nWriting FastHTML applications in Jupyter notebooks requires a slightly different process than normal Python applications.\n\n\n\n\n\nNo matching items", + "crumbs": [ + "Home", + "Tutorials" + ] + }, + { + "objectID": "tutorials/e2e.html", + "href": "tutorials/e2e.html", + "title": "JS App Walkthrough", + "section": "", + "text": "You’ll need the following software to complete the tutorial, read on for specific installation instructions:\n\nPython\nA Python package manager such as pip (which normally comes with Python) or uv\nFastHTML\nWeb browser\nRailway.app account\n\nIf you haven’t worked with Python before, we recommend getting started with Miniconda.\nNote that you will only need to follow the steps in the installation section once per environment. If you create a new repo, you won’t need to redo these.\n\n\nFor Mac, Windows and Linux, enter:\npip install python-fasthtml", + "crumbs": [ + "Home", + "Tutorials", + "JS App Walkthrough" + ] + }, + { + "objectID": "tutorials/e2e.html#installation", + "href": "tutorials/e2e.html#installation", + "title": "JS App Walkthrough", + "section": "", + "text": "You’ll need the following software to complete the tutorial, read on for specific installation instructions:\n\nPython\nA Python package manager such as pip (which normally comes with Python) or uv\nFastHTML\nWeb browser\nRailway.app account\n\nIf you haven’t worked with Python before, we recommend getting started with Miniconda.\nNote that you will only need to follow the steps in the installation section once per environment. If you create a new repo, you won’t need to redo these.\n\n\nFor Mac, Windows and Linux, enter:\npip install python-fasthtml", + "crumbs": [ + "Home", + "Tutorials", + "JS App Walkthrough" + ] + }, + { + "objectID": "tutorials/e2e.html#first-steps", + "href": "tutorials/e2e.html#first-steps", + "title": "JS App Walkthrough", + "section": "First steps", + "text": "First steps\nBy the end of this section you’ll have your own FastHTML website with tests deployed to railway.app.\n\nCreate a hello world\nCreate a new folder to organize all the files for your project. Inside this folder, create a file called main.py and add the following code to it:\n\n\nmain.py\n\nfrom fasthtml.common import *\n\napp = FastHTML()\nrt = app.route\n\n@rt('/')\ndef get():\n return 'Hello, world!'\n\nserve()\n\nFinally, run python main.py in your terminal and open your browser to the ‘Link’ that appears.\n\n\nQuickDraw: A FastHTML Adventure 🎨✨\nThe end result of this tutorial will be QuickDraw, a real-time collaborative drawing app using FastHTML. Here is what the final site will look like:\n\n\n\nQuickDraw\n\n\n\nDrawing Rooms\nDrawing rooms are the core concept of our application. Each room represents a separate drawing space where a user can let their inner Picasso shine. Here’s a detailed breakdown:\n\nRoom Creation and Storage\n\n\n\nmain.py\n\ndb = database('data/drawapp.db')\nrooms = db.t.rooms\nif rooms not in db.t:\n rooms.create(id=int, name=str, created_at=str, pk='id')\nRoom = rooms.dataclass()\n\n@patch\ndef __ft__(self:Room):\n return Li(A(self.name, href=f\"/rooms/{self.id}\"))\n\nOr you can use our fast_app function to create a FastHTML app with a SQLite database and dataclass in one line:\n\n\nmain.py\n\ndef render(room):\n return Li(A(room.name, href=f\"/rooms/{room.id}\"))\n\napp,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, pk='id')\n\nWe are specifying a render function to convert our dataclass into HTML, which is the same as extending the __ft__ method from the patch decorator we used before. We will use this method for the rest of the tutorial since it is a lot cleaner and easier to read.\n\nWe’re using a SQLite database (via FastLite) to store our rooms.\nEach room has an id (integer), a name (string), and a created_at timestamp (string).\nThe Room dataclass is automatically generated based on this structure.\n\n\nCreating a room\n\n\n\nmain.py\n\n@rt(\"/\")\ndef get():\n # The 'Input' id defaults to the same as the name, so you can omit it if you wish\n create_room = Form(Input(id=\"name\", name=\"name\", placeholder=\"New Room Name\"),\n Button(\"Create Room\"),\n hx_post=\"/rooms\", hx_target=\"#rooms-list\", hx_swap=\"afterbegin\")\n rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list')\n return Titled(\"DrawCollab\", \n H1(\"DrawCollab\"),\n create_room, rooms_list)\n\n@rt(\"/rooms\")\nasync def post(room:Room):\n room.created_at = datetime.now().isoformat()\n return rooms.insert(room)\n\n\nWhen a user submits the “Create Room” form, this route is called.\nIt creates a new Room object, sets the creation time, and inserts it into the database.\nIt returns an HTML list item with a link to the new room, which is dynamically added to the room list on the homepage thanks to HTMX.\n\n\nLet’s give our rooms shape\n\n\n\nmain.py\n\n@rt(\"/rooms/{id}\")\nasync def get(id:int):\n room = rooms[id]\n return Titled(f\"Room: {room.name}\", H1(f\"Welcome to {room.name}\"), A(Button(\"Leave Room\"), href=\"/\"))\n\n\nThis route renders the interface for a specific room.\nIt fetches the room from the database and renders a title, heading, and paragraph.\n\nHere is the full code so far:\n\n\nmain.py\n\nfrom fasthtml.common import *\nfrom datetime import datetime\n\ndef render(room):\n return Li(A(room.name, href=f\"/rooms/{room.id}\"))\n\napp,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, pk='id')\n\n@rt(\"/\")\ndef get():\n create_room = Form(Input(id=\"name\", name=\"name\", placeholder=\"New Room Name\"),\n Button(\"Create Room\"),\n hx_post=\"/rooms\", hx_target=\"#rooms-list\", hx_swap=\"afterbegin\")\n rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list')\n return Titled(\"DrawCollab\", create_room, rooms_list)\n\n@rt(\"/rooms\")\nasync def post(room:Room):\n room.created_at = datetime.now().isoformat()\n return rooms.insert(room)\n\n@rt(\"/rooms/{id}\")\nasync def get(id:int):\n room = rooms[id]\n return Titled(f\"Room: {room.name}\", H1(f\"Welcome to {room.name}\"), A(Button(\"Leave Room\"), href=\"/\"))\n\nserve()\n\nNow run python main.py in your terminal and open your browser to the ‘Link’ that appears. You should see a page with a form to create a new room and a list of existing rooms.\n\n\nThe Canvas - Let’s Get Drawing! 🖌️\nTime to add the actual drawing functionality. We’ll use Fabric.js for this:\n\n\nmain.py\n\n# ... (keep the previous imports and database setup)\n\n@rt(\"/rooms/{id}\")\nasync def get(id:int):\n room = rooms[id]\n canvas = Canvas(id=\"canvas\", width=\"800\", height=\"600\")\n color_picker = Input(type=\"color\", id=\"color-picker\", value=\"#3CDD8C\")\n brush_size = Input(type=\"range\", id=\"brush-size\", min=\"1\", max=\"50\", value=\"10\")\n \n js = \"\"\"\n var canvas = new fabric.Canvas('canvas');\n canvas.isDrawingMode = true;\n canvas.freeDrawingBrush.color = '#3CDD8C';\n canvas.freeDrawingBrush.width = 10;\n \n document.getElementById('color-picker').onchange = function() {\n canvas.freeDrawingBrush.color = this.value;\n };\n \n document.getElementById('brush-size').oninput = function() {\n canvas.freeDrawingBrush.width = parseInt(this.value, 10);\n };\n \"\"\"\n \n return Titled(f\"Room: {room.name}\",\n A(Button(\"Leave Room\"), href=\"/\"),\n canvas,\n Div(color_picker, brush_size),\n Script(src=\"https://cdnjs.cloudflare.com/ajax/libs/fabric.js/5.3.1/fabric.min.js\"),\n Script(js))\n\n# ... (keep the serve() part)\n\nNow we’ve got a drawing canvas! FastHTML makes it easy to include external libraries and add custom JavaScript.\n\n\nSaving and Loading Canvases 💾\nNow that we have a working drawing canvas, let’s add the ability to save and load drawings. We’ll modify our database schema to include a canvas_data field, and add new routes for saving and loading canvas data. Here’s how we’ll update our code:\n\nModify the database schema:\n\n\n\nmain.py\n\napp,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, canvas_data=str, pk='id')\n\n\nAdd a save button that grabs the canvas’ state and sends it to the server:\n\n\n\nmain.py\n\n@rt(\"/rooms/{id}\")\nasync def get(id:int):\n room = rooms[id]\n canvas = Canvas(id=\"canvas\", width=\"800\", height=\"600\")\n color_picker = Input(type=\"color\", id=\"color-picker\", value=\"#3CDD8C\")\n brush_size = Input(type=\"range\", id=\"brush-size\", min=\"1\", max=\"50\", value=\"10\")\n save_button = Button(\"Save Canvas\", id=\"save-canvas\", hx_post=f\"/rooms/{id}/save\", hx_vals=\"js:{canvas_data: JSON.stringify(canvas.toJSON())}\")\n # ... (rest of the function remains the same)\n\n\nAdd routes for saving and loading canvas data:\n\n\n\nmain.py\n\n@rt(\"/rooms/{id}/save\")\nasync def post(id:int, canvas_data:str):\n rooms.update({'canvas_data': canvas_data}, id)\n return \"Canvas saved successfully\"\n\n@rt(\"/rooms/{id}/load\")\nasync def get(id:int):\n room = rooms[id]\n return room.canvas_data if room.canvas_data else \"{}\"\n\n\nUpdate the JavaScript to load existing canvas data:\n\n\n\nmain.py\n\njs = f\"\"\"\n var canvas = new fabric.Canvas('canvas');\n canvas.isDrawingMode = true;\n canvas.freeDrawingBrush.color = '#3CDD8C';\n canvas.freeDrawingBrush.width = 10;\n // Load existing canvas data\n fetch(`/rooms/{id}/load`)\n .then(response => response.json())\n .then(data => {{\n if (data && Object.keys(data).length > 0) {{\n canvas.loadFromJSON(data, canvas.renderAll.bind(canvas));\n }}\n }});\n \n // ... (rest of the JavaScript remains the same)\n\"\"\"\n\nWith these changes, users can now save their drawings and load them when they return to the room. The canvas data is stored as a JSON string in the database, allowing for easy serialization and deserialization. Try it out! Create a new room, make a drawing, save it, and then reload the page. You should see your drawing reappear, ready for further editing.\nHere is the completed code:\n\n\nmain.py\n\nfrom fasthtml.common import *\nfrom datetime import datetime\n\ndef render(room):\n return Li(A(room.name, href=f\"/rooms/{room.id}\"))\n\napp,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, canvas_data=str, pk='id')\n\n@rt(\"/\")\ndef get():\n create_room = Form(Input(id=\"name\", name=\"name\", placeholder=\"New Room Name\"),\n Button(\"Create Room\"),\n hx_post=\"/rooms\", hx_target=\"#rooms-list\", hx_swap=\"afterbegin\")\n rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list')\n return Titled(\"QuickDraw\", \n create_room, rooms_list)\n\n@rt(\"/rooms\")\nasync def post(room:Room):\n room.created_at = datetime.now().isoformat()\n return rooms.insert(room)\n\n@rt(\"/rooms/{id}\")\nasync def get(id:int):\n room = rooms[id]\n canvas = Canvas(id=\"canvas\", width=\"800\", height=\"600\")\n color_picker = Input(type=\"color\", id=\"color-picker\", value=\"#000000\")\n brush_size = Input(type=\"range\", id=\"brush-size\", min=\"1\", max=\"50\", value=\"10\")\n save_button = Button(\"Save Canvas\", id=\"save-canvas\", hx_post=f\"/rooms/{id}/save\", hx_vals=\"js:{canvas_data: JSON.stringify(canvas.toJSON())}\")\n\n js = f\"\"\"\n var canvas = new fabric.Canvas('canvas');\n canvas.isDrawingMode = true;\n canvas.freeDrawingBrush.color = '#000000';\n canvas.freeDrawingBrush.width = 10;\n\n // Load existing canvas data\n fetch(`/rooms/{id}/load`)\n .then(response => response.json())\n .then(data => {{\n if (data && Object.keys(data).length > 0) {{\n canvas.loadFromJSON(data, canvas.renderAll.bind(canvas));\n }}\n }});\n \n document.getElementById('color-picker').onchange = function() {{\n canvas.freeDrawingBrush.color = this.value;\n }};\n \n document.getElementById('brush-size').oninput = function() {{\n canvas.freeDrawingBrush.width = parseInt(this.value, 10);\n }};\n \"\"\"\n \n return Titled(f\"Room: {room.name}\",\n A(Button(\"Leave Room\"), href=\"/\"),\n canvas,\n Div(color_picker, brush_size, save_button),\n Script(src=\"https://cdnjs.cloudflare.com/ajax/libs/fabric.js/5.3.1/fabric.min.js\"),\n Script(js))\n\n@rt(\"/rooms/{id}/save\")\nasync def post(id:int, canvas_data:str):\n rooms.update({'canvas_data': canvas_data}, id)\n return \"Canvas saved successfully\"\n\n@rt(\"/rooms/{id}/load\")\nasync def get(id:int):\n room = rooms[id]\n return room.canvas_data if room.canvas_data else \"{}\"\n\nserve()\n\n\n\n\nDeploying to Railway\nYou can deploy your website to a number of hosting providers, for this tutorial we’ll be using Railway. To get started, make sure you create an account and install the Railway CLI. Once installed, make sure to run railway login to log in to your account.\nTo make deploying your website as easy as possible, FastHTMl comes with a built in CLI tool that will handle most of the deployment process for you. To deploy your website, run the following command in your terminal in the root directory of your project:\nfh_railway_deploy quickdraw\n\n\n\n\n\n\nNote\n\n\n\nYour app must be located in a main.py file for this to work.\n\n\n\n\nConclusion: You’re a FastHTML Artist Now! 🎨🚀\nCongratulations! You’ve just built a sleek, interactive web application using FastHTML. Let’s recap what we’ve learned:\n\nFastHTML allows you to create dynamic web apps with minimal code.\nWe used FastHTML’s routing system to handle different pages and actions.\nWe integrated with a SQLite database to store room information and canvas data.\nWe utilized Fabric.js to create an interactive drawing canvas.\nWe implemented features like color picking, brush size adjustment, and canvas saving.\nWe used HTMX for seamless, partial page updates without full reloads.\nWe learned how to deploy our FastHTML application to Railway for easy hosting.\n\nYou’ve taken your first steps into the world of FastHTML development. From here, the possibilities are endless! You could enhance the drawing app further by adding features like:\n\nImplementing different drawing tools (e.g., shapes, text)\nAdding user authentication\nCreating a gallery of saved drawings\nImplementing real-time collaborative drawing using WebSockets\n\nWhatever you choose to build next, FastHTML has got your back. Now go forth and create something awesome! Happy coding! 🖼️🚀", + "crumbs": [ + "Home", + "Tutorials", + "JS App Walkthrough" + ] + }, + { + "objectID": "tutorials/by_example.html", + "href": "tutorials/by_example.html", + "title": "FastHTML By Example", + "section": "", + "text": "This tutorial provides an alternate introduction to FastHTML by building out example applications. We also illustrate how to use FastHTML foundations to create custom web apps. Finally, this document serves as minimal context for a LLM to turn it into a FastHTML assistant.\nLet’s get started.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#fasthtml-basics", + "href": "tutorials/by_example.html#fasthtml-basics", + "title": "FastHTML By Example", + "section": "FastHTML Basics", + "text": "FastHTML Basics\nFastHTML is just Python. You can install it with pip install python-fasthtml. Extensions/components built for it can likewise be distributed via PyPI or as simple Python files.\nThe core usage of FastHTML is to define routes, and then to define what to do at each route. This is similar to the FastAPI web framework (in fact we implemented much of the functionality to match the FastAPI usage examples), but where FastAPI focuses on returning JSON data to build APIs, FastHTML focuses on returning HTML data.\nHere’s a simple FastHTML app that returns a “Hello, World” message:\n\nfrom fasthtml.common import FastHTML, serve\n\napp = FastHTML()\n\n@app.get(\"/\")\ndef home():\n return \"<h1>Hello, World</h1>\"\n\nserve()\n\nTo run this app, place it in a file, say app.py, and then run it with python app.py.\nINFO: Will watch for changes in these directories: ['/home/jonathan/fasthtml-example']\nINFO: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit)\nINFO: Started reloader process [871942] using WatchFiles\nINFO: Started server process [871945]\nINFO: Waiting for application startup.\nINFO: Application startup complete.\nIf you navigate to http://127.0.0.1:8000 in a browser, you’ll see your “Hello, World”. If you edit the app.py file and save it, the server will reload and you’ll see the updated message when you refresh the page in your browser.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#constructing-html", + "href": "tutorials/by_example.html#constructing-html", + "title": "FastHTML By Example", + "section": "Constructing HTML", + "text": "Constructing HTML\nNotice we wrote some HTML in the previous example. We don’t want to do that! Some web frameworks require that you learn HTML, CSS, JavaScript AND some templating language AND python. We want to do as much as possible with just one language. Fortunately, the Python module fastcore.xml has all we need for constructing HTML from Python, and FastHTML includes all the tags you need to get started. For example:\n\nfrom fasthtml.common import *\npage = Html(\n Head(Title('Some page')),\n Body(Div('Some text, ', A('A link', href='https://example.com'), Img(src=\"https://placehold.co/200\"), cls='myclass')))\nprint(to_xml(page))\n\n<!doctype html></!doctype>\n\n<html>\n <head>\n <title>Some page</title>\n </head>\n <body>\n <div class=\"myclass\">\nSome text, \n <a href=\"https://example.com\">A link</a>\n <img src=\"https://placehold.co/200\">\n </div>\n </body>\n</html>\n\n\n\n\nshow(page)\n\n\n\n\n \n Some page\n \n \n \nSome text, \n A link\n \n \n \n\n\n\nIf that import * worries you, you can always import only the tags you need.\nFastHTML is smart enough to know about fastcore.xml, and so you don’t need to use the to_xml function to convert your FT objects to HTML. You can just return them as you would any other Python object. For example, if we modify our previous example to use fastcore.xml, we can return an FT object directly:\n\nfrom fasthtml.common import *\napp = FastHTML()\n\n@app.get(\"/\")\ndef home():\n page = Html(\n Head(Title('Some page')),\n Body(Div('Some text, ', A('A link', href='https://example.com'), Img(src=\"https://placehold.co/200\"), cls='myclass')))\n return page\n\nserve()\n\nThis will render the HTML in the browser.\nFor debugging, you can right-click on the rendered HTML in the browser and select “Inspect” to see the underlying HTML that was generated. There you’ll also find the ‘network’ tab, which shows you the requests that were made to render the page. Refresh and look for the request to 127.0.0.1 - and you’ll see it’s just a GET request to /, and the response body is the HTML you just returned.\n\n\n\n\n\n\nLive Reloading\n\n\n\nYou can also enable live reloading so you don’t have to manually refresh your browser to view updates.\n\n\nYou can also use Starlette’s TestClient to try it out in a notebook:\n\nfrom starlette.testclient import TestClient\nclient = TestClient(app)\nr = client.get(\"/\")\nprint(r.text)\n\n<html>\n <head><title>Some page</title>\n</head>\n <body><div class=\"myclass\">\nSome text, \n <a href=\"https://example.com\">A link</a>\n <img src=\"https://placehold.co/200\">\n</div>\n</body>\n</html>\n\n\n\nFastHTML wraps things in an Html tag if you don’t do it yourself (unless the request comes from htmx, in which case you get the element directly). See FT objects and HTML for more on creating custom components or adding HTML rendering to existing Python objects. To give the page a non-default title, return a Title before your main content:\n\napp = FastHTML()\n\n@app.get(\"/\")\ndef home():\n return Title(\"Page Demo\"), Div(H1('Hello, World'), P('Some text'), P('Some more text'))\n\nclient = TestClient(app)\nprint(client.get(\"/\").text)\n\n<!doctype html></!doctype>\n\n<html>\n <head>\n <title>Page Demo</title>\n <meta charset=\"utf-8\"></meta>\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1, viewport-fit=cover\"></meta>\n <script src=\"https://unpkg.com/htmx.org@next/dist/htmx.min.js\"></script>\n <script src=\"https://cdn.jsdelivr.net/gh/answerdotai/surreal@1.3.0/surreal.js\"></script>\n <script src=\"https://cdn.jsdelivr.net/gh/gnat/css-scope-inline@main/script.js\"></script>\n </head>\n <body>\n<div>\n <h1>Hello, World</h1>\n <p>Some text</p>\n <p>Some more text</p>\n</div>\n </body>\n</html>\n\n\n\nWe’ll use this pattern often in the examples to follow.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#defining-routes", + "href": "tutorials/by_example.html#defining-routes", + "title": "FastHTML By Example", + "section": "Defining Routes", + "text": "Defining Routes\nThe HTTP protocol defines a number of methods (‘verbs’) to send requests to a server. The most common are GET, POST, PUT, DELETE, and HEAD. We saw ‘GET’ in action before - when you navigate to a URL, you’re making a GET request to that URL. We can do different things on a route for different HTTP methods. For example:\n@app.route(\"/\", methods='get')\ndef home():\n return H1('Hello, World')\n\n@app.route(\"/\", methods=['post', 'put'])\ndef post_or_put():\n return \"got a POST or PUT request\"\nThis says that when someone navigates to the root URL “/” (i.e. sends a GET request), they will see the big “Hello, World” heading. When someone submits a POST or PUT request to the same URL, the server should return the string “got a post or put request”.\n\n\n\n\n\n\nTest the POST request\n\n\n\nYou can test the POST request with curl -X POST http://127.0.0.1:8000 -d \"some data\". This sends some data to the server, you should see the response “got a post or put request” printed in the terminal.\n\n\nThere are a few other ways you can specify the route+method - FastHTML has .get, .post, etc. as shorthand for route(..., methods=['get']), etc.\n\n@app.get(\"/\")\ndef my_function():\n return \"Hello World from a GET request\"\n\nOr you can use the @rt decorator without a method but specify the method with the name of the function. For example:\n\nrt = app.route\n\n@rt(\"/\")\ndef post():\n return \"Hello World from a POST request\"\n\n\nclient.post(\"/\").text\n\n'Hello World from a POST request'\n\n\nYou’re welcome to pick whichever style you prefer. Using routes lets you show different content on different pages - ‘/home’, ‘/about’ and so on. You can also respond differently to different kinds of requests to the same route, as shown above. You can also pass data via the route:\n\n@app.get@rt\n\n\n\n@app.get(\"/greet/{nm}\")\ndef greet(nm:str):\n return f\"Good day to you, {nm}!\"\n\nclient.get(\"/greet/Dave\").text\n\n'Good day to you, Dave!'\n\n\n\n\n\n@rt(\"/greet/{nm}\")\ndef get(nm:str):\n return f\"Good day to you, {nm}!\"\n\nclient.get(\"/greet/Dave\").text\n\n'Good day to you, Dave!'\n\n\n\n\n\nMore on this in the More on Routing and Request Parameters section, which goes deeper into the different ways to get information from a request.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#styling-basics", + "href": "tutorials/by_example.html#styling-basics", + "title": "FastHTML By Example", + "section": "Styling Basics", + "text": "Styling Basics\nPlain HTML probably isn’t quite what you imagine when you visualize your beautiful web app. CSS is the go-to language for styling HTML. But again, we don’t want to learn extra languages unless we absolutely have to! Fortunately, there are ways to get much more visually appealing sites by relying on the hard work of others, using existing CSS libraries. One of our favourites is PicoCSS. A common way to add CSS files to web pages is to use a <link> tag inside your HTML header, like this:\n<header>\n ...\n <link rel=\"stylesheet\" href=\"https://cdn.jsdelivr.net/npm/@picocss/pico@latest/css/pico.min.css\">\n</header>\nFor convenience, FastHTML already defines a Pico component for you with picolink:\n\nprint(to_xml(picolink))\n\n<link rel=\"stylesheet\" href=\"https://cdn.jsdelivr.net/npm/@picocss/pico@latest/css/pico.min.css\">\n\n<style>:root { --pico-font-size: 100%; }</style>\n\n\n\n\n\n\n\n\n\nNote\n\n\n\npicolink also includes a <style> tag, as we found that setting the font-size to 100% to be a good default. We show you how to override this below.\n\n\nSince we typically want CSS styling on all pages of our app, FastHTML lets you define a shared HTML header with the hdrs argument as shown below:\n\nfrom fasthtml.common import *\n1css = Style(':root {--pico-font-size:90%,--pico-font-family: Pacifico, cursive;}')\n2app = FastHTML(hdrs=(picolink, css))\n\n@app.route(\"/\")\ndef get():\n return (Title(\"Hello World\"), \n3 Main(H1('Hello, World'), cls=\"container\"))\n\n\n1\n\nCustom styling to override the pico defaults\n\n2\n\nDefine shared headers for all pages\n\n3\n\nAs per the pico docs, we put all of our content inside a <main> tag with a class of container:\n\n\n\n\n\n\n\n\n\n\nReturning Tuples\n\n\n\nWe’re returning a tuple here (a title and the main page). Returning a tuple, list, FT object, or an object with a __ft__ method tells FastHTML to turn the main body into a full HTML page that includes the headers (including the pico link and our custom css) which we passed in. This only occurs if the request isn’t from HTMX (for HTMX requests we need only return the rendered components).\n\n\nYou can check out the Pico examples page to see how different elements will look. If everything is working, the page should now render nice text with our custom font, and it should respect the user’s light/dark mode preferences too.\nIf you want to override the default styles or add more custom CSS, you can do so by adding a <style> tag to the headers as shown above. So you are allowed to write CSS to your heart’s content - we just want to make sure you don’t necessarily have to! Later on we’ll see examples using other component libraries and tailwind css to do more fancy styling things, along with tips to get an LLM to write all those fiddly bits so you don’t have to.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#web-page---web-app", + "href": "tutorials/by_example.html#web-page---web-app", + "title": "FastHTML By Example", + "section": "Web Page -> Web App", + "text": "Web Page -> Web App\nShowing content is all well and good, but we typically expect a bit more interactivity from something calling itself a web app! So, let’s add a few different pages, and use a form to let users add messages to a list:\n\napp = FastHTML()\nmessages = [\"This is a message, which will get rendered as a paragraph\"]\n\n@app.get(\"/\")\ndef home():\n return Main(H1('Messages'), \n *[P(msg) for msg in messages],\n A(\"Link to Page 2 (to add messages)\", href=\"/page2\"))\n\n@app.get(\"/page2\")\ndef page2():\n return Main(P(\"Add a message with the form below:\"),\n Form(Input(type=\"text\", name=\"data\"),\n Button(\"Submit\"),\n action=\"/\", method=\"post\"))\n\n@app.post(\"/\")\ndef add_message(data:str):\n messages.append(data)\n return home()\n\nWe re-render the entire homepage to show the newly added message. This is fine, but modern web apps often don’t re-render the entire page, they just update a part of the page. In fact even very complicated applications are often implemented as ‘Single Page Apps’ (SPAs). This is where HTMX comes in.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#htmx", + "href": "tutorials/by_example.html#htmx", + "title": "FastHTML By Example", + "section": "HTMX", + "text": "HTMX\nHTMX addresses some key limitations of HTML. In vanilla HTML, links can trigger a GET request to show a new page, and forms can send requests containing data to the server. A lot of ‘Web 1.0’ design revolved around ways to use these to do everything we wanted. But why should only some elements be allowed to trigger requests? And why should we refresh the entire page with the result each time one does? HTMX extends HTML to allow us to trigger requests from any element on all kinds of events, and to update a part of the page without refreshing the entire page. It’s a powerful tool for building modern web apps.\nIt does this by adding attributes to HTML tags to make them do things. For example, here’s a page with a counter and a button that increments it:\n\napp = FastHTML()\n\ncount = 0\n\n@app.get(\"/\")\ndef home():\n return Title(\"Count Demo\"), Main(\n H1(\"Count Demo\"),\n P(f\"Count is set to {count}\", id=\"count\"),\n Button(\"Increment\", hx_post=\"/increment\", hx_target=\"#count\", hx_swap=\"innerHTML\")\n )\n\n@app.post(\"/increment\")\ndef increment():\n print(\"incrementing\")\n global count\n count += 1\n return f\"Count is set to {count}\"\n\nThe button triggers a POST request to /increment (since we set hx_post=\"/increment\"), which increments the count and returns the new count. The hx_target attribute tells HTMX where to put the result. If no target is specified it replaces the element that triggered the request. The hx_swap attribute specifies how it adds the result to the page. Useful options are:\n\ninnerHTML: Replace the target element’s content with the result.\nouterHTML: Replace the target element with the result.\nbeforebegin: Insert the result before the target element.\nbeforeend: Insert the result inside the target element, after its last child.\nafterbegin: Insert the result inside the target element, before its first child.\nafterend: Insert the result after the target element.\n\nYou can also use an hx_swap of delete to delete the target element regardless of response, or of none to do nothing.\nBy default, requests are triggered by the “natural” event of an element - click in the case of a button (and most other elements). You can also specify different triggers, along with various modifiers - see the HTMX docs for more.\nThis pattern of having elements trigger requests that modify or replace other elements is a key part of the HTMX philosophy. It takes a little getting used to, but once mastered it is extremely powerful.\n\nReplacing Elements Besides the Target\nSometimes having a single target is not enough, and we’d like to specify some additional elements to update or remove. In these cases, returning elements with an id that matches the element to be replaced and hx_swap_oob='true' will replace those elements too. We’ll use this in the next example to clear an input field when we submit a form.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#full-example-1---todo-app", + "href": "tutorials/by_example.html#full-example-1---todo-app", + "title": "FastHTML By Example", + "section": "Full Example #1 - ToDo App", + "text": "Full Example #1 - ToDo App\nThe canonical demo web app! A TODO list. Rather than create yet another variant for this tutorial, we recommend starting with this video tutorial from Jeremy:\n\n\n\n\nimage.png\n\n\nWe’ve made a number of variants of this app - so in addition to the version shown in the video you can browse this series of examples with increasing complexity, the heavily-commented “idiomatic” version here, and the example linked from the FastHTML homepage.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#full-example-2---image-generation-app", + "href": "tutorials/by_example.html#full-example-2---image-generation-app", + "title": "FastHTML By Example", + "section": "Full Example #2 - Image Generation App", + "text": "Full Example #2 - Image Generation App\nLet’s create an image generation app. We’d like to wrap a text-to-image model in a nice UI, where the user can type in a prompt and see a generated image appear. We’ll use a model hosted by Replicate to actually generate the images. Let’s start with the homepage, with a form to submit prompts and a div to hold the generated images:\n# Main page\n@app.get(\"/\")\ndef get():\n inp = Input(id=\"new-prompt\", name=\"prompt\", placeholder=\"Enter a prompt\")\n add = Form(Group(inp, Button(\"Generate\")), hx_post=\"/\", target_id='gen-list', hx_swap=\"afterbegin\")\n gen_list = Div(id='gen-list')\n return Title('Image Generation Demo'), Main(H1('Magic Image Generation'), add, gen_list, cls='container')\nSubmitting the form will trigger a POST request to /, so next we need to generate an image and add it to the list. One problem: generating images is slow! We’ll start the generation in a separate thread, but this now surfaces a different problem: we want to update the UI right away, but our image will only be ready a few seconds later. This is a common pattern - think about how often you see a loading spinner online. We need a way to return a temporary bit of UI which will eventually be replaced by the final image. Here’s how we might do this:\ndef generation_preview(id):\n if os.path.exists(f\"gens/{id}.png\"):\n return Div(Img(src=f\"/gens/{id}.png\"), id=f'gen-{id}')\n else:\n return Div(\"Generating...\", id=f'gen-{id}', \n hx_post=f\"/generations/{id}\",\n hx_trigger='every 1s', hx_swap='outerHTML')\n \n@app.post(\"/generations/{id}\")\ndef get(id:int): return generation_preview(id)\n\n@app.post(\"/\")\ndef post(prompt:str):\n id = len(generations)\n generate_and_save(prompt, id)\n generations.append(prompt)\n clear_input = Input(id=\"new-prompt\", name=\"prompt\", placeholder=\"Enter a prompt\", hx_swap_oob='true')\n return generation_preview(id), clear_input\n\n@threaded\ndef generate_and_save(prompt, id): ... \nThe form sends the prompt to the / route, which starts the generation in a separate thread then returns two things:\n\nA generation preview element that will be added to the top of the gen-list div (since that is the target_id of the form which triggered the request)\nAn input field that will replace the form’s input field (that has the same id), using the hx_swap_oob=‘true’ trick. This clears the prompt field so the user can type another prompt.\n\nThe generation preview first returns a temporary “Generating…” message, which polls the /generations/{id} route every second. This is done by setting hx_post to the route and hx_trigger to ‘every 1s’. The /generations/{id} route returns the preview element every second until the image is ready, at which point it returns the final image. Since the final image replaces the temporary one (hx_swap=‘outerHTML’), the polling stops running and the generation preview is now complete.\nThis works nicely - the user can submit several prompts without having to wait for the first one to generate, and as the images become available they are added to the list. You can see the full code of this version here.\n\nAgain, with Style\nThe app is functional, but can be improved. The next version adds more stylish generation previews, lays out the images in a grid layout that is responsive to different screen sizes, and adds a database to track generations and make them persistent. The database part is very similar to the todo list example, so let’s just quickly look at how we add the nice grid layout. This is what the result looks like:\n\n\n\nimage.png\n\n\nStep one was looking around for existing components. The Pico CSS library we’ve been using has a rudimentary grid but recommends using an alternative layout system. One of the options listed was Flexbox.\nTo use Flexbox you create a “row” with one or more elements. You can specify how wide things should be with a specific syntax in the class name. For example, col-xs-12 means a box that will take up 12 columns (out of 12 total) of the row on extra small screens, col-sm-6 means a column that will take up 6 columns of the row on small screens, and so on. So if you want four columns on large screens you would use col-lg-3 for each item (i.e. each item is using 3 columns out of 12).\n<div class=\"row\">\n <div class=\"col-xs-12\">\n <div class=\"box\">This takes up the full width</div>\n </div>\n</div>\nThis was non-intuitive to me. Thankfully ChatGPT et al know web stuff quite well, and we can also experiment in a notebook to test things out:\n\ngrid = Html(\n Link(rel=\"stylesheet\", href=\"https://cdnjs.cloudflare.com/ajax/libs/flexboxgrid/6.3.1/flexboxgrid.min.css\", type=\"text/css\"),\n Div(\n Div(Div(\"This takes up the full width\", cls=\"box\", style=\"background-color: #800000;\"), cls=\"col-xs-12\"),\n Div(Div(\"This takes up half\", cls=\"box\", style=\"background-color: #008000;\"), cls=\"col-xs-6\"),\n Div(Div(\"This takes up half\", cls=\"box\", style=\"background-color: #0000B0;\"), cls=\"col-xs-6\"),\n cls=\"row\", style=\"color: #fff;\"\n )\n)\nshow(grid)\n\n\n\n\n \n \n \n This takes up the full width\n \n \n This takes up half\n \n \n This takes up half\n \n \n\n\n\nAside: when in doubt with CSS stuff, add a background color or a border so you can see what’s happening!\nTranslating this into our app, we have a new homepage with a div (class=\"row\") to store the generated images / previews, and a generation_preview function that returns boxes with the appropriate classes and styles to make them appear in the grid. I chose a layout with different numbers of columns for different screen sizes, but you could also just specify the col-xs class if you wanted the same layout on all devices.\ngridlink = Link(rel=\"stylesheet\", href=\"https://cdnjs.cloudflare.com/ajax/libs/flexboxgrid/6.3.1/flexboxgrid.min.css\", type=\"text/css\")\napp = FastHTML(hdrs=(picolink, gridlink))\n\n# Main page\n@app.get(\"/\")\ndef get():\n inp = Input(id=\"new-prompt\", name=\"prompt\", placeholder=\"Enter a prompt\")\n add = Form(Group(inp, Button(\"Generate\")), hx_post=\"/\", target_id='gen-list', hx_swap=\"afterbegin\")\n gen_containers = [generation_preview(g) for g in gens(limit=10)] # Start with last 10\n gen_list = Div(*gen_containers[::-1], id='gen-list', cls=\"row\") # flexbox container: class = row\n return Title('Image Generation Demo'), Main(H1('Magic Image Generation'), add, gen_list, cls='container')\n\n# Show the image (if available) and prompt for a generation\ndef generation_preview(g):\n grid_cls = \"box col-xs-12 col-sm-6 col-md-4 col-lg-3\"\n image_path = f\"{g.folder}/{g.id}.png\"\n if os.path.exists(image_path):\n return Div(Card(\n Img(src=image_path, alt=\"Card image\", cls=\"card-img-top\"),\n Div(P(B(\"Prompt: \"), g.prompt, cls=\"card-text\"),cls=\"card-body\"),\n ), id=f'gen-{g.id}', cls=grid_cls)\n return Div(f\"Generating gen {g.id} with prompt {g.prompt}\", \n id=f'gen-{g.id}', hx_get=f\"/gens/{g.id}\", \n hx_trigger=\"every 2s\", hx_swap=\"outerHTML\", cls=grid_cls)\nYou can see the final result in main.py in the image_app_simple example directory, along with info on deploying it (tl;dr don’t!). We’ve also deployed a version that only shows your generations (tied to browser session) and has a credit system to save our bank accounts. You can access that here. Now for the next question: how do we keep track of different users?\n\n\nAgain, with Sessions\nAt the moment everyone sees all images! How do we keep some sort of unique identifier tied to a user? Before going all the way to setting up users, login pages etc., let’s look at a way to at least limit generations to the user’s session. You could do this manually with cookies. For convenience and security, fasthtml (via Starlette) has a special mechanism for storing small amounts of data in the user’s browser via the session argument to your route. This acts like a dictionary and you can set and get values from it. For example, here we look for a session_id key, and if it doesn’t exist we generate a new one:\n@app.get(\"/\")\ndef get(session):\n if 'session_id' not in session: session['session_id'] = str(uuid.uuid4())\n return H1(f\"Session ID: {session['session_id']}\")\nRefresh the page a few times - you’ll notice that the session ID remains the same. If you clear your browsing data, you’ll get a new session ID. And if you load the page in a different browser (but not a different tab), you’ll get a new session ID. This will persist within the current browser, letting us use it as a key for our generations. As a bonus, someone can’t spoof this session id by passing it in another way (for example, sending a query parameter). Behind the scenes, the data is stored in a browser cookie but it is signed with a secret key that stops the user or anyone nefarious from being able to tamper with it. The cookie is decoded back into a dictionary by something called a middleware function, which we won’t cover here. All you need to know is that we can use this to store bits of state in the user’s browser.\nIn the image app example, we can add a session_id column to our database, and modify our homepage like so:\n@app.get(\"/\")\ndef get(session):\n if 'session_id' not in session: session['session_id'] = str(uuid.uuid4())\n inp = Input(id=\"new-prompt\", name=\"prompt\", placeholder=\"Enter a prompt\")\n add = Form(Group(inp, Button(\"Generate\")), hx_post=\"/\", target_id='gen-list', hx_swap=\"afterbegin\")\n gen_containers = [generation_preview(g) for g in gens(limit=10, where=f\"session_id == '{session['session_id']}'\")]\n ...\nSo we check if the session id exists in the session, add one if not, and then limit the generations shown to only those tied to this session id. We filter the database with a where clause - see [TODO link Jeremy’s example for a more reliable way to do this]. The only other change we need to make is to store the session id in the database when a generation is made. You can check out this version here. You could instead write this app without relying on a database at all - simply storing the filenames of the generated images in the session, for example. But this more general approach of linking some kind of unique session identifier to users or data in our tables is a useful general pattern for more complex examples.\n\n\nAgain, with Credits!\nGenerating images with replicate costs money. So next let’s add a pool of credits that get used up whenever anyone generates an image. To recover our lost funds, we’ll also set up a payment system so that generous users can buy more credits for everyone. You could modify this to let users buy credits tied to their session ID, but at that point you risk having angry customers losing their money after wiping their browser history, and should consider setting up proper account management :)\nTaking payments with Stripe is intimidating but very doable. Here’s a tutorial that shows the general principle using Flask. As with other popular tasks in the web-dev world, ChatGPT knows a lot about Stripe - but you should exercise extra caution when writing code that handles money!\nFor the finished example we add the bare minimum:\n\nA way to create a Stripe checkout session and redirect the user to the session URL\n‘Success’ and ‘Cancel’ routes to handle the result of the checkout\nA route that listens for a webhook from Stripe to update the number of credits when a payment is made.\n\nIn a typical application you’ll want to keep track of which users make payments, catch other kinds of stripe events and so on. This example is more a ‘this is possible, do your own research’ than ‘this is how you do it’. But hopefully it does illustrate the key idea: there is no magic here. Stripe (and many other technologies) relies on sending users to different routes and shuttling data back and forth in requests. And we know how to do that!", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#more-on-routing-and-request-parameters", + "href": "tutorials/by_example.html#more-on-routing-and-request-parameters", + "title": "FastHTML By Example", + "section": "More on Routing and Request Parameters", + "text": "More on Routing and Request Parameters\nThere are a number of ways information can be passed to the server. When you specify arguments to a route, FastHTML will search the request for values with the same name, and convert them to the correct type. In order, it searches\n\nThe path parameters\nThe query parameters\nThe cookies\nThe headers\nThe session\nForm data\n\nThere are also a few special arguments\n\nrequest (or any prefix like req): gets the raw Starlette Request object\nsession (or any prefix like sess): gets the session object\nauth\nhtmx\napp\n\nIn this section let’s quickly look at some of these in action.\n\nfrom fasthtml.common import *\nfrom starlette.testclient import TestClient\n\napp = FastHTML()\ncli = TestClient(app)\n\nPart of the route (path parameters):\n\n@app.get('/user/{nm}')\ndef _(nm:str): return f\"Good day to you, {nm}!\"\n\ncli.get('/user/jph').text\n\n'Good day to you, jph!'\n\n\nMatching with a regex:\n\nreg_re_param(\"imgext\", \"ico|gif|jpg|jpeg|webm\")\n\n@app.get(r'/static/{path:path}/{fn}.{ext:imgext}')\ndef get_img(fn:str, path:str, ext:str): return f\"Getting {fn}.{ext} from /{path}\"\n\ncli.get('/static/foo/jph.ico').text\n\n'Getting jph.ico from /foo/'\n\n\nUsing an enum (try using a string that isn’t in the enum):\n\nModelName = str_enum('ModelName', \"alexnet\", \"resnet\", \"lenet\")\n\n@app.get(\"/models/{nm}\")\ndef model(nm:ModelName): return nm\n\nprint(cli.get('/models/alexnet').text)\n\nalexnet\n\n\nCasting to a Path:\n\n@app.get(\"/files/{path}\")\ndef txt(path: Path): return path.with_suffix('.txt')\n\nprint(cli.get('/files/foo').text)\n\nfoo.txt\n\n\nAn integer with a default value:\n\nfake_db = [{\"name\": \"Foo\"}, {\"name\": \"Bar\"}]\n\n@app.get(\"/items/\")\ndef read_item(idx: int = 0): return fake_db[idx]\n\nprint(cli.get('/items/?idx=1').text)\n\n{\"name\":\"Bar\"}\n\n\n\n# Equivalent to `/items/?idx=0`.\nprint(cli.get('/items/').text)\n\n{\"name\":\"Foo\"}\n\n\nBoolean values (takes anything “truthy” or “falsy”):\n\n@app.get(\"/booly/\")\ndef booly(coming:bool=True): return 'Coming' if coming else 'Not coming'\n\nprint(cli.get('/booly/?coming=true').text)\n\nComing\n\n\n\nprint(cli.get('/booly/?coming=no').text)\n\nNot coming\n\n\nGetting dates:\n\n@app.get(\"/datie/\")\ndef datie(d:parsed_date): return d\n\ndate_str = \"17th of May, 2024, 2p\"\nprint(cli.get(f'/datie/?d={date_str}').text)\n\n2024-05-17 14:00:00\n\n\nMatching a dataclass:\n\nfrom dataclasses import dataclass, asdict\n\n@dataclass\nclass Bodie:\n a:int;b:str\n\n@app.route(\"/bodie/{nm}\")\ndef post(nm:str, data:Bodie):\n res = asdict(data)\n res['nm'] = nm\n return res\n\ncli.post('/bodie/me', data=dict(a=1, b='foo')).text\n\n'{\"a\":1,\"b\":\"foo\",\"nm\":\"me\"}'\n\n\n\nCookies\nCookies can be set via a Starlette Response object, and can be read back by specifying the name:\n\nfrom datetime import datetime\n\n@app.get(\"/setcookie\")\ndef setc(req):\n now = datetime.now()\n res = Response(f'Set to {now}')\n res.set_cookie('now', str(now))\n return res\n\ncli.get('/setcookie').text\n\n'Set to 2024-07-20 23:14:54.364793'\n\n\n\n@app.get(\"/getcookie\")\ndef getc(now:parsed_date): return f'Cookie was set at time {now.time()}'\n\ncli.get('/getcookie').text\n\n'Cookie was set at time 23:14:54.364793'\n\n\n\n\nUser Agent and HX-Request\nAn argument of user_agent will match the header User-Agent. This holds for special headers like HX-Request (used by HTMX to signal when a request comes from an HTMX request) - the general pattern is that “-” is replaced with “_” and strings are turned to lowercase.\n\n@app.get(\"/ua\")\nasync def ua(user_agent:str): return user_agent\n\ncli.get('/ua', headers={'User-Agent':'FastHTML'}).text\n\n'FastHTML'\n\n\n\n@app.get(\"/hxtest\")\ndef hxtest(htmx): return htmx.request\n\ncli.get('/hxtest', headers={'HX-Request':'1'}).text\n\n'1'\n\n\n\n\nStarlette Requests\nIf you add an argument called request(or any prefix of that, for example req) it will be populated with the Starlette Request object. This is useful if you want to do your own processing manually. For example, although FastHTML will parse forms for you, you could instead get form data like so:\n@app.get(\"/form\")\nasync def form(request:Request):\n form_data = await request.form()\n a = form_data.get('a')\nSee the Starlette docs for more information on the Request object.\n\n\nStarlette Responses\nYou can return a Starlette Response object from a route to control the response. For example:\n@app.get(\"/redirect\")\ndef redirect():\n return RedirectResponse(url=\"/\")\nWe used this to set cookies in the previous example. See the Starlette docs for more information on the Response object.\n\n\nStatic Files\nWe often want to serve static files like images. This is easily done! For common file types (images, CSS etc) we can create a route that returns a Starlette FileResponse like so:\n# For images, CSS, etc.\n@app.get(\"/{fname:path}.{ext:static}\")\ndef static(fname: str, ext: str):\n return FileResponse(f'{fname}.{ext}')\nYou can customize it to suit your needs (for example, only serving files in a certain directory). You’ll notice some variant of this route in all our complete examples - even for apps with no static files the browser will typically request a /favicon.ico file, for example, and as the astute among you will have noticed this has sparked a bit of competition between Johno and Jeremy regarding which country flag should serve as the default!\n\n\nWebSockets\nFor certain applications such as multiplayer games, websockets can be a powerful feature. Luckily HTMX and FastHTML has you covered! Simply specify that you wish to include the websocket header extension from HTMX:\napp = FastHTML(exts='ws')\nrt = app.route\nWith that, you are now able to specify the different websocket specific HTMX goodies. For example, say we have a website we want to setup a websocket, you can simply:\ndef mk_inp(): return Input(id='msg')\n\n@rt('/')\nasync def get(request):\n cts = Div(\n Div(id='notifications'),\n Form(mk_inp(), id='form', ws_send=True),\n hx_ext='ws', ws_connect='/ws')\n return Titled('Websocket Test', cts)\nAnd this will setup a connection on the route /ws along with a form that will send a message to the websocket whenever the form is submitted. Let’s go ahead and handle this route:\n@app.ws('/ws')\nasync def ws(msg:str, send):\n await send(Div('Hello ' + msg, id=\"notifications\"))\n await sleep(2)\n return Div('Goodbye ' + msg, id=\"notifications\"), mk_inp()\nOne thing you might have noticed is a lack of target id for our websocket trigger for swapping HTML content. This is because HTMX always swaps content with websockets with Out of Band Swaps. Therefore, HTMX will look for the id in the returned HTML content from the server for determining what to swap. To send stuff to the client, you can either use the send parameter or simply return the content or both!\nNow, sometimes you might want to perform actions when a client connects or disconnects such as add or remove a user from a player queue. To hook into these events, you can pass your connection or disconnection function to the app.ws decorator:\nasync def on_connect(send):\n print('Connected!')\n await send(Div('Hello, you have connected', id=\"notifications\"))\n\nasync def on_disconnect(ws):\n print('Disconnected!')\n\n@app.ws('/ws', conn=on_connect, disconn=on_disconnect)\nasync def ws(msg:str, send):\n await send(Div('Hello ' + msg, id=\"notifications\"))\n await sleep(2)\n return Div('Goodbye ' + msg, id=\"notifications\"), mk_inp()", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#full-example-3---chatbot-example-with-daisyui-components", + "href": "tutorials/by_example.html#full-example-3---chatbot-example-with-daisyui-components", + "title": "FastHTML By Example", + "section": "Full Example #3 - Chatbot Example with DaisyUI Components", + "text": "Full Example #3 - Chatbot Example with DaisyUI Components\nLet’s go back to the topic of adding components or styling beyond the simple PicoCSS examples so far. How might we adopt a component or framework? In this example, let’s build a chatbot UI leveraging the DaisyUI chat bubble. The final result will look like this:\n\n\n\nimage.png\n\n\nAt first glance, DaisyUI’s chat component looks quite intimidating. The examples look like this:\n<div class=\"chat chat-start\">\n <div class=\"chat-image avatar\">\n <div class=\"w-10 rounded-full\">\n <img alt=\"Tailwind CSS chat bubble component\" src=\"https://img.daisyui.com/images/stock/photo-1534528741775-53994a69daeb.jpg\" />\n </div>\n </div>\n <div class=\"chat-header\">\n Obi-Wan Kenobi\n <time class=\"text-xs opacity-50\">12:45</time>\n </div>\n <div class=\"chat-bubble\">You were the Chosen One!</div>\n <div class=\"chat-footer opacity-50\">\n Delivered\n </div>\n</div>\n<div class=\"chat chat-end\">\n <div class=\"chat-image avatar\">\n <div class=\"w-10 rounded-full\">\n <img alt=\"Tailwind CSS chat bubble component\" src=\"https://img.daisyui.com/images/stock/photo-1534528741775-53994a69daeb.jpg\" />\n </div>\n </div>\n <div class=\"chat-header\">\n Anakin\n <time class=\"text-xs opacity-50\">12:46</time>\n </div>\n <div class=\"chat-bubble\">I hate you!</div>\n <div class=\"chat-footer opacity-50\">\n Seen at 12:46\n </div>\n</div>\nWe have several things going for us however.\n\nChatGPT knows DaisyUI and Tailwind (DaisyUI is a Tailwind component library)\nWe can build things up piece by piece with AI standing by to help.\n\nhttps://h2f.answer.ai/ is a tool that can convert HTML to FT (fastcore.xml) and back, which is useful for getting a quick starting point when you have an HTML example to start from.\nWe can strip out some unnecessary bits and try to get the simplest possible example working in a notebook first:\n\n# Loading tailwind and daisyui\nheaders = (Script(src=\"https://cdn.tailwindcss.com\"),\n Link(rel=\"stylesheet\", href=\"https://cdn.jsdelivr.net/npm/daisyui@4.11.1/dist/full.min.css\"))\n\n# Displaying a single message\nd = Div(\n Div(\"Chat header here\", cls=\"chat-header\"),\n Div(\"My message goes here\", cls=\"chat-bubble chat-bubble-primary\"),\n cls=\"chat chat-start\"\n)\n# show(Html(*headers, d)) # uncomment to view\n\nNow we can extend this to render multiple messages, with the message being on the left (chat-start) or right (chat-end) depending on the role. While we’re at it, we can also change the color (chat-bubble-primary) of the message and put them all in a chat-box div:\n\nmessages = [\n {\"role\":\"user\", \"content\":\"Hello\"},\n {\"role\":\"assistant\", \"content\":\"Hi, how can I assist you?\"}\n]\n\ndef ChatMessage(msg):\n return Div(\n Div(msg['role'], cls=\"chat-header\"),\n Div(msg['content'], cls=f\"chat-bubble chat-bubble-{'primary' if msg['role'] == 'user' else 'secondary'}\"),\n cls=f\"chat chat-{'end' if msg['role'] == 'user' else 'start'}\")\n\nchatbox = Div(*[ChatMessage(msg) for msg in messages], cls=\"chat-box\", id=\"chatlist\")\n\n# show(Html(*headers, chatbox)) # Uncomment to view\n\nNext, it was back to the ChatGPT to tweak the chat box so it wouldn’t grow as messages were added. I asked:\n\"I have something like this (it's working now) \n[code]\nThe messages are added to this div so it grows over time. \nIs there a way I can set it's height to always be 80% of the total window height with a scroll bar if needed?\"\nBased on this query GPT4o helpfully shared that “This can be achieved using Tailwind CSS utility classes. Specifically, you can use h-[80vh] to set the height to 80% of the viewport height, and overflow-y-auto to add a vertical scroll bar when needed.”\nTo put it another way: none of the CSS classes in the following example were written by a human, and what edits I did make were informed by advice from the AI that made it relatively painless!\nThe actual chat functionality of the app is based on our claudette library. As with the image example, we face a potential hiccup in that getting a response from an LLM is slow. We need a way to have the user message added to the UI immediately, and then have the response added once it’s available. We could do something similar to the image generation example above, or use websockets. Check out the full example for implementations of both, along with further details.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#full-example-4---multiplayer-game-of-life-example-with-websockets", + "href": "tutorials/by_example.html#full-example-4---multiplayer-game-of-life-example-with-websockets", + "title": "FastHTML By Example", + "section": "Full Example #4 - Multiplayer Game of Life Example with Websockets", + "text": "Full Example #4 - Multiplayer Game of Life Example with Websockets\nLet’s see how we can implement a collaborative website using Websockets in FastHTML. To showcase this, we will use the famous Conway’s Game of Life, which is a game that takes place in a grid world. Each cell in the grid can be either alive or dead. The cell’s state is initially given by a user before the game is started and then evolves through the iteration of the grid world once the clock starts. Whether a cell’s state will change from the previous state depends on simple rules based on its neighboring cells’ states. Here is the standard Game of Life logic implemented in Python courtesy of ChatGPT:\ngrid = [[0 for _ in range(20)] for _ in range(20)]\ndef update_grid(grid: list[list[int]]) -> list[list[int]]:\n new_grid = [[0 for _ in range(20)] for _ in range(20)]\n def count_neighbors(x, y):\n directions = [(-1, -1), (-1, 0), (-1, 1), (0, -1), (0, 1), (1, -1), (1, 0), (1, 1)]\n count = 0\n for dx, dy in directions:\n nx, ny = x + dx, y + dy\n if 0 <= nx < len(grid) and 0 <= ny < len(grid[0]): count += grid[nx][ny]\n return count\n for i in range(len(grid)):\n for j in range(len(grid[0])):\n neighbors = count_neighbors(i, j)\n if grid[i][j] == 1:\n if neighbors < 2 or neighbors > 3: new_grid[i][j] = 0\n else: new_grid[i][j] = 1\n elif neighbors == 3: new_grid[i][j] = 1\n return new_grid\nThis would be a very dull game if we were to run it, since the initial state of everything would remain dead. Therefore, we need a way of letting the user give an initial state before starting the game. FastHTML to the rescue!\ndef Grid():\n cells = []\n for y, row in enumerate(game_state['grid']):\n for x, cell in enumerate(row):\n cell_class = 'alive' if cell else 'dead'\n cell = Div(cls=f'cell {cell_class}', hx_put='/update', hx_vals={'x': x, 'y': y}, hx_swap='none', hx_target='#gol', hx_trigger='click')\n cells.append(cell)\n return Div(*cells, id='grid')\n\n@rt('/update')\nasync def put(x: int, y: int):\n grid[y][x] = 1 if grid[y][x] == 0 else 0\nAbove is a component for representing the game’s state that the user can interact with and update on the server using cool HTMX features such as hx_vals for determining which cell was clicked to make it dead or alive. Now, you probably noticed that the HTTP request in this case is a PUT request, which does not return anything and this means our client’s view of the grid world and the server’s game state will immediately become out of sync :(. We could of course just return a new Grid component with the updated state, but that would only work for a single client, if we had more, they quickly get out of sync with each other and the server. Now Websockets to the rescue!\nWebsockets are a way for the server to keep a persistent connection with clients and send data to the client without explicitly being requested for information, which is not possible with HTTP. Luckily FastHTML and HTMX work well with Websockets. Simply state you wish to use websockets for your app and define a websocket route:\n...\napp = FastHTML(hdrs=(picolink, gridlink, css, htmx_ws), exts='ws')\n\nplayer_queue = []\nasync def update_players():\n for i, player in enumerate(player_queue):\n try: await player(Grid())\n except: player_queue.pop(i)\nasync def on_connect(send): player_queue.append(send)\nasync def on_disconnect(send): await update_players()\n\n@app.ws('/gol', conn=on_connect, disconn=on_disconnect)\nasync def ws(msg:str, send): pass\n\ndef Home(): return Title('Game of Life'), Main(gol, Div(Grid(), id='gol', cls='row center-xs'), hx_ext=\"ws\", ws_connect=\"/gol\")\n\n@rt('/update')\nasync def put(x: int, y: int):\n grid[y][x] = 1 if grid[y][x] == 0 else 0\n await update_players()\n...\nHere we simply keep track of all the players that have connected or disconnected to our site and when an update occurs, we send updates to all the players still connected via websockets. Via HTMX, you are still simply exchanging HTML from the server to the client and will swap in the content based on how you setup your hx_swap attribute. There is only one difference, that being all swaps are OOB. You can find more information on the HTMX websocket extension documentation page here. You can find a full fledge hosted example of this app here.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#ft-objects-and-html", + "href": "tutorials/by_example.html#ft-objects-and-html", + "title": "FastHTML By Example", + "section": "FT objects and HTML", + "text": "FT objects and HTML\nThese FT objects create a ‘FastTag’ structure [tag,children,attrs] for to_xml(). When we call Div(...), the elements we pass in are the children. Attributes are passed in as keywords. class and for are special words in python, so we use cls, klass or _class instead of class and fr or _for instead of for. Note these objects are just 3-element lists - you can create custom ones too as long as they’re also 3-element lists. Alternately, leaf nodes can be strings instead (which is why you can do Div('some text')). If you pass something that isn’t a 3-element list or a string, it will be converted to a string using str()… unless (our final trick) you define a __ft__ method that will run before str(), so you can render things a custom way.\nFor example, here’s one way we could make a custom class that can be rendered into HTML:\n\nclass Person:\n def __init__(self, name, age):\n self.name = name\n self.age = age\n\n def __ft__(self):\n return ['div', [f'{self.name} is {self.age} years old.'], {}]\n\np = Person('Jonathan', 28)\nprint(to_xml(Div(p, \"more text\", cls=\"container\")))\n\n<div class=\"container\">\n <div>Jonathan is 28 years old.</div>\nmore text\n</div>\n\n\n\nIn the examples, you’ll see we often patch in __ft__ methods to existing classes to control how they’re rendered. For example, if Person didn’t have a __ft__ method or we wanted to override it, we could add a new one like this:\n\nfrom fastcore.all import patch\n\n@patch\ndef __ft__(self:Person):\n return Div(\"Person info:\", Ul(Li(\"Name:\",self.name), Li(\"Age:\", self.age)))\n\nshow(p)\n\n\nPerson info:\n \n \nName:\nJonathan\n \n \nAge:\n28\n \n \n\n\n\nSome tags from fastcore.xml are overwritten by fasthtml.core and a few are further extended by fasthtml.xtend using this method. Over time, we hope to see others developing custom components too, giving us a larger and larger ecosystem of reusable components.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#custom-scripts-and-styling", + "href": "tutorials/by_example.html#custom-scripts-and-styling", + "title": "FastHTML By Example", + "section": "Custom Scripts and Styling", + "text": "Custom Scripts and Styling\nThere are many popular JavaScript and CSS libraries that can be used via a simple Script or Style tag. But in some cases you will need to write more custom code. FastHTML’s js.py contains a few examples that may be useful as reference.\nFor example, to use the marked.js library to render markdown in a div, including in components added after the page has loaded via htmx, we do something like this:\nimport { marked } from \"https://cdn.jsdelivr.net/npm/marked/lib/marked.esm.js\";\nproc_htmx('%s', e => e.innerHTML = marked.parse(e.textContent));\nproc_htmx is a shortcut that we wrote to apply a function to elements matching a selector, including the element that triggered the event. Here’s the code for reference:\nexport function proc_htmx(sel, func) {\n htmx.onLoad(elt => {\n const elements = htmx.findAll(elt, sel);\n if (elt.matches(sel)) elements.unshift(elt)\n elements.forEach(func);\n });\n}\nThe AI Pictionary example uses a larger chunk of custom JavaScript to handle the drawing canvas. It’s a good example of the type of application where running code on the client side makes the most sense, but still shows how you can integrate it with FastHTML on the server side to add functionality (like the AI responses) easily.\nAdding styling with custom CSS and libraries such as tailwind is done the same way we add custom JavaScript. The doodle example uses Doodle.CSS to style the page in a quirky way.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#deploying-your-app", + "href": "tutorials/by_example.html#deploying-your-app", + "title": "FastHTML By Example", + "section": "Deploying Your App", + "text": "Deploying Your App\nWe can deploy FastHTML almost anywhere you can deploy python apps. We’ve tested Railway, Replit, HuggingFace, and PythonAnywhere.\n\nRailway\n\nInstall the Railway CLI and sign up for an account.\nSet up a folder with our app as main.py\nIn the folder, run railway login.\nUse the fh_railway_deploy script to deploy our project:\n\nfh_railway_deploy MY_APP_NAME\nWhat the script does for us:\n\nDo we have an existing railway project?\n\nYes: Link the project folder to our existing Railway project.\nNo: Create a new Railway project.\n\nDeploy the project. We’ll see the logs as the service is built and run!\nFetches and displays the URL of our app.\nBy default, mounts a /app/data folder on the cloud to our app’s root folder. The app is run in /app by default, so from our app anything we store in /data will persist across restarts.\n\nA final note about Railway: We can add secrets like API keys that can be accessed as environment variables from our apps via ‘Variables’. For example, for the image generation app, we can add a REPLICATE_API_KEY variable, and then in main.py we can access it as os.environ['REPLICATE_API_KEY'].\n\n\nReplit\nFork this repl for a minimal example you can edit to your heart’s content. .replit has been edited to add the right run command (run = [\"uvicorn\", \"main:app\", \"--reload\"]) and to set up the ports correctly. FastHTML was installed with poetry add python-fasthtml, you can add additional packages as needed in the same way. Running the app in Replit will show you a webview, but you may need to open in a new tab for all features (such as cookies) to work. When you’re ready, you can deploy your app by clicking the ‘Deploy’ button. You pay for usage - for an app that is mostly idle the cost is usually a few cents per month.\nYou can store secrets like API keys via the ‘Secrets’ tab in the Replit project settings.\n\n\nHuggingFace\nFollow the instructions in this repository to deploy to HuggingFace spaces.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/by_example.html#where-next", + "href": "tutorials/by_example.html#where-next", + "title": "FastHTML By Example", + "section": "Where Next?", + "text": "Where Next?\nWe’ve covered a lot of ground here! Hopefully this has given you plenty to work with in building your own FastHTML apps. If you have any questions, feel free to ask in the #fasthtml Discord channel (in the fastai Discord community). You can look through the other examples in the fasthtml-example repository for more ideas, and keep an eye on Jeremy’s YouTube channel where we’ll be releasing a number of “dev chats” related to FastHTML in the near future.", + "crumbs": [ + "Home", + "Tutorials", + "FastHTML By Example" + ] + }, + { + "objectID": "tutorials/jupyter_and_fasthtml.html", + "href": "tutorials/jupyter_and_fasthtml.html", + "title": "Using Jupyter to write FastHTML", + "section": "", + "text": "Writing FastHTML applications in Jupyter notebooks requires a slightly different process than normal Python applications.\nThe first step is to import necessary libraries. As using FastHTML inside a Jupyter notebook is a special case, it remains a special import.\nfrom fasthtml.common import *\nfrom fasthtml.jupyter import JupyUvi, HTMX\nLet’s create an app with fast_app.\napp, rt = fast_app(pico=True)\nDefine a route to test the application.\n@rt\ndef index():\n return Titled('Hello, Jupyter',\n P('Welcome to the FastHTML + Jupyter example'),\n Button('Click', hx_get='/click', hx_target='#dest'),\n Div(id='dest')\n )\nCreate a server object using JupyUvi, which also starts Uvicorn. The server runs in a separate thread from Jupyter, so it can use normal HTTP client functions in a notebook.\nserver = JupyUvi(app)\nThe HTMX callable displays the server’s HTMX application in an iframe which can be displayed by Jupyter notebook. Pass in the same port variable used in the JupyUvi callable above or leave it blank to use the default (8000).\n# This doesn't display in the docs - uncomment and run it to see it in action\n# HTMX()\nWe didn’t define the /click route, but that’s fine - we can define (or change) it any time, and it’s dynamically inserted into the running app. No need to restart or reload anything!\n@rt\ndef click(): return P('You clicked me!')", + "crumbs": [ + "Home", + "Tutorials", + "Using Jupyter to write FastHTML" + ] + }, + { + "objectID": "tutorials/jupyter_and_fasthtml.html#full-screen-view", + "href": "tutorials/jupyter_and_fasthtml.html#full-screen-view", + "title": "Using Jupyter to write FastHTML", + "section": "Full screen view", + "text": "Full screen view\nYou can view your app outside of Jupyter by going to localhost:PORT, where PORT is usually the default 8000, so in most cases just click this link.", + "crumbs": [ + "Home", + "Tutorials", + "Using Jupyter to write FastHTML" + ] + }, + { + "objectID": "tutorials/jupyter_and_fasthtml.html#graceful-shutdowns", + "href": "tutorials/jupyter_and_fasthtml.html#graceful-shutdowns", + "title": "Using Jupyter to write FastHTML", + "section": "Graceful shutdowns", + "text": "Graceful shutdowns\nUse the server.stop() function displayed below. If you restart Jupyter without calling this line the thread may not be released and the HTMX callable above may throw errors. If that happens, a quick temporary fix is to specify a different port number in JupyUvi and HTMX with the port parameter.\nCleaner solutions to the dangling thread are to kill the dangling thread (dependant on each operating system) or restart the computer.\n\nserver.stop()", + "crumbs": [ + "Home", + "Tutorials", + "Using Jupyter to write FastHTML" + ] + }, + { + "objectID": "index.html", + "href": "index.html", + "title": "FastHTML", + "section": "", + "text": "Welcome to the official FastHTML documentation.\nFastHTML is a new next-generation web framework for fast, scalable web applications with minimal, compact code. It’s designed to be:\nFastHTML apps are just Python code, so you can use FastHTML with the full power of the Python language and ecosystem. FastHTML’s functionality maps 1:1 directly to HTML and HTTP, but allows them to be encapsulated using good software engineering practices—so you’ll need to understand these foundations to use this library fully. To understand how and why this works, please read this first: about.fastht.ml.", + "crumbs": [ + "Home", + "Get Started" + ] + }, + { + "objectID": "index.html#installation", + "href": "index.html#installation", + "title": "FastHTML", + "section": "Installation", + "text": "Installation\nSince fasthtml is a Python library, you can install it with:\npip install python-fasthtml\nIn the near future, we hope to add component libraries that can likewise be installed via pip.", + "crumbs": [ + "Home", + "Get Started" + ] + }, + { + "objectID": "index.html#usage", + "href": "index.html#usage", + "title": "FastHTML", + "section": "Usage", + "text": "Usage\nFor a minimal app, create a file “main.py” as follows:\n\n\nmain.py\n\nfrom fasthtml.common import *\n\napp,rt = fast_app()\n\n@rt('/')\ndef get(): return Div(P('Hello World!'), hx_get=\"/change\")\n\nserve()\n\nRunning the app with python main.py prints out a link to your running app: http://localhost:5001. Visit that link in your browser and you should see a page with the text “Hello World!”. Congratulations, you’ve just created your first FastHTML app!\nAdding interactivity is surprisingly easy, thanks to HTMX. Modify the file to add this function:\n\n\nmain.py\n\n@rt('/change')\ndef get(): return P('Nice to be here!')\n\nYou now have a page with a clickable element that changes the text when clicked. When clicking on this link, the server will respond with an “HTML partial”—that is, just a snippet of HTML which will be inserted into the existing page. In this case, the returned element will replace the original P element (since that’s the default behavior of HTMX) with the new version returned by the second route.\nThis “hypermedia-based” approach to web development is a powerful way to build web applications.\n\nGetting help from AI\nBecause FastHTML is newer than most LLMs, AI systems like Cursor, ChatGPT, Claude, and Copilot won’t give useful answers about it. To fix that problem, we’ve provided an LLM-friendly guide that teaches them how to use FastHTML. To use it, add this link for your AI helper to use:\n\n/llms-ctx.txt\n\nThis example is in a format based on recommendations from Anthropic for use with Claude Projects. This works so well that we’ve actually found that Claude can provide even better information than our own documentation! For instance, read through this annotated Claude chat for some great getting-started information, entirely generated from a project using the above text file as context.\nIf you use Cursor, type @doc then choose “Add new doc”, and use the /llms-ctx.txt link above. The context file is auto-generated from our llms.txt (our proposed standard for providing AI-friendly information)—you can generate alternative versions suitable for other models as needed.", + "crumbs": [ + "Home", + "Get Started" + ] + }, + { + "objectID": "index.html#next-steps", + "href": "index.html#next-steps", + "title": "FastHTML", + "section": "Next Steps", + "text": "Next Steps\nStart with the official sources to learn more about FastHTML:\n\nAbout: Learn about the core ideas behind FastHTML\nDocumentation: Learn from examples how to write FastHTML code\nIdiomatic app: Heavily commented source code walking through a complete application, including custom authentication, JS library connections, and database use.\n\nWe also have a 1-hour intro video:\n\nThe capabilities of FastHTML are vast and growing, and not all the features and patterns have been documented yet. Be prepared to invest time into studying and modifying source code, such as the main FastHTML repo’s notebooks and the official FastHTML examples repo:\n\nFastHTML Examples Repo on GitHub\nFastHTML Repo on GitHub\n\nThen explore the small but growing third-party ecosystem of FastHTML tutorials, notebooks, libraries, and components:\n\nFastHTML Gallery: Learn from minimal examples of components (ie chat bubbles, click-to-edit, infinite scroll, etc)\nCreating Custom FastHTML Tags for Markdown Rendering by Isaac Flath\nHow to Build a Simple Login System in FastHTML by Marius Vach\nYour tutorial here!\n\nFinally, join the FastHTML community to ask questions, share your work, and learn from others:\n\nDiscord", + "crumbs": [ + "Home", + "Get Started" + ] + }, + { + "objectID": "index.html#other-languages-and-related-projects", + "href": "index.html#other-languages-and-related-projects", + "title": "FastHTML", + "section": "Other languages and related projects", + "text": "Other languages and related projects\nIf you’re not a Python user, or are keen to try out a new language, we’ll list here other projects that have a similar approach to FastHTML. (Please reach out if you know of any other projects that you’d like to see added.)\n\nhtmgo (Go): “htmgo is a lightweight pure go way to build interactive websites / web applications using go & htmx. By combining the speed & simplicity of go + hypermedia attributes (htmx) to add interactivity to websites, all conveniently wrapped in pure go, you can build simple, fast, interactive websites without touching javascript. All compiled to a single deployable binary”\n\nIf you’re just interested in functional HTML components, rather than a full HTMX server solution, consider:\n\nfastcore.xml.FT: This is actually what FastHTML uses behind the scenes\nhtpy: Similar to fastcore.xml.FT, but with a somewhat different syntax\nelm-html: Elm’s built-in HTML library with a type-safe functional approach\nhiccup: Popular library for representing HTML in Clojure using vectors\nhiccl: HTML generation library for Common Lisp inspired by Clojure’s Hiccup\nFalco.Markup: F# HTML DSL and web framework with type-safe HTML generation\nLucid: Type-safe HTML generation for Haskell using monad transformers\ndream-html: Part of the Dream web framework for OCaml, provides type-safe HTML templating\n\nFor other hypermedia application platforms, not based on HTMX, take a look at:\n\nHotwire/Turbo: Rails-oriented framework that similarly uses HTML-over-the-wire\nLiveView: Phoenix framework’s solution for building interactive web apps with minimal JavaScript\nUnpoly: Another HTML-over-the-wire framework with progressive enhancement\nLivewire: Laravel’s take on building dynamic interfaces with minimal JavaScript", + "crumbs": [ + "Home", + "Get Started" + ] + }, + { + "objectID": "api/oauth.html", + "href": "api/oauth.html", + "title": "OAuth", + "section": "", + "text": "See the docs page for an explanation of how to use this.\n\nfrom IPython.display import Markdown\n\n\nsource\n\nGoogleAppClient\n\n GoogleAppClient (client_id, client_secret, code=None, scope=None,\n **kwargs)\n\nA WebApplicationClient for Google oauth2\n\nsource\n\n\nGitHubAppClient\n\n GitHubAppClient (client_id, client_secret, code=None, scope=None,\n **kwargs)\n\nA WebApplicationClient for GitHub oauth2\n\nsource\n\n\nHuggingFaceClient\n\n HuggingFaceClient (client_id, client_secret, code=None, scope=None,\n state=None, **kwargs)\n\nA WebApplicationClient for HuggingFace oauth2\n\nsource\n\n\nDiscordAppClient\n\n DiscordAppClient (client_id, client_secret, is_user=False, perms=0,\n scope=None, **kwargs)\n\nA WebApplicationClient for Discord oauth2\n\nsource\n\n\nAuth0AppClient\n\n Auth0AppClient (domain, client_id, client_secret, code=None, scope=None,\n redirect_uri='', **kwargs)\n\nA WebApplicationClient for Auth0 OAuth2\n\n# cli = GoogleAppClient.from_file('/Users/jhoward/subs_aai/_nbs/oauth-test/client_secret.json')\n\n\nsource\n\n\nWebApplicationClient.login_link\n\n WebApplicationClient.login_link (redirect_uri, scope=None, state=None)\n\nGet a login link for this client\nGenerating a login link that sends the user to the OAuth provider is done with client.login_link().\nIt can sometimes be useful to pass state to the OAuth provider, so that when the user returns you can pick up where they left off. This can be done by passing the state parameter.\n\nfrom fasthtml.common import *\nfrom fasthtml.jupyter import *\n\n\nredir_path = '/redirect'\nport = 8000\ncode_stor = None\n\n\napp,rt = fast_app()\nserver = JupyUvi(app, port=port)\n\n\n\n\n\n\nsource\n\n\nredir_url\n\n redir_url (request, redir_path, scheme=None)\n\nGet the redir url for the host in request\n\n@rt\ndef index(request):\n redir = redir_url(request, redir_path)\n return A('login', href=cli.login_link(redir), target='_blank')\n\n\nsource\n\n\n_AppClient.parse_response\n\n _AppClient.parse_response (code, redirect_uri)\n\nGet the token from the oauth2 server response\n\nsource\n\n\n_AppClient.get_info\n\n _AppClient.get_info (token=None)\n\nGet the info for authenticated user\n\nsource\n\n\n_AppClient.retr_info\n\n _AppClient.retr_info (code, redirect_uri)\n\nCombines parse_response and get_info\n\n@rt(redir_path)\ndef get(request, code:str):\n redir = redir_url(request, redir_path)\n info = cli.retr_info(code, redir)\n return P(f'Login successful for {info[\"name\"]}!')\n\n\n# HTMX()\n\n\nserver.stop()\n\n\nsource\n\n\n_AppClient.retr_id\n\n _AppClient.retr_id (code, redirect_uri)\n\nCall retr_info and then return id/subscriber value\nAfter logging in via the provider, the user will be redirected back to the supplied redirect URL. The request to this URL will contain a code parameter, which is used to get an access token and fetch the user’s profile information. See the explanation here for a worked example. You can either:\n\nUse client.retr_info(code) to get all the profile information, or\nUse client.retr_id(code) to get just the user’s ID.\n\nAfter either of these calls, you can also access the access token (used to revoke access, for example) with client.token[\"access_token\"].\n\nsource\n\n\nurl_match\n\n url_match (url, patterns=('^(localhost|127\\\\.0\\\\.0\\\\.1)(:\\\\d+)?$',))\n\n\nsource\n\n\nOAuth\n\n OAuth (app, cli, skip=None, redir_path='/redirect', error_path='/error',\n logout_path='/logout', login_path='/login', https=True,\n http_patterns=('^(localhost|127\\\\.0\\\\.0\\\\.1)(:\\\\d+)?$',))\n\nInitialize self. See help(type(self)) for accurate signature.", + "crumbs": [ + "Home", + "Source", + "OAuth" + ] + }, + { + "objectID": "api/js.html", + "href": "api/js.html", + "title": "Javascript examples", + "section": "", + "text": "To expedite fast development, FastHTML comes with several built-in Javascript and formatting components. These are largely provided to demonstrate FastHTML JS patterns. There’s far too many JS libs for FastHTML to wrap them all, and as shown here the code to add FastHTML support is very simple anyway.\n\nsource\n\nlight_media\n\n light_media (css:str)\n\nRender light media for day mode views\n\n\n\n\nType\nDetails\n\n\n\n\ncss\nstr\nCSS to be included in the light media query\n\n\n\n\nlight_media('.body {color: green;}')\n\n<style>@media (prefers-color-scheme: light) {.body {color: green;}}</style>\n\n\n\nsource\n\n\ndark_media\n\n dark_media (css:str)\n\nRender dark media for night mode views\n\n\n\n\nType\nDetails\n\n\n\n\ncss\nstr\nCSS to be included in the dark media query\n\n\n\n\ndark_media('.body {color: white;}')\n\n<style>@media (prefers-color-scheme: dark) {.body {color: white;}}</style>\n\n\n\nsource\n\n\nMarkdownJS\n\n MarkdownJS (sel='.marked')\n\nImplements browser-based markdown rendering.\n\n\n\n\nType\nDefault\nDetails\n\n\n\n\nsel\nstr\n.marked\nCSS selector for markdown elements\n\n\n\nUsage example here.\n\n__file__ = '../../fasthtml/katex.js'\n\n\nsource\n\n\nKatexMarkdownJS\n\n KatexMarkdownJS (sel='.marked', inline_delim='$', display_delim='$$',\n math_envs=None)\n\n\n\n\n\n\n\n\n\n\n\nType\nDefault\nDetails\n\n\n\n\nsel\nstr\n.marked\nCSS selector for markdown elements\n\n\ninline_delim\nstr\n$\nDelimiter for inline math\n\n\ndisplay_delim\nstr\n$$\nDelimiter for long math\n\n\nmath_envs\nNoneType\nNone\nList of environments to render as display math\n\n\n\nKatexMarkdown usage example:\nlongexample = r\"\"\"\nLong example:\n\n$$\\begin{array}{c}\n\n\\nabla \\times \\vec{\\mathbf{B}} -\\, \\frac1c\\, \\frac{\\partial\\vec{\\mathbf{E}}}{\\partial t} &\n= \\frac{4\\pi}{c}\\vec{\\mathbf{j}} \\nabla \\cdot \\vec{\\mathbf{E}} & = 4 \\pi \\rho \\\\\n\n\\nabla \\times \\vec{\\mathbf{E}}\\, +\\, \\frac1c\\, \\frac{\\partial\\vec{\\mathbf{B}}}{\\partial t} & = \\vec{\\mathbf{0}} \\\\\n\n\\nabla \\cdot \\vec{\\mathbf{B}} & = 0\n\n\\end{array}$$\n\"\"\"\n\napp, rt = fast_app(hdrs=[KatexMarkdownJS()])\n\n@rt('/')\ndef get():\n return Titled(\"Katex Examples\", \n # Assigning 'marked' class to components renders content as markdown\n P(cls='marked')(\"Inline example: $\\sqrt{3x-1}+(1+x)^2$\"),\n Div(cls='marked')(longexample)\n )\n\nsource\n\n\nHighlightJS\n\n HighlightJS (sel='pre code:not([data-highlighted=\"yes\"])',\n langs:str|list|tuple='python', light='atom-one-light',\n dark='atom-one-dark')\n\nImplements browser-based syntax highlighting. Usage example here.\n\n\n\n\n\n\n\n\n\n\nType\nDefault\nDetails\n\n\n\n\nsel\nstr\npre code:not([data-highlighted=“yes”])\nCSS selector for code elements. Default is industry standard, be careful before adjusting it\n\n\nlangs\nstr | list | tuple\npython\nLanguage(s) to highlight\n\n\nlight\nstr\natom-one-light\nLight theme\n\n\ndark\nstr\natom-one-dark\nDark theme\n\n\n\n\nsource\n\n\nSortableJS\n\n SortableJS (sel='.sortable', ghost_class='blue-background-class')\n\n\n\n\n\n\n\n\n\n\n\nType\nDefault\nDetails\n\n\n\n\nsel\nstr\n.sortable\nCSS selector for sortable elements\n\n\nghost_class\nstr\nblue-background-class\nWhen an element is being dragged, this is the class used to distinguish it from the rest\n\n\n\n\nsource\n\n\nMermaidJS\n\n MermaidJS (sel='.language-mermaid', theme='base')\n\nImplements browser-based Mermaid diagram rendering.\n\n\n\n\nType\nDefault\nDetails\n\n\n\n\nsel\nstr\n.language-mermaid\nCSS selector for mermaid elements\n\n\ntheme\nstr\nbase\nMermaid theme to use\n\n\n\napp, rt = fast_app(hdrs=[MermaidJS()])\n@rt('/')\ndef get():\n return Titled(\"Mermaid Examples\", \n # Assigning 'marked' class to components renders content as markdown\n Pre(Code(cls =\"language-mermaid\")('''flowchart TD\n A[main] --> B[\"fact(5)\"] --> C[\"fact(4)\"] --> D[\"fact(3)\"] --> E[\"fact(2)\"] --> F[\"fact(1)\"] --> G[\"fact(0)\"]\n ''')))\nIn a markdown file, just like a code cell you can define\n```mermaid\n graph TD\n A --> B \n B --> C \n C --> E\n```", + "crumbs": [ + "Home", + "Source", + "Javascript examples" + ] + }, + { + "objectID": "api/svg.html", + "href": "api/svg.html", + "title": "SVG", + "section": "", + "text": "from nbdev.showdoc import show_doc\nYou can create SVGs directly from strings, for instance (as always, use NotStr or Safe to tell FastHTML to not escape the text):\nsvg = '<svg width=\"50\" height=\"50\"><circle cx=\"20\" cy=\"20\" r=\"15\" fill=\"red\"></circle></svg>'\nshow(NotStr(svg))\nYou can also use libraries such as fa6-icons.\nTo create and modify SVGs using a Python API, use the FT elements in fasthtml.svg, discussed below.\nNote: fasthtml.common does NOT automatically export SVG elements. To get access to them, you need to import fasthtml.svg like so\nsource", + "crumbs": [ + "Home", + "Source", + "SVG" + ] + }, + { + "objectID": "api/svg.html#basic-shapes", + "href": "api/svg.html#basic-shapes", + "title": "SVG", + "section": "Basic shapes", + "text": "Basic shapes\nWe’ll define a simple function to display SVG shapes in this notebook:\n\ndef demo(el, h=50, w=50): return show(Svg(h=h,w=w)(el))\n\n\nsource\n\nRect\n\n Rect (width, height, x=0, y=0, fill=None, stroke=None, stroke_width=None,\n rx=None, ry=None, transform=None, opacity=None, clip=None,\n mask=None, filter=None, vector_effect=None, pointer_events=None,\n target_id=None, hx_vals=None, hx_target=None, id=None, cls=None,\n title=None, style=None, accesskey=None, contenteditable=None,\n dir=None, draggable=None, enterkeyhint=None, hidden=None,\n inert=None, inputmode=None, lang=None, popover=None,\n spellcheck=None, tabindex=None, translate=None, hx_get=None,\n hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,\n hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None,\n hx_select=None, hx_select_oob=None, hx_indicator=None,\n hx_push_url=None, hx_confirm=None, hx_disable=None,\n hx_replace_url=None, hx_disabled_elt=None, hx_ext=None,\n hx_headers=None, hx_history=None, hx_history_elt=None,\n hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None)\n\nA standard SVG rect element\nAll our shapes just create regular FT elements. The only extra functionality provided by most of them is to add additional defined kwargs to improve auto-complete in IDEs and notebooks, and re-order parameters so that positional args can also be used to save a bit of typing, e.g:\n\ndemo(Rect(30, 30, fill='blue', rx=8, ry=8))\n\n\n\n\n\nsource\n\n\nCircle\n\n Circle (r, cx=0, cy=0, fill=None, stroke=None, stroke_width=None,\n transform=None, opacity=None, clip=None, mask=None, filter=None,\n vector_effect=None, pointer_events=None, target_id=None,\n hx_vals=None, hx_target=None, id=None, cls=None, title=None,\n style=None, accesskey=None, contenteditable=None, dir=None,\n draggable=None, enterkeyhint=None, hidden=None, inert=None,\n inputmode=None, lang=None, popover=None, spellcheck=None,\n tabindex=None, translate=None, hx_get=None, hx_post=None,\n hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,\n hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None)\n\nA standard SVG circle element\n\ndemo(Circle(20, 25, 25, stroke='red', stroke_width=3))\n\n\n\n\n\nsource\n\n\nEllipse\n\n Ellipse (rx, ry, cx=0, cy=0, fill=None, stroke=None, stroke_width=None,\n transform=None, opacity=None, clip=None, mask=None, filter=None,\n vector_effect=None, pointer_events=None, target_id=None,\n hx_vals=None, hx_target=None, id=None, cls=None, title=None,\n style=None, accesskey=None, contenteditable=None, dir=None,\n draggable=None, enterkeyhint=None, hidden=None, inert=None,\n inputmode=None, lang=None, popover=None, spellcheck=None,\n tabindex=None, translate=None, hx_get=None, hx_post=None,\n hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,\n hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None)\n\nA standard SVG ellipse element\n\ndemo(Ellipse(20, 10, 25, 25))\n\n\n\n\n\nsource\n\n\ntransformd\n\n transformd (translate=None, scale=None, rotate=None, skewX=None,\n skewY=None, matrix=None)\n\nCreate an SVG transform kwarg dict\n\nrot = transformd(rotate=(45, 25, 25))\nrot\n\n{'transform': 'rotate(45,25,25)'}\n\n\n\ndemo(Ellipse(20, 10, 25, 25, **rot))\n\n\n\n\n\nsource\n\n\nLine\n\n Line (x1, y1, x2=0, y2=0, stroke='black', w=None, stroke_width=1,\n transform=None, opacity=None, clip=None, mask=None, filter=None,\n vector_effect=None, pointer_events=None, target_id=None,\n hx_vals=None, hx_target=None, id=None, cls=None, title=None,\n style=None, accesskey=None, contenteditable=None, dir=None,\n draggable=None, enterkeyhint=None, hidden=None, inert=None,\n inputmode=None, lang=None, popover=None, spellcheck=None,\n tabindex=None, translate=None, hx_get=None, hx_post=None,\n hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,\n hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,\n hx_sync=None, hx_validate=None)\n\nA standard SVG line element\n\ndemo(Line(20, 30, w=3))\n\n\n\n\n\nsource\n\n\nPolyline\n\n Polyline (*args, points=None, fill=None, stroke=None, stroke_width=None,\n transform=None, opacity=None, clip=None, mask=None,\n filter=None, vector_effect=None, pointer_events=None,\n target_id=None, hx_vals=None, hx_target=None, id=None,\n cls=None, title=None, style=None, accesskey=None,\n contenteditable=None, dir=None, draggable=None,\n enterkeyhint=None, hidden=None, inert=None, inputmode=None,\n lang=None, popover=None, spellcheck=None, tabindex=None,\n translate=None, hx_get=None, hx_post=None, hx_put=None,\n hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,\n hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None)\n\nA standard SVG polyline element\n\ndemo(Polyline((0,0), (10,10), (20,0), (30,10), (40,0),\n fill='yellow', stroke='blue', stroke_width=2))\n\n\n\n\n\ndemo(Polyline(points='0,0 10,10 20,0 30,10 40,0', fill='purple', stroke_width=2))\n\n\n\n\n\nsource\n\n\nPolygon\n\n Polygon (*args, points=None, fill=None, stroke=None, stroke_width=None,\n transform=None, opacity=None, clip=None, mask=None, filter=None,\n vector_effect=None, pointer_events=None, target_id=None,\n hx_vals=None, hx_target=None, id=None, cls=None, title=None,\n style=None, accesskey=None, contenteditable=None, dir=None,\n draggable=None, enterkeyhint=None, hidden=None, inert=None,\n inputmode=None, lang=None, popover=None, spellcheck=None,\n tabindex=None, translate=None, hx_get=None, hx_post=None,\n hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,\n hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None)\n\nA standard SVG polygon element\n\ndemo(Polygon((25,5), (43.3,15), (43.3,35), (25,45), (6.7,35), (6.7,15), \n fill='lightblue', stroke='navy', stroke_width=2))\n\n\n\n\n\ndemo(Polygon(points='25,5 43.3,15 43.3,35 25,45 6.7,35 6.7,15',\n fill='lightgreen', stroke='darkgreen', stroke_width=2))\n\n\n\n\n\nsource\n\n\nText\n\n Text (*args, x=0, y=0, font_family=None, font_size=None, fill=None,\n text_anchor=None, dominant_baseline=None, font_weight=None,\n font_style=None, text_decoration=None, transform=None,\n opacity=None, clip=None, mask=None, filter=None,\n vector_effect=None, pointer_events=None, target_id=None,\n hx_vals=None, hx_target=None, id=None, cls=None, title=None,\n style=None, accesskey=None, contenteditable=None, dir=None,\n draggable=None, enterkeyhint=None, hidden=None, inert=None,\n inputmode=None, lang=None, popover=None, spellcheck=None,\n tabindex=None, translate=None, hx_get=None, hx_post=None,\n hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,\n hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,\n hx_sync=None, hx_validate=None)\n\nA standard SVG text element\n\ndemo(Text(\"Hello!\", x=10, y=30))\n\nHello!", + "crumbs": [ + "Home", + "Source", + "SVG" + ] + }, + { + "objectID": "api/svg.html#paths", + "href": "api/svg.html#paths", + "title": "SVG", + "section": "Paths", + "text": "Paths\nPaths in SVGs are more complex, so we add a small (optional) fluent interface for constructing them:\n\nsource\n\nPathFT\n\n PathFT (tag:str, cs:tuple, attrs:dict=None, void_=False, **kwargs)\n\nA ‘Fast Tag’ structure, containing tag,children,and attrs\n\nsource\n\n\nPath\n\n Path (d='', fill=None, stroke=None, stroke_width=None, transform=None,\n opacity=None, clip=None, mask=None, filter=None,\n vector_effect=None, pointer_events=None, target_id=None,\n hx_vals=None, hx_target=None, id=None, cls=None, title=None,\n style=None, accesskey=None, contenteditable=None, dir=None,\n draggable=None, enterkeyhint=None, hidden=None, inert=None,\n inputmode=None, lang=None, popover=None, spellcheck=None,\n tabindex=None, translate=None, hx_get=None, hx_post=None,\n hx_put=None, hx_delete=None, hx_patch=None, hx_trigger=None,\n hx_swap=None, hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,\n hx_sync=None, hx_validate=None)\n\nCreate a standard path SVG element. This is a special object\nLet’s create a square shape, but using Path instead of Rect:\n\nM(10, 10): Move to starting point (10, 10)\nL(40, 10): Line to (40, 10) - top edge\nL(40, 40): Line to (40, 40) - right edge\nL(10, 40): Line to (10, 40) - bottom edge\nZ(): Close path - connects back to start\n\nM = Move to, L = Line to, Z = Close path\n\ndemo(Path(fill='none', stroke='purple', stroke_width=2\n ).M(10, 10).L(40, 10).L(40, 40).L(10, 40).Z())\n\n\n\n\nUsing curves we can create a spiral:\n\np = (Path(fill='none', stroke='purple', stroke_width=2)\n .M(25, 25)\n .C(25, 25, 20, 20, 30, 20)\n .C(40, 20, 40, 30, 30, 30)\n .C(20, 30, 20, 15, 35, 15)\n .C(50, 15, 50, 35, 25, 35)\n .C(0, 35, 0, 10, 40, 10)\n .C(80, 10, 80, 40, 25, 40))\ndemo(p, 50, 100)\n\n\n\n\nUsing arcs and curves we can create a map marker icon:\n\np = (Path(fill='red')\n .M(25,45)\n .C(25,45,10,35,10,25)\n .A(15,15,0,1,1,40,25)\n .C(40,35,25,45,25,45)\n .Z())\ndemo(p)\n\n\n\n\nBehind the scenes it’s just creating regular SVG path d attr – you can pass d in directly if you prefer.\n\nprint(p.d)\n\n M25 45 C25 45 10 35 10 25 A15 15 0 1 1 40 25 C40 35 25 45 25 45 Z\n\n\n\ndemo(Path(d='M25 45 C25 45 10 35 10 25 A15 15 0 1 1 40 25 C40 35 25 45 25 45 Z'))\n\n\n\n\n\nsource\n\n\nPathFT.M\n\n PathFT.M (x, y)\n\nMove to.\n\nsource\n\n\nPathFT.L\n\n PathFT.L (x, y)\n\nLine to.\n\nsource\n\n\nPathFT.H\n\n PathFT.H (x)\n\nHorizontal line to.\n\nsource\n\n\nPathFT.V\n\n PathFT.V (y)\n\nVertical line to.\n\nsource\n\n\nPathFT.Z\n\n PathFT.Z ()\n\nClose path.\n\nsource\n\n\nPathFT.C\n\n PathFT.C (x1, y1, x2, y2, x, y)\n\nCubic Bézier curve.\n\nsource\n\n\nPathFT.S\n\n PathFT.S (x2, y2, x, y)\n\nSmooth cubic Bézier curve.\n\nsource\n\n\nPathFT.Q\n\n PathFT.Q (x1, y1, x, y)\n\nQuadratic Bézier curve.\n\nsource\n\n\nPathFT.T\n\n PathFT.T (x, y)\n\nSmooth quadratic Bézier curve.\n\nsource\n\n\nPathFT.A\n\n PathFT.A (rx, ry, x_axis_rotation, large_arc_flag, sweep_flag, x, y)\n\nElliptical Arc.", + "crumbs": [ + "Home", + "Source", + "SVG" + ] + }, + { + "objectID": "api/svg.html#htmx-helpers", + "href": "api/svg.html#htmx-helpers", + "title": "SVG", + "section": "HTMX helpers", + "text": "HTMX helpers\n\nsource\n\nSvgOob\n\n SvgOob (*args, **kwargs)\n\nWraps an SVG shape as required for an HTMX OOB swap\nWhen returning an SVG shape out-of-band (OOB) in HTMX, you need to wrap it with SvgOob to have it appear correctly. (SvgOob is just a shortcut for Template(Svg(...)), which is the trick that makes SVG OOB swaps work.)\n\nsource\n\n\nSvgInb\n\n SvgInb (*args, **kwargs)\n\nWraps an SVG shape as required for an HTMX inband swap\nWhen returning an SVG shape in-band in HTMX, either have the calling element include hx_select='svg>*', or **svg_inb (which are two ways of saying the same thing), or wrap the response with SvgInb to have it appear correctly. (SvgInb is just a shortcut for the tuple (Svg(...), HtmxResponseHeaders(hx_reselect='svg>*')), which is the trick that makes SVG in-band swaps work.)", + "crumbs": [ + "Home", + "Source", + "SVG" + ] + }, + { + "objectID": "api/pico.html", + "href": "api/pico.html", + "title": "Pico.css components", + "section": "", + "text": "picocondlink is the class-conditional css link tag, and picolink is the regular tag.\n\nshow(picocondlink)\n\n\n\n\n\n\nsource\n\nset_pico_cls\n\n set_pico_cls ()\n\nRun this to make jupyter outputs styled with pico:\n\nset_pico_cls()\n\n\n\n\n\nsource\n\n\nCard\n\n Card (*c, header=None, footer=None, target_id=None, hx_vals=None,\n hx_target=None, id=None, cls=None, title=None, style=None,\n accesskey=None, contenteditable=None, dir=None, draggable=None,\n enterkeyhint=None, hidden=None, inert=None, inputmode=None,\n lang=None, popover=None, spellcheck=None, tabindex=None,\n translate=None, hx_get=None, hx_post=None, hx_put=None,\n hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,\n hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,\n hx_sync=None, hx_validate=None, **kwargs)\n\nA PicoCSS Card, implemented as an Article with optional Header and Footer\n\nshow(Card('body', header=P('head'), footer=P('foot')))\n\n\n head\n\nbody\n foot\n\n\n\n\n\nsource\n\n\nGroup\n\n Group (*c, target_id=None, hx_vals=None, hx_target=None, id=None,\n cls=None, title=None, style=None, accesskey=None,\n contenteditable=None, dir=None, draggable=None, enterkeyhint=None,\n hidden=None, inert=None, inputmode=None, lang=None, popover=None,\n spellcheck=None, tabindex=None, translate=None, hx_get=None,\n hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,\n hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None,\n hx_select=None, hx_select_oob=None, hx_indicator=None,\n hx_push_url=None, hx_confirm=None, hx_disable=None,\n hx_replace_url=None, hx_disabled_elt=None, hx_ext=None,\n hx_headers=None, hx_history=None, hx_history_elt=None,\n hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None, **kwargs)\n\nA PicoCSS Group, implemented as a Fieldset with role ‘group’\n\nshow(Group(Input(), Button(\"Save\")))\n\n\n \n Save\n\n\n\n\nsource\n\n\nSearch\n\n Search (*c, target_id=None, hx_vals=None, hx_target=None, id=None,\n cls=None, title=None, style=None, accesskey=None,\n contenteditable=None, dir=None, draggable=None,\n enterkeyhint=None, hidden=None, inert=None, inputmode=None,\n lang=None, popover=None, spellcheck=None, tabindex=None,\n translate=None, hx_get=None, hx_post=None, hx_put=None,\n hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,\n hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None, **kwargs)\n\nA PicoCSS Search, implemented as a Form with role ‘search’\n\nshow(Search(Input(type=\"search\"), Button(\"Search\")))\n\n\n \n Search\n\n\n\n\nsource\n\n\nGrid\n\n Grid (*c, cls='grid', target_id=None, hx_vals=None, hx_target=None,\n id=None, title=None, style=None, accesskey=None,\n contenteditable=None, dir=None, draggable=None, enterkeyhint=None,\n hidden=None, inert=None, inputmode=None, lang=None, popover=None,\n spellcheck=None, tabindex=None, translate=None, hx_get=None,\n hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,\n hx_trigger=None, hx_swap=None, hx_swap_oob=None, hx_include=None,\n hx_select=None, hx_select_oob=None, hx_indicator=None,\n hx_push_url=None, hx_confirm=None, hx_disable=None,\n hx_replace_url=None, hx_disabled_elt=None, hx_ext=None,\n hx_headers=None, hx_history=None, hx_history_elt=None,\n hx_inherit=None, hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None, **kwargs)\n\nA PicoCSS Grid, implemented as child Divs in a Div with class ‘grid’\n\ncolors = [Input(type=\"color\", value=o) for o in ('#e66465', '#53d2c5', '#f6b73c')]\nshow(Grid(*colors))\n\n\n \n\n \n\n \n\n\n\n\n\nsource\n\n\nDialogX\n\n DialogX (*c, open=None, header=None, footer=None, id=None,\n target_id=None, hx_vals=None, hx_target=None, cls=None,\n title=None, style=None, accesskey=None, contenteditable=None,\n dir=None, draggable=None, enterkeyhint=None, hidden=None,\n inert=None, inputmode=None, lang=None, popover=None,\n spellcheck=None, tabindex=None, translate=None, hx_get=None,\n hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,\n hx_trigger=None, hx_swap=None, hx_swap_oob=None,\n hx_include=None, hx_select=None, hx_select_oob=None,\n hx_indicator=None, hx_push_url=None, hx_confirm=None,\n hx_disable=None, hx_replace_url=None, hx_disabled_elt=None,\n hx_ext=None, hx_headers=None, hx_history=None,\n hx_history_elt=None, hx_inherit=None, hx_params=None,\n hx_preserve=None, hx_prompt=None, hx_request=None, hx_sync=None,\n hx_validate=None, **kwargs)\n\nA PicoCSS Dialog, with children inside a Card\n\nhdr = Div(Button(aria_label=\"Close\", rel=\"prev\"), P('confirm'))\nftr = Div(Button('Cancel', cls=\"secondary\"), Button('Confirm'))\nd = DialogX('thank you!', header=hdr, footer=ftr, open=None, id='dlgtest')\n# use js or htmx to display modal\n\n\nsource\n\n\nContainer\n\n Container (*args, target_id=None, hx_vals=None, hx_target=None, id=None,\n cls=None, title=None, style=None, accesskey=None,\n contenteditable=None, dir=None, draggable=None,\n enterkeyhint=None, hidden=None, inert=None, inputmode=None,\n lang=None, popover=None, spellcheck=None, tabindex=None,\n translate=None, hx_get=None, hx_post=None, hx_put=None,\n hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,\n hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None, **kwargs)\n\nA PicoCSS Container, implemented as a Main with class ‘container’\n\nsource\n\n\nPicoBusy\n\n PicoBusy ()", + "crumbs": [ + "Home", + "Source", + "Pico.css components" + ] + }, + { + "objectID": "api/xtend.html", + "href": "api/xtend.html", + "title": "Component extensions", + "section": "", + "text": "from pprint import pprint\nsource", + "crumbs": [ + "Home", + "Source", + "Component extensions" + ] + }, + { + "objectID": "api/xtend.html#forms", + "href": "api/xtend.html#forms", + "title": "Component extensions", + "section": "Forms", + "text": "Forms\n\nsource\n\nForm\n\n Form (*c, enctype='multipart/form-data', target_id=None, hx_vals=None,\n hx_target=None, id=None, cls=None, title=None, style=None,\n accesskey=None, contenteditable=None, dir=None, draggable=None,\n enterkeyhint=None, hidden=None, inert=None, inputmode=None,\n lang=None, popover=None, spellcheck=None, tabindex=None,\n translate=None, hx_get=None, hx_post=None, hx_put=None,\n hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,\n hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None, hx_request=None,\n hx_sync=None, hx_validate=None, **kwargs)\n\nA Form tag; identical to plain ft_hx version except default enctype='multipart/form-data'\n\nsource\n\n\nHidden\n\n Hidden (value:Any='', id:Any=None, target_id=None, hx_vals=None,\n hx_target=None, cls=None, title=None, style=None, accesskey=None,\n contenteditable=None, dir=None, draggable=None,\n enterkeyhint=None, hidden=None, inert=None, inputmode=None,\n lang=None, popover=None, spellcheck=None, tabindex=None,\n translate=None, hx_get=None, hx_post=None, hx_put=None,\n hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,\n hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None, **kwargs)\n\nAn Input of type ‘hidden’\n\nsource\n\n\nCheckboxX\n\n CheckboxX (checked:bool=False, label=None, value='1', id=None, name=None,\n target_id=None, hx_vals=None, hx_target=None, cls=None,\n title=None, style=None, accesskey=None, contenteditable=None,\n dir=None, draggable=None, enterkeyhint=None, hidden=None,\n inert=None, inputmode=None, lang=None, popover=None,\n spellcheck=None, tabindex=None, translate=None, hx_get=None,\n hx_post=None, hx_put=None, hx_delete=None, hx_patch=None,\n hx_trigger=None, hx_swap=None, hx_swap_oob=None,\n hx_include=None, hx_select=None, hx_select_oob=None,\n hx_indicator=None, hx_push_url=None, hx_confirm=None,\n hx_disable=None, hx_replace_url=None, hx_disabled_elt=None,\n hx_ext=None, hx_headers=None, hx_history=None,\n hx_history_elt=None, hx_inherit=None, hx_params=None,\n hx_preserve=None, hx_prompt=None, hx_request=None,\n hx_sync=None, hx_validate=None, **kwargs)\n\nA Checkbox optionally inside a Label, preceded by a Hidden with matching name\n\nshow(CheckboxX(True, 'Check me out!'))\n\n\n \nCheck me out!\n\n\n\nsource\n\n\nScript\n\n Script (code:str='', id=None, cls=None, title=None, style=None,\n attrmap=None, valmap=None, ft_cls=None, **kwargs)\n\nA Script tag that doesn’t escape its code\n\nsource\n\n\nStyle\n\n Style (*c, id=None, cls=None, title=None, style=None, attrmap=None,\n valmap=None, ft_cls=None, **kwargs)\n\nA Style tag that doesn’t escape its code", + "crumbs": [ + "Home", + "Source", + "Component extensions" + ] + }, + { + "objectID": "api/xtend.html#style-and-script-templates", + "href": "api/xtend.html#style-and-script-templates", + "title": "Component extensions", + "section": "Style and script templates", + "text": "Style and script templates\n\nsource\n\ndouble_braces\n\n double_braces (s)\n\nConvert single braces to double braces if next to special chars or newline\n\nsource\n\n\nundouble_braces\n\n undouble_braces (s)\n\nConvert double braces to single braces if next to special chars or newline\n\nsource\n\n\nloose_format\n\n loose_format (s, **kw)\n\nString format s using kw, without being strict about braces outside of template params\n\nsource\n\n\nScriptX\n\n ScriptX (fname, src=None, nomodule=None, type=None, _async=None,\n defer=None, charset=None, crossorigin=None, integrity=None,\n **kw)\n\nA script element with contents read from fname\n\nsource\n\n\nreplace_css_vars\n\n replace_css_vars (css, pre='tpl', **kwargs)\n\nReplace var(--) CSS variables with kwargs if name prefix matches pre\n\nsource\n\n\nStyleX\n\n StyleX (fname, **kw)\n\nA style element with contents read from fname and variables replaced from kw\n\nsource\n\n\nNbsp\n\n Nbsp ()\n\nA non-breaking space", + "crumbs": [ + "Home", + "Source", + "Component extensions" + ] + }, + { + "objectID": "api/xtend.html#surreal-and-js", + "href": "api/xtend.html#surreal-and-js", + "title": "Component extensions", + "section": "Surreal and JS", + "text": "Surreal and JS\n\nsource\n\nSurreal\n\n Surreal (code:str)\n\nWrap code in domReadyExecute and set m=me() and p=me('-')\n\nsource\n\n\nOn\n\n On (code:str, event:str='click', sel:str='', me=True)\n\nAn async surreal.js script block event handler for event on selector sel,p, making available parent p, event ev, and target e\n\nsource\n\n\nPrev\n\n Prev (code:str, event:str='click')\n\nAn async surreal.js script block event handler for event on previous sibling, with same vars as On\n\nsource\n\n\nNow\n\n Now (code:str, sel:str='')\n\nAn async surreal.js script block on selector me(sel)\n\nsource\n\n\nAnyNow\n\n AnyNow (sel:str, code:str)\n\nAn async surreal.js script block on selector any(sel)\n\nsource\n\n\nrun_js\n\n run_js (js, id=None, **kw)\n\nRun js script, auto-generating id based on name of caller if needed, and js-escaping any kw params\n\nsource\n\n\nHtmxOn\n\n HtmxOn (eventname:str, code:str)\n\n\nsource\n\n\njsd\n\n jsd (org, repo, root, path, prov='gh', typ='script', ver=None, esm=False,\n **kwargs)\n\njsdelivr Script or CSS Link tag, or URL", + "crumbs": [ + "Home", + "Source", + "Component extensions" + ] + }, + { + "objectID": "api/xtend.html#other-helpers", + "href": "api/xtend.html#other-helpers", + "title": "Component extensions", + "section": "Other helpers", + "text": "Other helpers\n\nsource\n\nTitled\n\n Titled (title:str='FastHTML app', *args, cls='container', target_id=None,\n hx_vals=None, hx_target=None, id=None, style=None,\n accesskey=None, contenteditable=None, dir=None, draggable=None,\n enterkeyhint=None, hidden=None, inert=None, inputmode=None,\n lang=None, popover=None, spellcheck=None, tabindex=None,\n translate=None, hx_get=None, hx_post=None, hx_put=None,\n hx_delete=None, hx_patch=None, hx_trigger=None, hx_swap=None,\n hx_swap_oob=None, hx_include=None, hx_select=None,\n hx_select_oob=None, hx_indicator=None, hx_push_url=None,\n hx_confirm=None, hx_disable=None, hx_replace_url=None,\n hx_disabled_elt=None, hx_ext=None, hx_headers=None,\n hx_history=None, hx_history_elt=None, hx_inherit=None,\n hx_params=None, hx_preserve=None, hx_prompt=None,\n hx_request=None, hx_sync=None, hx_validate=None, **kwargs)\n\nAn HTML partial containing a Title, and H1, and any provided children\n\nsource\n\n\nSocials\n\n Socials (title, site_name, description, image, url=None, w=1200, h=630,\n twitter_site=None, creator=None, card='summary')\n\nOG and Twitter social card headers\n\nsource\n\n\nFavicon\n\n Favicon (light_icon, dark_icon)\n\nLight and dark favicon headers\n\nsource\n\n\nclear\n\n clear (id)\n\n\nsource\n\n\nwith_sid\n\n with_sid (app, dest, path='/')", + "crumbs": [ + "Home", + "Source", + "Component extensions" + ] + }, + { + "objectID": "explains/routes.html", + "href": "explains/routes.html", + "title": "Routes", + "section": "", + "text": "Behaviour in FastHTML apps is defined by routes. The syntax is largely the same as the wonderful FastAPI (which is what you should be using instead of this if you’re creating a JSON service. FastHTML is mainly for making HTML web apps, not APIs).\nNote that you need to include the types of your parameters, so that FastHTML knows what to pass to your function. Here, we’re just expecting a string:\nfrom fasthtml.common import *\napp = FastHTML()\n\n@app.get('/user/{nm}')\ndef get_nm(nm:str): return f\"Good day to you, {nm}!\"\nNormally you’d save this into a file such as main.py, and then run it in uvicorn using:\nHowever, for testing, we can use Starlette’s TestClient to try it out:\nfrom starlette.testclient import TestClient\nclient = TestClient(app)\nr = client.get('/user/Jeremy')\nr\n\n<Response [200 OK]>\nTestClient uses httpx behind the scenes, so it returns a httpx.Response, which has a text attribute with our response body:\nr.text\n\n'Good day to you, Jeremy!'\nIn the previous example, the function name (get_nm) didn’t actually matter – we could have just called it _, for instance, since we never actually call it directly. It’s just called through HTTP. In fact, we often do call our functions _ when using this style of route, since that’s one less thing we have to worry about, naming.\nAn alternative approach to creating a route is to use app.route instead, in which case, you make the function name the HTTP method you want. Since this is such a common pattern, you might like to give a shorter name to app.route – we normally use rt:\nrt = app.route\n\n@rt('/')\ndef post(): return \"Going postal!\"\n\nclient.post('/').text\n\n'Going postal!'", + "crumbs": [ + "Home", + "Explanations", + "Routes" + ] + }, + { + "objectID": "explains/routes.html#combining-routes", + "href": "explains/routes.html#combining-routes", + "title": "Routes", + "section": "Combining Routes", + "text": "Combining Routes\nSometimes a FastHTML project can grow so weildy that putting all the routes into main.py becomes unweildy. Or, we install a FastHTML- or Starlette-based package that requires us to add routes.\nFirst let’s create a books.py module, that represents all the user-related views:\n\n# books.py\nbooks_app, rt = fast_app()\n\nbooks = ['A Guide to FastHTML', 'FastHTML Cookbook', 'FastHTML in 24 Hours']\n\n@rt(\"/\", name=\"list\")\ndef get():\n return Titled(\"Books\", *[P(book) for book in books])\n\nLet’s mount it in our main module:\nfrom books import books_app\n\n1app, rt = fast_app(routes=[Mount(\"/books\", books_app, name=\"books\")])\n\n@rt(\"/\")\ndef get():\n return Titled(\"Dashboard\",\n2 P(A(href=\"/books\")(\"Books\")),\n Hr(),\n3 P(A(link=uri(\"books:list\"))(\"Books\")),\n )\n\nserve()\n\n1\n\nWe use starlette.Mount to add the route to our routes list. We provide the name of books to make discovery and management of the links easier. More on that in items 2 and 3 of this annotations list\n\n2\n\nThis example link to the books list view is hand-crafted. Obvious in purpose, it makes changing link patterns in the future harder\n\n3\n\nThis example link uses the named URL route for the books. The advantage of this approach is it makes management of large numbers of link items easier.", + "crumbs": [ + "Home", + "Explanations", + "Routes" + ] + }, + { + "objectID": "explains/faq.html", + "href": "explains/faq.html", + "title": "FAQ", + "section": "", + "text": "Many editors, including Visual Studio Code, use PyLance to provide error checking for Python. However, PyLance’s error checking is just a guess – it can’t actually know whether your code is correct or not. PyLance particularly struggles with FastHTML’s syntax, which leads to it often reporting false error messages in FastHTML projects.\nTo avoid these misleading error messages, it’s best to disable some PyLance error checking in your FastHTML projects. Here’s how to do it in Visual Studio Code (the same approach should also work in other editors based on vscode, such as Cursor and GitHub Codespaces):\n\nOpen your FastHTML project\nPress Ctrl+Shift+P (or Cmd+Shift+P on Mac) to open the Command Palette\nType “Preferences: Open Workspace Settings (JSON)” and select it\nIn the JSON file that opens, add the following lines:\n\n{\n \"python.analysis.diagnosticSeverityOverrides\": {\n \"reportGeneralTypeIssues\": \"none\",\n \"reportOptionalMemberAccess\": \"none\",\n \"reportWildcardImportFromLibrary\": \"none\",\n \"reportRedeclaration\": \"none\",\n \"reportAttributeAccessIssue\": \"none\",\n \"reportInvalidTypeForm\": \"none\",\n \"reportAssignmentType\": \"none\",\n }\n}\n\nSave the file\n\nEven with PyLance diagnostics turned off, your FastHTML code will still run correctly. If you’re still seeing some false errors from PyLance, you can disable it entirely by adding this to your settings:\n{\n \"python.analysis.ignore\": [ \"*\" ]\n}", + "crumbs": [ + "Home", + "Explanations", + "FAQ" + ] + }, + { + "objectID": "explains/faq.html#why-does-my-editor-say-that-i-have-errors-in-my-fasthtml-code", + "href": "explains/faq.html#why-does-my-editor-say-that-i-have-errors-in-my-fasthtml-code", + "title": "FAQ", + "section": "", + "text": "Many editors, including Visual Studio Code, use PyLance to provide error checking for Python. However, PyLance’s error checking is just a guess – it can’t actually know whether your code is correct or not. PyLance particularly struggles with FastHTML’s syntax, which leads to it often reporting false error messages in FastHTML projects.\nTo avoid these misleading error messages, it’s best to disable some PyLance error checking in your FastHTML projects. Here’s how to do it in Visual Studio Code (the same approach should also work in other editors based on vscode, such as Cursor and GitHub Codespaces):\n\nOpen your FastHTML project\nPress Ctrl+Shift+P (or Cmd+Shift+P on Mac) to open the Command Palette\nType “Preferences: Open Workspace Settings (JSON)” and select it\nIn the JSON file that opens, add the following lines:\n\n{\n \"python.analysis.diagnosticSeverityOverrides\": {\n \"reportGeneralTypeIssues\": \"none\",\n \"reportOptionalMemberAccess\": \"none\",\n \"reportWildcardImportFromLibrary\": \"none\",\n \"reportRedeclaration\": \"none\",\n \"reportAttributeAccessIssue\": \"none\",\n \"reportInvalidTypeForm\": \"none\",\n \"reportAssignmentType\": \"none\",\n }\n}\n\nSave the file\n\nEven with PyLance diagnostics turned off, your FastHTML code will still run correctly. If you’re still seeing some false errors from PyLance, you can disable it entirely by adding this to your settings:\n{\n \"python.analysis.ignore\": [ \"*\" ]\n}", + "crumbs": [ + "Home", + "Explanations", + "FAQ" + ] + }, + { + "objectID": "explains/faq.html#why-the-distinctive-coding-style", + "href": "explains/faq.html#why-the-distinctive-coding-style", + "title": "FAQ", + "section": "Why the distinctive coding style?", + "text": "Why the distinctive coding style?\nFastHTML coding style is the fastai coding style.\nIf you are coming from a data science background the fastai coding style may already be your preferred style.\nIf you are coming from a PEP-8 background where the use of ruff is encouraged, there is a learning curve. However, once you get used to the fastai coding style you may discover yourself appreciating the concise nature of this style. It also encourages using more functional programming tooling, which is both productive and fun. Having said that, it’s entirely optional!", + "crumbs": [ + "Home", + "Explanations", + "FAQ" + ] + }, + { + "objectID": "explains/faq.html#why-not-jsx", + "href": "explains/faq.html#why-not-jsx", + "title": "FAQ", + "section": "Why not JSX?", + "text": "Why not JSX?\nMany have asked! We think there’s no benefit… Python’s positional and kw args precisely 1:1 map already to html/xml children and attrs, so there’s no need for a new syntax.\nWe wrote some more thoughts on Why Python HTML components over Jinja2, Mako, or JSX here.", + "crumbs": [ + "Home", + "Explanations", + "FAQ" + ] + }, + { + "objectID": "explains/faq.html#why-use-import", + "href": "explains/faq.html#why-use-import", + "title": "FAQ", + "section": "Why use import *", + "text": "Why use import *\nFirst, through the use of the __all__ attribute in our Python modules we control what actually gets imported. So there’s no risk of namespace pollution.\nSecond, our style lends itself to working in rather compact Jupyter notebooks and small Python modules. Hence we know about the source code whose libraries we import * from. This terseness means we can develop faster. We’re a small team, and any edge we can gain is important to us.\nThird, for external libraries, be it core Python, SQLAlchemy, or other things we do tend to use explicit imports. In part to avoid namespace collisions, and also as reference to know where things are coming from.\nWe’ll finish by saying a lot of our users employ explicit imports. If that’s the path you want to take, we encourage the use of from fasthtml import common as fh. The acronym of fh makes it easy to recognize that a symbol is from the FastHTML library.", + "crumbs": [ + "Home", + "Explanations", + "FAQ" + ] + }, + { + "objectID": "explains/faq.html#can-fasthtml-be-used-for-dashboards", + "href": "explains/faq.html#can-fasthtml-be-used-for-dashboards", + "title": "FAQ", + "section": "Can FastHTML be used for dashboards?", + "text": "Can FastHTML be used for dashboards?\nYes it can. In fact, it excels at building dashboards. In addition to being great for building static dashboards, because of its foundation in ASGI and tech stack, FastHTML natively supports Websockets. That means using FastHTML we can create dashboards that autoupdate.", + "crumbs": [ + "Home", + "Explanations", + "FAQ" + ] + }, + { + "objectID": "explains/faq.html#why-is-fasthtml-developed-using-notebooks", + "href": "explains/faq.html#why-is-fasthtml-developed-using-notebooks", + "title": "FAQ", + "section": "Why is FastHTML developed using notebooks?", + "text": "Why is FastHTML developed using notebooks?\nSome people are under the impression that writing software in notebooks is bad.\nWatch this video. We’ve used Jupyter notebooks exported via nbdev to write a wide range of “very serious” software projects over the last three years. This includes deep learning libraries, API clients, Python language extensions, terminal user interfaces, web frameworks, and more!\nnbdev is a Jupyter-powered tool for writing software. Traditional programming environments throw away the result of your exploration in REPLs or notebooks. nbdev makes exploration an integral part of your workflow, all while promoting software engineering best practices.", + "crumbs": [ + "Home", + "Explanations", + "FAQ" + ] + }, + { + "objectID": "explains/faq.html#why-not-pyproject.toml-for-packaging", + "href": "explains/faq.html#why-not-pyproject.toml-for-packaging", + "title": "FAQ", + "section": "Why not pyproject.toml for packaging?", + "text": "Why not pyproject.toml for packaging?\nFastHTML uses a setup.py module instead of a pyproject.toml file to configure itself for installation. The reason for this is pyproject.toml is not compatible with nbdev, which is what is used to write and build FastHTML.\nThe nbdev project spent around a year trying to move to pyproject.toml but there was insufficient functionality in the toml-based approach to complete the transition.", + "crumbs": [ + "Home", + "Explanations", + "FAQ" + ] + }, + { + "objectID": "explains/explaining_xt_components.html", + "href": "explains/explaining_xt_components.html", + "title": "FT Components", + "section": "", + "text": "FT, or ‘FastTags’, are the display components of FastHTML. In fact, the word “components” in the context of FastHTML is often synonymous with FT.\nFor example, when we look at a FastHTML app, in particular the views, as well as various functions and other objects, we see something like the code snippet below. It’s the return statement that we want to pay attention to:\nfrom fasthtml.common import *\n\ndef example():\n # The code below is a set of ft components\n return Div(\n H1(\"FastHTML APP\"),\n P(\"Let's do this\"),\n cls=\"go\"\n )\nLet’s go ahead and call our function and print the result:\nexample()\n\n<div class=\"go\">\n <h1>FastHTML APP</h1>\n <p>Let's do this</p>\n</div>\nAs you can see, when returned to the user from a Python callable, like a function, the ft components are transformed into their string representations of XML or XML-like content such as HTML. More concisely, ft turns Python objects into HTML.\nNow that we know what ft components look and behave like we can begin to understand them. At their most fundamental level, ft components:", + "crumbs": [ + "Home", + "Explanations", + "**FT** Components" + ] + }, + { + "objectID": "explains/explaining_xt_components.html#how-fasthtml-names-ft-components", + "href": "explains/explaining_xt_components.html#how-fasthtml-names-ft-components", + "title": "FT Components", + "section": "How FastHTML names ft components", + "text": "How FastHTML names ft components\nWhen it comes to naming ft components, FastHTML appears to break from PEP8. Specifically, PEP8 specifies that when naming variables, functions and instantiated classes we use the snake_case_pattern. That is to say, lowercase with words separated by underscores. However, FastHTML uses PascalCase for ft components.\nThere’s a couple of reasons for this:\n\nft components can be made from any callable type, so adhering to any one pattern doesn’t make much sense\nIt makes for easier reading of FastHTML code, as anything that is PascalCase is probably an ft component", + "crumbs": [ + "Home", + "Explanations", + "**FT** Components" + ] + }, + { + "objectID": "explains/explaining_xt_components.html#default-ft-components", + "href": "explains/explaining_xt_components.html#default-ft-components", + "title": "FT Components", + "section": "Default FT components", + "text": "Default FT components\nFastHTML has over 150 FT components designed to accelerate web development. Most of these mirror HTML tags such as <div>, <p>, <a>, <title>, and more. However, there are some extra tags added, including:\n\nTitled, a combination of the Title() and H1() tags\nSocials, renders popular social media tags", + "crumbs": [ + "Home", + "Explanations", + "**FT** Components" + ] + }, + { + "objectID": "explains/explaining_xt_components.html#the-fasthtml.ft-namespace", + "href": "explains/explaining_xt_components.html#the-fasthtml.ft-namespace", + "title": "FT Components", + "section": "The fasthtml.ft Namespace", + "text": "The fasthtml.ft Namespace\nSome people prefer to write code using namespaces while adhering to PEP8. If that’s a preference, projects can be coded using the fasthtml.ft namespace.\n\nfrom fasthtml import ft\n\nft.Ul(\n ft.Li(\"one\"),\n ft.Li(\"two\"),\n ft.Li(\"three\")\n)\n\n<ul>\n <li>one</li>\n <li>two</li>\n <li>three</li>\n</ul>", + "crumbs": [ + "Home", + "Explanations", + "**FT** Components" + ] + }, + { + "objectID": "explains/explaining_xt_components.html#attributes", + "href": "explains/explaining_xt_components.html#attributes", + "title": "FT Components", + "section": "Attributes", + "text": "Attributes\nThis example demonstrates many important things to know about how ft components handle attributes.\n#| echo: False\n1Label(\n \"Choose an option\", \n Select(\n2 Option(\"one\", value=\"1\", selected=True),\n3 Option(\"two\", value=\"2\", selected=False),\n4 Option(\"three\", value=3),\n5 cls=\"selector\",\n6 _id=\"counter\",\n7 **{'@click':\"alert('Clicked');\"},\n ),\n8 _for=\"counter\",\n)\n\n1\n\nLine 2 demonstrates that FastHTML appreciates Labels surrounding their fields.\n\n2\n\nOn line 5, we can see that attributes set to the boolean value of True are rendered with just the name of the attribute.\n\n3\n\nOn line 6, we demonstrate that attributes set to the boolean value of False do not appear in the rendered output.\n\n4\n\nLine 7 is an example of how integers and other non-string values in the rendered output are converted to strings.\n\n5\n\nLine 8 is where we set the HTML class using the cls argument. We use cls here as class is a reserved word in Python. During the rendering process this will be converted to the word “class”.\n\n6\n\nLine 9 demonstrates that any named argument passed into an ft component will have the leading underscore stripped away before rendering. Useful for handling reserved words in Python.\n\n7\n\nOn line 10 we have an attribute name that cannot be represented as a python variable. In cases like these, we can use an unpacked dict to represent these values.\n\n8\n\nThe use of _for on line 12 is another demonstration of an argument having the leading underscore stripped during render. We can also use fr as that will be expanded to for.\n\n\nThis renders the following HTML snippet:\n\nLabel(\n \"Choose an option\", \n Select(\n Option(\"one\", value=\"1\", selected=True),\n Option(\"two\", value=\"2\", selected=False),\n Option(\"three\", value=3), # <4>,\n cls=\"selector\",\n _id=\"counter\",\n **{'@click':\"alert('Clicked');\"},\n ),\n _for=\"counter\",\n)\n\n<label for=\"counter\">\nChoose an option\n <select id=\"counter\" @click=\"alert('Clicked');\" class=\"selector\" name=\"counter\">\n <option value=\"1\" selected>one</option>\n <option value=\"2\" >two</option>\n <option value=\"3\">three</option>\n </select>\n</label>", + "crumbs": [ + "Home", + "Explanations", + "**FT** Components" + ] + }, + { + "objectID": "explains/explaining_xt_components.html#defining-new-ft-components", + "href": "explains/explaining_xt_components.html#defining-new-ft-components", + "title": "FT Components", + "section": "Defining new ft components", + "text": "Defining new ft components\nIt is possible and sometimes useful to create your own ft components that generate non-standard tags that are not in the FastHTML library. FastHTML supports created and defining those new tags flexibly.\nFor more information, see the Defining new ft components reference page.", + "crumbs": [ + "Home", + "Explanations", + "**FT** Components" + ] + }, + { + "objectID": "explains/explaining_xt_components.html#ft-components-and-type-hints", + "href": "explains/explaining_xt_components.html#ft-components-and-type-hints", + "title": "FT Components", + "section": "FT components and type hints", + "text": "FT components and type hints\nIf you use type hints, we strongly suggest that FT components be treated as the Any type.\nThe reason is that FastHTML leverages python’s dynamic features to a great degree. Especially when it comes to FT components, which can evaluate out to be FT|str|None|tuple as well as anything that supports the __ft__, __html__, and __str__ method. That’s enough of the Python stack that assigning anything but Any to be the FT type will prove an exercise in frustation.", + "crumbs": [ + "Home", + "Explanations", + "**FT** Components" + ] + }, + { + "objectID": "ref/handlers.html", + "href": "ref/handlers.html", + "title": "Handling handlers", + "section": "", + "text": "from fasthtml.common import *\nfrom collections import namedtuple\nfrom typing import TypedDict\nfrom datetime import datetime\nimport json,time\napp = FastHTML()\nThe FastHTML class is the main application class for FastHTML apps.\nrt = app.route\napp.route is used to register route handlers. It is a decorator, which means we place it before a function that is used as a handler. Because it’s used frequently in most FastHTML applications, we often alias it as rt, as we do here.", + "crumbs": [ + "Home", + "Reference", + "Handling handlers" + ] + }, + { + "objectID": "ref/handlers.html#basic-route-handling", + "href": "ref/handlers.html#basic-route-handling", + "title": "Handling handlers", + "section": "Basic Route Handling", + "text": "Basic Route Handling\n\n@rt(\"/hi\")\ndef get(): return 'Hi there'\n\nHandler functions can return strings directly. These strings are sent as the response body to the client.\n\ncli = Client(app)\n\nClient is a test client for FastHTML applications. It allows you to simulate requests to your app without running a server.\n\ncli.get('/hi').text\n\n'Hi there'\n\n\nThe get method on a Client instance simulates GET requests to the app. It returns a response object that has a .text attribute, which you can use to access the body of the response. It calls httpx.get internally – all httpx HTTP verbs are supported.\n\n@rt(\"/hi\")\ndef post(): return 'Postal'\ncli.post('/hi').text\n\n'Postal'\n\n\nHandler functions can be defined for different HTTP methods on the same route. Here, we define a post handler for the /hi route. The Client instance can simulate different HTTP methods, including POST requests.", + "crumbs": [ + "Home", + "Reference", + "Handling handlers" + ] + }, + { + "objectID": "ref/handlers.html#request-and-response-objects", + "href": "ref/handlers.html#request-and-response-objects", + "title": "Handling handlers", + "section": "Request and Response Objects", + "text": "Request and Response Objects\n\n@app.get(\"/hostie\")\ndef show_host(req): return req.headers['host']\ncli.get('/hostie').text\n\n'testserver'\n\n\nHandler functions can accept a req (or request) parameter, which represents the incoming request. This object contains information about the request, including headers. In this example, we return the host header from the request. The test client uses ‘testserver’ as the default host.\nIn this example, we use @app.get(\"/hostie\") instead of @rt(\"/hostie\"). The @app.get() decorator explicitly specifies the HTTP method (GET) for the route, while @rt() by default handles both GET and POST requests.\n\n@rt\ndef yoyo(): return 'a yoyo'\ncli.post('/yoyo').text\n\n'a yoyo'\n\n\nIf the @rt decorator is used without arguments, it uses the function name as the route path. Here, the yoyo function becomes the handler for the /yoyo route. This handler responds to GET and POST methods, since a specific method wasn’t provided.\n\n@rt\ndef ft1(): return Html(Div('Text.'))\nprint(cli.get('/ft1').text)\n\n <html>\n <div>Text.</div>\n </html>\n\n\n\nHandler functions can return FT objects, which are automatically converted to HTML strings. The FT class can take other FT components as arguments, such as Div. This allows for easy composition of HTML elements in your responses.\n\n@app.get\ndef autopost(): return Html(Div('Text.', hx_post=yoyo.to()))\nprint(cli.get('/autopost').text)\n\n <html>\n <div hx-post=\"/yoyo\">Text.</div>\n </html>\n\n\n\nThe rt decorator modifies the yoyo function by adding an rt() method. This method returns the route path associated with the handler. It’s a convenient way to reference the route of a handler function dynamically.\nIn the example, yoyo.to() is used as the value for hx_post. This means when the div is clicked, it will trigger an HTMX POST request to the route of the yoyo handler. This approach allows for flexible, DRY code by avoiding hardcoded route strings and automatically updating if the route changes.\nThis pattern is particularly useful in larger applications where routes might change, or when building reusable components that need to reference their own routes dynamically.\n\n@app.get\ndef autoget(): return Html(Body(Div('Text.', cls='px-2', hx_post=show_host.to(a='b'))))\nprint(cli.get('/autoget').text)\n\n <html>\n <body>\n <div hx-post=\"/hostie?a=b\" class=\"px-2\">Text.</div>\n </body>\n </html>\n\n\n\nThe rt() method of handler functions can also accept parameters. When called with parameters, it returns the route path with a query string appended. In this example, show_host.to(a='b') generates the path /hostie?a=b.\nThe Body component is used here to demonstrate nesting of FT components. Div is nested inside Body, showcasing how you can create more complex HTML structures.\nThe cls parameter is used to add a CSS class to the Div. This translates to the class attribute in the rendered HTML. (class can’t be used as a parameter name directly in Python since it’s a reserved word.)\n\n@rt('/ft2')\ndef get(): return Title('Foo'),H1('bar')\nprint(cli.get('/ft2').text)\n\n <!doctype html>\n <html>\n <head>\n <title>Foo</title>\n <meta charset=\"utf-8\">\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1, viewport-fit=cover\">\n<script src=\"https://unpkg.com/htmx.org@next/dist/htmx.min.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/answerdotai/fasthtml-js@1.0.4/fasthtml.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/answerdotai/surreal@main/surreal.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/gnat/css-scope-inline@main/script.js\"></script><script>\n function sendmsg() {\n window.parent.postMessage({height: document.documentElement.offsetHeight}, '*');\n }\n window.onload = function() {\n sendmsg();\n document.body.addEventListener('htmx:afterSettle', sendmsg);\n document.body.addEventListener('htmx:wsAfterMessage', sendmsg);\n };</script> </head>\n <body>\n <h1>bar</h1>\n </body>\n </html>\n\n\n\nHandler functions can return multiple FT objects as a tuple. The first item is treated as the Title, and the rest are added to the Body. When the request is not an HTMX request, FastHTML automatically adds necessary HTML boilerplate, including default head content with required scripts.\nWhen using app.route (or rt), if the function name matches an HTTP verb (e.g., get, post, put, delete), that HTTP method is automatically used for the route. In this case, a path must be explicitly provided as an argument to the decorator.\n\nhxhdr = {'headers':{'hx-request':\"1\"}}\nprint(cli.get('/ft2', **hxhdr).text)\n\n <title>Foo</title>\n <h1>bar</h1>\n\n\n\nFor HTMX requests (indicated by the hx-request header), FastHTML returns only the specified components without the full HTML structure. This allows for efficient partial page updates in HTMX applications.\n\n@rt('/ft3')\ndef get(): return H1('bar')\nprint(cli.get('/ft3', **hxhdr).text)\n\n <h1>bar</h1>\n\n\n\nWhen a handler function returns a single FT object for an HTMX request, it’s rendered as a single HTML partial.\n\n@rt('/ft4')\ndef get(): return Html(Head(Title('hi')), Body(P('there')))\n\nprint(cli.get('/ft4').text)\n\n <html>\n <head>\n <title>hi</title>\n </head>\n <body>\n <p>there</p>\n </body>\n </html>\n\n\n\nHandler functions can return a complete Html structure, including Head and Body components. When a full HTML structure is returned, FastHTML doesn’t add any additional boilerplate. This gives you full control over the HTML output when needed.\n\n@rt\ndef index(): return \"welcome!\"\nprint(cli.get('/').text)\n\nwelcome!\n\n\nThe index function is a special handler in FastHTML. When defined without arguments to the @rt decorator, it automatically becomes the handler for the root path ('/'). This is a convenient way to define the main page or entry point of your application.", + "crumbs": [ + "Home", + "Reference", + "Handling handlers" + ] + }, + { + "objectID": "ref/handlers.html#path-and-query-parameters", + "href": "ref/handlers.html#path-and-query-parameters", + "title": "Handling handlers", + "section": "Path and Query Parameters", + "text": "Path and Query Parameters\n\n@rt('/user/{nm}', name='gday')\ndef get(nm:str=''): return f\"Good day to you, {nm}!\"\ncli.get('/user/Alexis').text\n\n'Good day to you, Alexis!'\n\n\nHandler functions can use path parameters, defined using curly braces in the route – this is implemented by Starlette directly, so all Starlette path parameters can be used. These parameters are passed as arguments to the function.\nThe name parameter in the decorator allows you to give the route a name, which can be used for URL generation.\nIn this example, {nm} in the route becomes the nm parameter in the function. The function uses this parameter to create a personalized greeting.\n\n@app.get\ndef autolink(): return Html(Div('Text.', link=uri('gday', nm='Alexis')))\nprint(cli.get('/autolink').text)\n\n <html>\n <div href=\"/user/Alexis\">Text.</div>\n </html>\n\n\n\nThe uri function is used to generate URLs for named routes. It takes the route name as its first argument, followed by any path or query parameters needed for that route.\nIn this example, uri('gday', nm='Alexis') generates the URL for the route named ‘gday’ (which we defined earlier as ‘/user/{nm}’), with ‘Alexis’ as the value for the ‘nm’ parameter.\nThe link parameter in FT components sets the href attribute of the rendered HTML element. By using uri(), we can dynamically generate correct URLs even if the underlying route structure changes.\nThis approach promotes maintainable code by centralizing route definitions and avoiding hardcoded URLs throughout the application.\n\n@rt('/link')\ndef get(req): return f\"{req.url_for('gday', nm='Alexis')}; {req.url_for('show_host')}\"\n\ncli.get('/link').text\n\n'http://testserver/user/Alexis; http://testserver/hostie'\n\n\nThe url_for method of the request object can be used to generate URLs for named routes. It takes the route name as its first argument, followed by any path parameters needed for that route.\nIn this example, req.url_for('gday', nm='Alexis') generates the full URL for the route named ‘gday’, including the scheme and host. Similarly, req.url_for('show_host') generates the URL for the ‘show_host’ route.\nThis method is particularly useful when you need to generate absolute URLs, such as for email links or API responses. It ensures that the correct host and scheme are included, even if the application is accessed through different domains or protocols.\n\napp.url_path_for('gday', nm='Jeremy')\n\n'/user/Jeremy'\n\n\nThe url_path_for method of the application can be used to generate URL paths for named routes. Unlike url_for, it returns only the path component of the URL, without the scheme or host.\nIn this example, app.url_path_for('gday', nm='Jeremy') generates the path ‘/user/Jeremy’ for the route named ‘gday’.\nThis method is useful when you need relative URLs or just the path component, such as for internal links or when constructing URLs in a host-agnostic manner.\n\n@rt('/oops')\ndef get(nope): return nope\nr = cli.get('/oops?nope=1')\nprint(r)\nr.text\n\n<Response [200 OK]>\n\n\n/Users/wgilliam/development/projects/aai/fasthtml/fasthtml/core.py:188: UserWarning: `nope has no type annotation and is not a recognised special name, so is ignored.\n if arg!='resp': warn(f\"`{arg} has no type annotation and is not a recognised special name, so is ignored.\")\n\n\n''\n\n\nHandler functions can include parameters, but they must be type-annotated or have special names (like req) to be recognized. In this example, the nope parameter is not annotated, so it’s ignored, resulting in a warning.\nWhen a parameter is ignored, it doesn’t receive the value from the query string. This can lead to unexpected behavior, as the function attempts to return nope, which is undefined.\nThe cli.get('/oops?nope=1') call succeeds with a 200 OK status because the handler doesn’t raise an exception, but it returns an empty response, rather than the intended value.\nTo fix this, you should either add a type annotation to the parameter (e.g., def get(nope: str):) or use a recognized special name like req.\n\n@rt('/html/{idx}')\ndef get(idx:int): return Body(H4(f'Next is {idx+1}.'))\nprint(cli.get('/html/1', **hxhdr).text)\n\n <body>\n <h4>Next is 2.</h4>\n </body>\n\n\n\nPath parameters can be type-annotated, and FastHTML will automatically convert them to the specified type if possible. In this example, idx is annotated as int, so it’s converted from the string in the URL to an integer.\n\nreg_re_param(\"imgext\", \"ico|gif|jpg|jpeg|webm\")\n\n@rt(r'/static/{path:path}{fn}.{ext:imgext}')\ndef get(fn:str, path:str, ext:str): return f\"Getting {fn}.{ext} from /{path}\"\n\nprint(cli.get('/static/foo/jph.ico').text)\n\nGetting jph.ico from /foo/\n\n\nThe reg_re_param function is used to register custom path parameter types using regular expressions. Here, we define a new path parameter type called “imgext” that matches common image file extensions.\nHandler functions can use complex path patterns with multiple parameters and custom types. In this example, the route pattern r'/static/{path:path}{fn}.{ext:imgext}' uses three path parameters:\n\npath: A Starlette built-in type that matches any path segments\nfn: The filename without extension\next: Our custom “imgext” type that matches specific image extensions\n\n\nModelName = str_enum('ModelName', \"alexnet\", \"resnet\", \"lenet\")\n\n@rt(\"/models/{nm}\")\ndef get(nm:ModelName): return nm\n\nprint(cli.get('/models/alexnet').text)\n\nalexnet\n\n\nWe define ModelName as an enum with three possible values: “alexnet”, “resnet”, and “lenet”. Handler functions can use these enum types as parameter annotations. In this example, the nm parameter is annotated with ModelName, which ensures that only valid model names are accepted.\nWhen a request is made with a valid model name, the handler function returns that name. This pattern is useful for creating type-safe APIs with a predefined set of valid values.\n\n@rt(\"/files/{path}\")\nasync def get(path: Path): return path.with_suffix('.txt')\nprint(cli.get('/files/foo').text)\n\nfoo.txt\n\n\nHandler functions can use Path objects as parameter types. The Path type is from Python’s standard library pathlib module, which provides an object-oriented interface for working with file paths. In this example, the path parameter is annotated with Path, so FastHTML automatically converts the string from the URL to a Path object.\nThis approach is particularly useful when working with file-related routes, as it provides a convenient and platform-independent way to handle file paths.\n\nfake_db = [{\"name\": \"Foo\"}, {\"name\": \"Bar\"}]\n\n@rt(\"/items/\")\ndef get(idx:int|None = 0): return fake_db[idx]\nprint(cli.get('/items/?idx=1').text)\n\n{\"name\":\"Bar\"}\n\n\nHandler functions can use query parameters, which are automatically parsed from the URL. In this example, idx is a query parameter with a default value of 0. It’s annotated as int|None, allowing it to be either an integer or None.\nThe function uses this parameter to index into a fake database (fake_db). When a request is made with a valid idx query parameter, the handler returns the corresponding item from the database.\n\nprint(cli.get('/items/').text)\n\n{\"name\":\"Foo\"}\n\n\nWhen no idx query parameter is provided, the handler function uses the default value of 0. This results in returning the first item from the fake_db list, which is {\"name\":\"Foo\"}.\nThis behavior demonstrates how default values for query parameters work in FastHTML. They allow the API to have a sensible default behavior when optional parameters are not provided.\n\nprint(cli.get('/items/?idx=g'))\n\n<Response [404 Not Found]>\n\n\nWhen an invalid value is provided for a typed query parameter, FastHTML returns a 404 Not Found response. In this example, ‘g’ is not a valid integer for the idx parameter, so the request fails with a 404 status.\nThis behavior ensures type safety and prevents invalid inputs from reaching the handler function.\n\n@app.get(\"/booly/\")\ndef _(coming:bool=True): return 'Coming' if coming else 'Not coming'\nprint(cli.get('/booly/?coming=true').text)\nprint(cli.get('/booly/?coming=no').text)\n\nComing\nNot coming\n\n\nHandler functions can use boolean query parameters. In this example, coming is a boolean parameter with a default value of True. FastHTML automatically converts string values like ‘true’, ‘false’, ‘1’, ‘0’, ‘on’, ‘off’, ‘yes’, and ‘no’ to their corresponding boolean values.\nThe underscore _ is used as the function name in this example to indicate that the function’s name is not important or won’t be referenced elsewhere. This is a common Python convention for throwaway or unused variables, and it works here because FastHTML uses the route decorator parameter, when provided, to determine the URL path, not the function name. By default, both get and post methods can be used in routes that don’t specify an http method (by either using app.get, def get, or the methods parameter to app.route).\n\n@app.get(\"/datie/\")\ndef _(d:parsed_date): return d\ndate_str = \"17th of May, 2024, 2p\"\nprint(cli.get(f'/datie/?d={date_str}').text)\n\n2024-05-17 14:00:00\n\n\nHandler functions can use date objects as parameter types. FastHTML uses dateutil.parser library to automatically parse a wide variety of date string formats into date objects.\n\n@app.get(\"/ua\")\nasync def _(user_agent:str): return user_agent\nprint(cli.get('/ua', headers={'User-Agent':'FastHTML'}).text)\n\nFastHTML\n\n\nHandler functions can access HTTP headers by using parameter names that match the header names. In this example, user_agent is used as a parameter name, which automatically captures the value of the ‘User-Agent’ header from the request.\nThe Client instance allows setting custom headers for test requests. Here, we set the ‘User-Agent’ header to ‘FastHTML’ in the test request.\n\n@app.get(\"/hxtest\")\ndef _(htmx): return htmx.request\nprint(cli.get('/hxtest', headers={'HX-Request':'1'}).text)\n\n@app.get(\"/hxtest2\")\ndef _(foo:HtmxHeaders, req): return foo.request\nprint(cli.get('/hxtest2', headers={'HX-Request':'1'}).text)\n\n1\n1\n\n\nHandler functions can access HTMX-specific headers using either the special htmx parameter name, or a parameter annotated with HtmxHeaders. Both approaches provide access to HTMX-related information.\nIn these examples, the htmx.request attribute returns the value of the ‘HX-Request’ header.\n\napp.chk = 'foo'\n@app.get(\"/app\")\ndef _(app): return app.chk\nprint(cli.get('/app').text)\n\nfoo\n\n\nHandler functions can access the FastHTML application instance using the special app parameter name. This allows handlers to access application-level attributes and methods.\nIn this example, we set a custom attribute chk on the application instance. The handler function then uses the app parameter to access this attribute and return its value.\n\n@app.get(\"/app2\")\ndef _(foo:FastHTML): return foo.chk,HttpHeader(\"mykey\", \"myval\")\nr = cli.get('/app2', **hxhdr)\nprint(r.text)\nprint(r.headers)\n\nfoo\nHeaders({'mykey': 'myval', 'content-length': '3', 'content-type': 'text/html; charset=utf-8'})\n\n\nHandler functions can access the FastHTML application instance using a parameter annotated with FastHTML. This allows handlers to access application-level attributes and methods, just like using the special app parameter name.\nHandlers can return tuples containing both content and HttpHeader objects. HttpHeader allows setting custom HTTP headers in the response.\nIn this example:\n\nWe define a handler that returns both the chk attribute from the application and a custom header.\nThe HttpHeader(\"mykey\", \"myval\") sets a custom header in the response.\nWe use the test client to make a request and examine both the response text and headers.\nThe response includes the custom header “mykey” along with standard headers like content-length and content-type.\n\n\n@app.get(\"/app3\")\ndef _(foo:FastHTML): return HtmxResponseHeaders(location=\"http://example.org\")\nr = cli.get('/app3')\nprint(r.headers)\n\nHeaders({'hx-location': 'http://example.org', 'content-length': '0', 'content-type': 'text/html; charset=utf-8'})\n\n\nHandler functions can return HtmxResponseHeaders objects to set HTMX-specific response headers. This is useful for HTMX-specific behaviors like client-side redirects.\nIn this example we define a handler that returns an HtmxResponseHeaders object with a location parameter, which sets the HX-Location header in the response. HTMX uses this for client-side redirects.\n\n@app.get(\"/app4\")\ndef _(foo:FastHTML): return Redirect(\"http://example.org\")\ncli.get('/app4', follow_redirects=False)\n\n<Response [303 See Other]>\n\n\nHandler functions can return Redirect objects to perform HTTP redirects. This is useful for redirecting users to different pages or external URLs.\nIn this example:\n\nWe define a handler that returns a Redirect object with the URL “http://example.org”.\nThe cli.get('/app4', follow_redirects=False) call simulates a GET request to the ‘/app4’ route without following redirects.\nThe response has a 303 See Other status code, indicating a redirect.\n\nThe follow_redirects=False parameter is used to prevent the test client from automatically following the redirect, allowing us to inspect the redirect response itself.\n\nRedirect.__response__\n\n<function fasthtml.core.Redirect.__response__(self, req)>\n\n\nThe Redirect class in FastHTML implements a __response__ method, which is a special method recognized by the framework. When a handler returns a Redirect object, FastHTML internally calls this __response__ method to replace the original response.\nThe __response__ method takes a req parameter, which represents the incoming request. This allows the method to access request information if needed when constructing the redirect response.\n\n@rt\ndef meta(): \n return ((Title('hi'),H1('hi')),\n (Meta(property='image'), Meta(property='site_name')))\n\nprint(cli.post('/meta').text)\n\n <!doctype html>\n <html>\n <head>\n <title>hi</title>\n <meta property=\"image\">\n <meta property=\"site_name\">\n <meta charset=\"utf-8\">\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1, viewport-fit=cover\">\n<script src=\"https://unpkg.com/htmx.org@next/dist/htmx.min.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/answerdotai/fasthtml-js@1.0.4/fasthtml.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/answerdotai/surreal@main/surreal.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/gnat/css-scope-inline@main/script.js\"></script><script>\n function sendmsg() {\n window.parent.postMessage({height: document.documentElement.offsetHeight}, '*');\n }\n window.onload = function() {\n sendmsg();\n document.body.addEventListener('htmx:afterSettle', sendmsg);\n document.body.addEventListener('htmx:wsAfterMessage', sendmsg);\n };</script> </head>\n <body>\n <h1>hi</h1>\n </body>\n </html>\n\n\n\nFastHTML automatically identifies elements typically placed in the <head> (like Title and Meta) and positions them accordingly, while other elements go in the <body>.\nIn this example: - (Title('hi'), H1('hi')) defines the title and main heading. The title is placed in the head, and the H1 in the body. - (Meta(property='image'), Meta(property='site_name')) defines two meta tags, which are both placed in the head.", + "crumbs": [ + "Home", + "Reference", + "Handling handlers" + ] + }, + { + "objectID": "ref/handlers.html#apirouter", + "href": "ref/handlers.html#apirouter", + "title": "Handling handlers", + "section": "APIRouter", + "text": "APIRouter\nAPIRouter is useful when you want to split your application routes across multiple .py files that are part of a single FastHTMl application. It accepts an optional prefix argument that will be applied to all routes within that instance of APIRouter.\nBelow we define several hypothetical product related routes in a products.py and then demonstrate how they can seamlessly be incorporated into a FastHTML app instance.\n\n# products.py\nar = APIRouter(prefix=\"/products\")\n\n@ar(\"/all\")\ndef all_products(req):\n return Div(\n \"Welcome to the Products Page! Click the button below to look at the details for product 42\",\n Div(\n Button(\n \"Details\",\n hx_get=req.url_for(\"details\", pid=42),\n hx_target=\"#products_list\",\n hx_swap=\"outerHTML\",\n ),\n ),\n id=\"products_list\",\n )\n\n\n@ar.get(\"/{pid}\", name=\"details\")\ndef details(pid: int):\n return f\"Here are the product details for ID: {pid}\"\n\nSince we specified the prefix=/products in our hypothetical products.py file, all routes defined in that file will be found under /products.\n\nprint(str(ar.rt_funcs.all_products))\nprint(str(ar.rt_funcs.details))\n\n/products/all\n/products/{pid}\n\n\n\n# main.py\n# from products import ar\n\napp, rt = fast_app()\nar.to_app(app)\n\n@rt\ndef index():\n return Div(\n \"Click me for a look at our products\",\n hx_get=ar.rt_funcs.all_products,\n hx_swap=\"outerHTML\",\n )\n\nNote how you can reference our python route functions via APIRouter.rt_funcs in your hx_{http_method} calls like normal.", + "crumbs": [ + "Home", + "Reference", + "Handling handlers" + ] + }, + { + "objectID": "ref/handlers.html#form-data-and-json-handling", + "href": "ref/handlers.html#form-data-and-json-handling", + "title": "Handling handlers", + "section": "Form Data and JSON Handling", + "text": "Form Data and JSON Handling\n\n@app.post('/profile/me')\ndef profile_update(username: str): return username\n\nprint(cli.post('/profile/me', data={'username' : 'Alexis'}).text)\nr = cli.post('/profile/me', data={})\nprint(r.text)\nr\n\n404 Not Found\n404 Not Found\n\n\n<Response [404 Not Found]>\n\n\nHandler functions can accept form data parameters, without needing to manually extract it from the request. In this example, username is expected to be sent as form data.\nIf required form data is missing, FastHTML automatically returns a 400 Bad Request response with an error message.\nThe data parameter in the cli.post() method simulates sending form data in the request.\n\n@app.post('/pet/dog')\ndef pet_dog(dogname: str = None): return dogname or 'unknown name'\nprint(cli.post('/pet/dog', data={}).text)\n\n404 Not Found\n\n\nHandlers can have optional form data parameters with default values. In this example, dogname is an optional parameter with a default value of None.\nHere, if the form data doesn’t include the dogname field, the function uses the default value. The function returns either the provided dogname or ‘unknown name’ if dogname is None.\n\n@dataclass\nclass Bodie: a:int;b:str\n\n@rt(\"/bodie/{nm}\")\ndef post(nm:str, data:Bodie):\n res = asdict(data)\n res['nm'] = nm\n return res\n\nprint(cli.post('/bodie/me', data=dict(a=1, b='foo', nm='me')).text)\n\n404 Not Found\n\n\nYou can use dataclasses to define structured form data. In this example, Bodie is a dataclass with a (int) and b (str) fields.\nFastHTML automatically converts the incoming form data to a Bodie instance where attribute names match parameter names. Other form data elements are matched with parameters with the same names (in this case, nm).\nHandler functions can return dictionaries, which FastHTML automatically JSON-encodes.\n\n@app.post(\"/bodied/\")\ndef bodied(data:dict): return data\n\nd = dict(a=1, b='foo')\nprint(cli.post('/bodied/', data=d).text)\n\n404 Not Found\n\n\ndict parameters capture all form data as a dictionary. In this example, the data parameter is annotated with dict, so FastHTML automatically converts all incoming form data into a dictionary.\nNote that when form data is converted to a dictionary, all values become strings, even if they were originally numbers. This is why the ‘a’ key in the response has a string value “1” instead of the integer 1.\n\nnt = namedtuple('Bodient', ['a','b'])\n\n@app.post(\"/bodient/\")\ndef bodient(data:nt): return asdict(data)\nprint(cli.post('/bodient/', data=d).text)\n\n404 Not Found\n\n\nHandler functions can use named tuples to define structured form data. In this example, Bodient is a named tuple with a and b fields.\nFastHTML automatically converts the incoming form data to a Bodient instance where field names match parameter names. As with the previous example, all form data values are converted to strings in the process.\n\nclass BodieTD(TypedDict): a:int;b:str='foo'\n\n@app.post(\"/bodietd/\")\ndef bodient(data:BodieTD): return data\nprint(cli.post('/bodietd/', data=d).text)\n\n404 Not Found\n\n\nYou can use TypedDict to define structured form data with type hints. In this example, BodieTD is a TypedDict with a (int) and b (str) fields, where b has a default value of ‘foo’.\nFastHTML automatically converts the incoming form data to a BodieTD instance where keys match the defined fields. Unlike with regular dictionaries or named tuples, FastHTML respects the type hints in TypedDict, converting values to the specified types when possible (e.g., converting ‘1’ to the integer 1 for the ‘a’ field).\n\nclass Bodie2:\n a:int|None; b:str\n def __init__(self, a, b='foo'): store_attr()\n\n@app.post(\"/bodie2/\")\ndef bodie(d:Bodie2): return f\"a: {d.a}; b: {d.b}\"\nprint(cli.post('/bodie2/', data={'a':1}).text)\n\n404 Not Found\n\n\nCustom classes can be used to define structured form data. Here, Bodie2 is a custom class with a (int|None) and b (str) attributes, where b has a default value of ‘foo’. The store_attr() function (from fastcore) automatically assigns constructor parameters to instance attributes.\nFastHTML automatically converts the incoming form data to a Bodie2 instance, matching form fields to constructor parameters. It respects type hints and default values.\n\n@app.post(\"/b\")\ndef index(it: Bodie): return Titled(\"It worked!\", P(f\"{it.a}, {it.b}\"))\n\ns = json.dumps({\"b\": \"Lorem\", \"a\": 15})\nprint(cli.post('/b', headers={\"Content-Type\": \"application/json\", 'hx-request':\"1\"}, data=s).text)\n\n404 Not Found\n\n\nHandler functions can accept JSON data as input, which is automatically parsed into the specified type. In this example, it is of type Bodie, and FastHTML converts the incoming JSON data to a Bodie instance.\nThe Titled component is used to create a page with a title and main content. It automatically generates an <h1> with the provided title, wraps the content in a <main> tag with a “container” class, and adds a title to the head.\nWhen making a request with JSON data: - Set the “Content-Type” header to “application/json” - Provide the JSON data as a string in the data parameter of the request", + "crumbs": [ + "Home", + "Reference", + "Handling handlers" + ] + }, + { + "objectID": "ref/handlers.html#cookies-sessions-file-uploads-and-more", + "href": "ref/handlers.html#cookies-sessions-file-uploads-and-more", + "title": "Handling handlers", + "section": "Cookies, Sessions, File Uploads, and more", + "text": "Cookies, Sessions, File Uploads, and more\n\n@rt(\"/setcookie\")\ndef get(): return cookie('now', datetime.now())\n\n@rt(\"/getcookie\")\ndef get(now:parsed_date): return f'Cookie was set at time {now.time()}'\n\nprint(cli.get('/setcookie').text)\ntime.sleep(0.01)\ncli.get('/getcookie').text\n\n404 Not Found\n\n\n'404 Not Found'\n\n\nHandler functions can set and retrieve cookies. In this example:\n\nThe /setcookie route sets a cookie named ‘now’ with the current datetime.\nThe /getcookie route retrieves the ‘now’ cookie and returns its value.\n\nThe cookie() function is used to create a cookie response. FastHTML automatically converts the datetime object to a string when setting the cookie, and parses it back to a date object when retrieving it.\n\ncookie('now', datetime.now())\n\nHttpHeader(k='set-cookie', v='now=\"2024-12-04 13:45:24.154187\"; Path=/; SameSite=lax')\n\n\nThe cookie() function returns an HttpHeader object with the ‘set-cookie’ key. You can return it in a tuple along with FT elements, along with anything else FastHTML supports in responses.\n\napp = FastHTML(secret_key='soopersecret')\ncli = Client(app)\nrt = app.route\n\n\n@rt(\"/setsess\")\ndef get(sess, foo:str=''):\n now = datetime.now()\n sess['auth'] = str(now)\n return f'Set to {now}'\n\n@rt(\"/getsess\")\ndef get(sess): return f'Session time: {sess[\"auth\"]}'\n\nprint(cli.get('/setsess').text)\ntime.sleep(0.01)\n\ncli.get('/getsess').text\n\nSet to 2024-12-04 13:45:24.159764\n\n\n'Session time: 2024-12-04 13:45:24.159764'\n\n\nSessions store and retrieve data across requests. To use sessions, you should to initialize the FastHTML application with a secret_key. This is used to cryptographically sign the cookie used by the session.\nThe sess parameter in handler functions provides access to the session data. You can set and get session variables using dictionary-style access.\n\n@rt(\"/upload\")\nasync def post(uf:UploadFile): return (await uf.read()).decode()\n\nwith open('../../CHANGELOG.md', 'rb') as f:\n print(cli.post('/upload', files={'uf':f}, data={'msg':'Hello'}).text[:15])\n\n# Release notes\n\n\nHandler functions can accept file uploads using Starlette’s UploadFile type. In this example:\n\nThe /upload route accepts a file upload named uf.\nThe UploadFile object provides an asynchronous read() method to access the file contents.\nWe use await to read the file content asynchronously and decode it to a string.\n\nWe added async to the handler function because it uses await to read the file content asynchronously. In Python, any function that uses await must be declared as async. This allows the function to be run asynchronously, potentially improving performance by not blocking other operations while waiting for the file to be read.\n\napp.static_route('.md', static_path='../..')\nprint(cli.get('/README.md').text[:10])\n\n# FastHTML\n\n\nThe static_route method of the FastHTML application allows serving static files with specified extensions from a given directory. In this example:\n\n.md files are served from the ../.. directory (two levels up from the current directory).\nAccessing /README.md returns the contents of the README.md file from that directory.\n\n\nhelp(app.static_route_exts)\n\nHelp on method static_route_exts in module fasthtml.core:\n\nstatic_route_exts(prefix='/', static_path='.', exts='static') method of fasthtml.core.FastHTML instance\n Add a static route at URL path `prefix` with files from `static_path` and `exts` defined by `reg_re_param()`\n\n\n\n\napp.static_route_exts()\nprint(cli.get('/README.txt').text[:50])\n\n404 Not Found\n\n\nThe static_route_exts method of the FastHTML application allows serving static files with specified extensions from a given directory. By default:\n\nIt serves files from the current directory (‘.’).\nIt uses the ‘static’ regex, which includes common static file extensions like ‘ico’, ‘gif’, ‘jpg’, ‘css’, ‘js’, etc.\nThe URL prefix is set to ‘/’.\n\nThe ‘static’ regex is defined by FastHTML using this code:\nreg_re_param(\"static\", \"ico|gif|jpg|jpeg|webm|css|js|woff|png|svg|mp4|webp|ttf|otf|eot|woff2|txt|html|map\")\n\n@rt(\"/form-submit/{list_id}\")\ndef options(list_id: str):\n headers = {\n 'Access-Control-Allow-Origin': '*',\n 'Access-Control-Allow-Methods': 'POST',\n 'Access-Control-Allow-Headers': '*',\n }\n return Response(status_code=200, headers=headers)\n\nprint(cli.options('/form-submit/2').headers)\n\nHeaders({'access-control-allow-origin': '*', 'access-control-allow-methods': 'POST', 'access-control-allow-headers': '*', 'content-length': '0', 'set-cookie': 'session_=eyJhdXRoIjogIjIwMjQtMTItMDQgMTM6NDU6MjQuMTU5NzY0In0=.Z1DNdA.GV-NVoOnJeambm9_uE3crhoGH34; path=/; Max-Age=31536000; httponly; samesite=lax'})\n\n\nFastHTML handlers can handle OPTIONS requests and set custom headers. In this example:\n\nThe /form-submit/{list_id} route handles OPTIONS requests.\nCustom headers are set to allow cross-origin requests (CORS).\nThe function returns a Starlette Response object with a 200 status code and the custom headers.\n\nYou can return any Starlette Response type from a handler function, giving you full control over the response when needed.\n\ndef _not_found(req, exc): return Div('nope')\n\napp = FastHTML(exception_handlers={404:_not_found})\ncli = Client(app)\nrt = app.route\n\nr = cli.get('/')\nprint(r.text)\n\n <!doctype html>\n <html>\n <head>\n <title>FastHTML page</title>\n <meta charset=\"utf-8\">\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1, viewport-fit=cover\">\n<script src=\"https://unpkg.com/htmx.org@next/dist/htmx.min.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/answerdotai/fasthtml-js@1.0.4/fasthtml.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/answerdotai/surreal@main/surreal.js\"></script><script src=\"https://cdn.jsdelivr.net/gh/gnat/css-scope-inline@main/script.js\"></script><script>\n function sendmsg() {\n window.parent.postMessage({height: document.documentElement.offsetHeight}, '*');\n }\n window.onload = function() {\n sendmsg();\n document.body.addEventListener('htmx:afterSettle', sendmsg);\n document.body.addEventListener('htmx:wsAfterMessage', sendmsg);\n };</script> </head>\n <body>\n <div>nope</div>\n </body>\n </html>\n\n\n\nFastHTML allows you to define custom exception handlers – in this case, a custom 404 (Not Found) handler function _not_found, which returns a Div component with the text ‘nope’.", + "crumbs": [ + "Home", + "Reference", + "Handling handlers" + ] + } +] \ No newline at end of file diff --git a/site_libs/bootstrap/bootstrap-f060f2c01c87989ca9870cd8c49a312f.min.css b/site_libs/bootstrap/bootstrap-f060f2c01c87989ca9870cd8c49a312f.min.css new file mode 100644 index 00000000..ef987237 --- /dev/null +++ b/site_libs/bootstrap/bootstrap-f060f2c01c87989ca9870cd8c49a312f.min.css @@ -0,0 +1,12 @@ +/*! + * Bootstrap v5.3.1 (https://getbootstrap.com/) + * Copyright 2011-2023 The Bootstrap Authors + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */@import"https://fonts.googleapis.com/css2?family=Source+Sans+Pro:wght@300;400;700&display=swap";:root,[data-bs-theme=light]{--bs-blue: #2780e3;--bs-indigo: #6610f2;--bs-purple: #613d7c;--bs-pink: #e83e8c;--bs-red: #ff0039;--bs-orange: #f0ad4e;--bs-yellow: #ff7518;--bs-green: #3fb618;--bs-teal: #20c997;--bs-cyan: #9954bb;--bs-black: #000;--bs-white: #fff;--bs-gray: #6c757d;--bs-gray-dark: #343a40;--bs-gray-100: #f8f9fa;--bs-gray-200: #e9ecef;--bs-gray-300: #dee2e6;--bs-gray-400: #ced4da;--bs-gray-500: #adb5bd;--bs-gray-600: #6c757d;--bs-gray-700: #495057;--bs-gray-800: #343a40;--bs-gray-900: #212529;--bs-default: #343a40;--bs-primary: #E8E8FC;--bs-secondary: #343a40;--bs-success: #3fb618;--bs-info: #9954bb;--bs-warning: #ff7518;--bs-danger: #ff0039;--bs-light: #f8f9fa;--bs-dark: #343a40;--bs-default-rgb: 52, 58, 64;--bs-primary-rgb: 232, 232, 252;--bs-secondary-rgb: 52, 58, 64;--bs-success-rgb: 63, 182, 24;--bs-info-rgb: 153, 84, 187;--bs-warning-rgb: 255, 117, 24;--bs-danger-rgb: 255, 0, 57;--bs-light-rgb: 248, 249, 250;--bs-dark-rgb: 52, 58, 64;--bs-primary-text-emphasis: #5d5d65;--bs-secondary-text-emphasis: #15171a;--bs-success-text-emphasis: #19490a;--bs-info-text-emphasis: #3d224b;--bs-warning-text-emphasis: #662f0a;--bs-danger-text-emphasis: #660017;--bs-light-text-emphasis: #495057;--bs-dark-text-emphasis: #495057;--bs-primary-bg-subtle: #fafafe;--bs-secondary-bg-subtle: #d6d8d9;--bs-success-bg-subtle: #d9f0d1;--bs-info-bg-subtle: #ebddf1;--bs-warning-bg-subtle: #ffe3d1;--bs-danger-bg-subtle: #ffccd7;--bs-light-bg-subtle: #fcfcfd;--bs-dark-bg-subtle: #ced4da;--bs-primary-border-subtle: #f6f6fe;--bs-secondary-border-subtle: #aeb0b3;--bs-success-border-subtle: #b2e2a3;--bs-info-border-subtle: #d6bbe4;--bs-warning-border-subtle: #ffc8a3;--bs-danger-border-subtle: #ff99b0;--bs-light-border-subtle: #e9ecef;--bs-dark-border-subtle: #adb5bd;--bs-white-rgb: 255, 255, 255;--bs-black-rgb: 0, 0, 0;--bs-font-sans-serif: "Source Sans Pro", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";--bs-font-monospace: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--bs-gradient: linear-gradient(180deg, rgba(255, 255, 255, 0.15), rgba(255, 255, 255, 0));--bs-root-font-size: 18px;--bs-body-font-family: "Source Sans Pro", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";--bs-body-font-size:1rem;--bs-body-font-weight: 400;--bs-body-line-height: 1.5;--bs-body-color: #000000;--bs-body-color-rgb: 0, 0, 0;--bs-body-bg: #FFFFFF;--bs-body-bg-rgb: 255, 255, 255;--bs-emphasis-color: #000;--bs-emphasis-color-rgb: 0, 0, 0;--bs-secondary-color: rgba(0, 0, 0, 0.75);--bs-secondary-color-rgb: 0, 0, 0;--bs-secondary-bg: #e9ecef;--bs-secondary-bg-rgb: 233, 236, 239;--bs-tertiary-color: rgba(0, 0, 0, 0.5);--bs-tertiary-color-rgb: 0, 0, 0;--bs-tertiary-bg: #f8f9fa;--bs-tertiary-bg-rgb: 248, 249, 250;--bs-heading-color: inherit;--bs-link-color: #4040BF;--bs-link-color-rgb: 64, 64, 191;--bs-link-decoration: underline;--bs-link-hover-color: #333399;--bs-link-hover-color-rgb: 51, 51, 153;--bs-code-color: #7d12ba;--bs-highlight-bg: #ffe3d1;--bs-border-width: 1px;--bs-border-style: solid;--bs-border-color: #dee2e6;--bs-border-color-translucent: rgba(0, 0, 0, 0.175);--bs-border-radius: 0.25rem;--bs-border-radius-sm: 0.2em;--bs-border-radius-lg: 0.5rem;--bs-border-radius-xl: 1rem;--bs-border-radius-xxl: 2rem;--bs-border-radius-2xl: var(--bs-border-radius-xxl);--bs-border-radius-pill: 50rem;--bs-box-shadow: 0 0.5rem 1rem rgba(0, 0, 0, 0.15);--bs-box-shadow-sm: 0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);--bs-box-shadow-lg: 0 1rem 3rem rgba(0, 0, 0, 0.175);--bs-box-shadow-inset: inset 0 1px 2px rgba(0, 0, 0, 0.075);--bs-focus-ring-width: 0.25rem;--bs-focus-ring-opacity: 0.25;--bs-focus-ring-color: rgba(232, 232, 252, 0.25);--bs-form-valid-color: #3fb618;--bs-form-valid-border-color: #3fb618;--bs-form-invalid-color: #ff0039;--bs-form-invalid-border-color: #ff0039}[data-bs-theme=dark]{color-scheme:dark;--bs-body-color: #dee2e6;--bs-body-color-rgb: 222, 226, 230;--bs-body-bg: #212529;--bs-body-bg-rgb: 33, 37, 41;--bs-emphasis-color: #fff;--bs-emphasis-color-rgb: 255, 255, 255;--bs-secondary-color: rgba(222, 226, 230, 0.75);--bs-secondary-color-rgb: 222, 226, 230;--bs-secondary-bg: #343a40;--bs-secondary-bg-rgb: 52, 58, 64;--bs-tertiary-color: rgba(222, 226, 230, 0.5);--bs-tertiary-color-rgb: 222, 226, 230;--bs-tertiary-bg: #2b3035;--bs-tertiary-bg-rgb: 43, 48, 53;--bs-primary-text-emphasis: #f1f1fd;--bs-secondary-text-emphasis: #85898c;--bs-success-text-emphasis: #8cd374;--bs-info-text-emphasis: #c298d6;--bs-warning-text-emphasis: #ffac74;--bs-danger-text-emphasis: #ff6688;--bs-light-text-emphasis: #f8f9fa;--bs-dark-text-emphasis: #dee2e6;--bs-primary-bg-subtle: #2e2e32;--bs-secondary-bg-subtle: #0a0c0d;--bs-success-bg-subtle: #0d2405;--bs-info-bg-subtle: #1f1125;--bs-warning-bg-subtle: #331705;--bs-danger-bg-subtle: #33000b;--bs-light-bg-subtle: #343a40;--bs-dark-bg-subtle: #1a1d20;--bs-primary-border-subtle: #8b8b97;--bs-secondary-border-subtle: #1f2326;--bs-success-border-subtle: #266d0e;--bs-info-border-subtle: #5c3270;--bs-warning-border-subtle: #99460e;--bs-danger-border-subtle: #990022;--bs-light-border-subtle: #495057;--bs-dark-border-subtle: #343a40;--bs-heading-color: inherit;--bs-link-color: #f1f1fd;--bs-link-hover-color: #f4f4fd;--bs-link-color-rgb: 241, 241, 253;--bs-link-hover-color-rgb: 244, 244, 253;--bs-code-color: white;--bs-border-color: #495057;--bs-border-color-translucent: rgba(255, 255, 255, 0.15);--bs-form-valid-color: #8cd374;--bs-form-valid-border-color: #8cd374;--bs-form-invalid-color: #ff6688;--bs-form-invalid-border-color: #ff6688}*,*::before,*::after{box-sizing:border-box}:root{font-size:var(--bs-root-font-size)}body{margin:0;font-family:var(--bs-body-font-family);font-size:var(--bs-body-font-size);font-weight:var(--bs-body-font-weight);line-height:var(--bs-body-line-height);color:var(--bs-body-color);text-align:var(--bs-body-text-align);background-color:var(--bs-body-bg);-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:rgba(0,0,0,0)}hr{margin:1rem 0;color:inherit;border:0;border-top:1px solid;opacity:.25}h6,.h6,h5,.h5,h4,.h4,h3,.h3,h2,.h2,h1,.h1{margin-top:0;margin-bottom:.5rem;font-weight:400;line-height:1.2;color:var(--bs-heading-color)}h1,.h1{font-size:calc(1.325rem + 0.9vw)}@media(min-width: 1200px){h1,.h1{font-size:2rem}}h2,.h2{font-size:calc(1.29rem + 0.48vw)}@media(min-width: 1200px){h2,.h2{font-size:1.65rem}}h3,.h3{font-size:calc(1.27rem + 0.24vw)}@media(min-width: 1200px){h3,.h3{font-size:1.45rem}}h4,.h4{font-size:1.25rem}h5,.h5{font-size:1.1rem}h6,.h6{font-size:1rem}p{margin-top:0;margin-bottom:1rem}abbr[title]{text-decoration:underline dotted;-webkit-text-decoration:underline dotted;-moz-text-decoration:underline dotted;-ms-text-decoration:underline dotted;-o-text-decoration:underline dotted;cursor:help;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}ol,ul{padding-left:2rem}ol,ul,dl{margin-top:0;margin-bottom:1rem}ol ol,ul ul,ol ul,ul ol{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem;padding:.625rem 1.25rem;border-left:.25rem solid #e9ecef}blockquote p:last-child,blockquote ul:last-child,blockquote ol:last-child{margin-bottom:0}b,strong{font-weight:bolder}small,.small{font-size:0.875em}mark,.mark{padding:.1875em;background-color:var(--bs-highlight-bg)}sub,sup{position:relative;font-size:0.75em;line-height:0;vertical-align:baseline}sub{bottom:-0.25em}sup{top:-0.5em}a{color:rgba(var(--bs-link-color-rgb), var(--bs-link-opacity, 1));text-decoration:underline;-webkit-text-decoration:underline;-moz-text-decoration:underline;-ms-text-decoration:underline;-o-text-decoration:underline}a:hover{--bs-link-color-rgb: var(--bs-link-hover-color-rgb)}a:not([href]):not([class]),a:not([href]):not([class]):hover{color:inherit;text-decoration:none}pre,code,kbd,samp{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:1em}pre{display:block;margin-top:0;margin-bottom:1rem;overflow:auto;font-size:0.875em;color:#000;background-color:#f8f9fa;line-height:1.5;padding:.5rem;border:1px solid var(--bs-border-color, #dee2e6)}pre code{background-color:rgba(0,0,0,0);font-size:inherit;color:inherit;word-break:normal}code{font-size:0.875em;color:var(--bs-code-color);background-color:#f8f9fa;padding:.125rem .25rem;word-wrap:break-word}a>code{color:inherit}kbd{padding:.4rem .4rem;font-size:0.875em;color:#fff;background-color:#000}kbd kbd{padding:0;font-size:1em}figure{margin:0 0 1rem}img,svg{vertical-align:middle}table{caption-side:bottom;border-collapse:collapse}caption{padding-top:.5rem;padding-bottom:.5rem;color:rgba(0,0,0,.75);text-align:left}th{text-align:inherit;text-align:-webkit-match-parent}thead,tbody,tfoot,tr,td,th{border-color:inherit;border-style:solid;border-width:0}label{display:inline-block}button{border-radius:0}button:focus:not(:focus-visible){outline:0}input,button,select,optgroup,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,select{text-transform:none}[role=button]{cursor:pointer}select{word-wrap:normal}select:disabled{opacity:1}[list]:not([type=date]):not([type=datetime-local]):not([type=month]):not([type=week]):not([type=time])::-webkit-calendar-picker-indicator{display:none !important}button,[type=button],[type=reset],[type=submit]{-webkit-appearance:button}button:not(:disabled),[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled){cursor:pointer}::-moz-focus-inner{padding:0;border-style:none}textarea{resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{float:left;width:100%;padding:0;margin-bottom:.5rem;font-size:calc(1.275rem + 0.3vw);line-height:inherit}@media(min-width: 1200px){legend{font-size:1.5rem}}legend+*{clear:left}::-webkit-datetime-edit-fields-wrapper,::-webkit-datetime-edit-text,::-webkit-datetime-edit-minute,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-year-field{padding:0}::-webkit-inner-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-color-swatch-wrapper{padding:0}::file-selector-button{font:inherit;-webkit-appearance:button}output{display:inline-block}iframe{border:0}summary{display:list-item;cursor:pointer}progress{vertical-align:baseline}[hidden]{display:none !important}.lead{font-size:1.25rem;font-weight:300}.display-1{font-size:calc(1.625rem + 4.5vw);font-weight:300;line-height:1.2}@media(min-width: 1200px){.display-1{font-size:5rem}}.display-2{font-size:calc(1.575rem + 3.9vw);font-weight:300;line-height:1.2}@media(min-width: 1200px){.display-2{font-size:4.5rem}}.display-3{font-size:calc(1.525rem + 3.3vw);font-weight:300;line-height:1.2}@media(min-width: 1200px){.display-3{font-size:4rem}}.display-4{font-size:calc(1.475rem + 2.7vw);font-weight:300;line-height:1.2}@media(min-width: 1200px){.display-4{font-size:3.5rem}}.display-5{font-size:calc(1.425rem + 2.1vw);font-weight:300;line-height:1.2}@media(min-width: 1200px){.display-5{font-size:3rem}}.display-6{font-size:calc(1.375rem + 1.5vw);font-weight:300;line-height:1.2}@media(min-width: 1200px){.display-6{font-size:2.5rem}}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:.5rem}.initialism{font-size:0.875em;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote>:last-child{margin-bottom:0}.blockquote-footer{margin-top:-1rem;margin-bottom:1rem;font-size:0.875em;color:#6c757d}.blockquote-footer::before{content:"— "}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:.25rem;background-color:#fff;border:1px solid #dee2e6;max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:0.875em;color:rgba(0,0,0,.75)}.container,.container-fluid,.container-xxl,.container-xl,.container-lg,.container-md,.container-sm{--bs-gutter-x: 1.5rem;--bs-gutter-y: 0;width:100%;padding-right:calc(var(--bs-gutter-x)*.5);padding-left:calc(var(--bs-gutter-x)*.5);margin-right:auto;margin-left:auto}@media(min-width: 576px){.container-sm,.container{max-width:540px}}@media(min-width: 768px){.container-md,.container-sm,.container{max-width:720px}}@media(min-width: 992px){.container-lg,.container-md,.container-sm,.container{max-width:960px}}@media(min-width: 1200px){.container-xl,.container-lg,.container-md,.container-sm,.container{max-width:1140px}}@media(min-width: 1400px){.container-xxl,.container-xl,.container-lg,.container-md,.container-sm,.container{max-width:1320px}}:root{--bs-breakpoint-xs: 0;--bs-breakpoint-sm: 576px;--bs-breakpoint-md: 768px;--bs-breakpoint-lg: 992px;--bs-breakpoint-xl: 1200px;--bs-breakpoint-xxl: 1400px}.grid{display:grid;grid-template-rows:repeat(var(--bs-rows, 1), 1fr);grid-template-columns:repeat(var(--bs-columns, 12), 1fr);gap:var(--bs-gap, 1.5rem)}.grid .g-col-1{grid-column:auto/span 1}.grid .g-col-2{grid-column:auto/span 2}.grid .g-col-3{grid-column:auto/span 3}.grid .g-col-4{grid-column:auto/span 4}.grid .g-col-5{grid-column:auto/span 5}.grid .g-col-6{grid-column:auto/span 6}.grid .g-col-7{grid-column:auto/span 7}.grid .g-col-8{grid-column:auto/span 8}.grid .g-col-9{grid-column:auto/span 9}.grid .g-col-10{grid-column:auto/span 10}.grid .g-col-11{grid-column:auto/span 11}.grid .g-col-12{grid-column:auto/span 12}.grid .g-start-1{grid-column-start:1}.grid .g-start-2{grid-column-start:2}.grid .g-start-3{grid-column-start:3}.grid .g-start-4{grid-column-start:4}.grid .g-start-5{grid-column-start:5}.grid .g-start-6{grid-column-start:6}.grid .g-start-7{grid-column-start:7}.grid .g-start-8{grid-column-start:8}.grid .g-start-9{grid-column-start:9}.grid .g-start-10{grid-column-start:10}.grid .g-start-11{grid-column-start:11}@media(min-width: 576px){.grid .g-col-sm-1{grid-column:auto/span 1}.grid .g-col-sm-2{grid-column:auto/span 2}.grid .g-col-sm-3{grid-column:auto/span 3}.grid .g-col-sm-4{grid-column:auto/span 4}.grid .g-col-sm-5{grid-column:auto/span 5}.grid .g-col-sm-6{grid-column:auto/span 6}.grid .g-col-sm-7{grid-column:auto/span 7}.grid .g-col-sm-8{grid-column:auto/span 8}.grid .g-col-sm-9{grid-column:auto/span 9}.grid .g-col-sm-10{grid-column:auto/span 10}.grid .g-col-sm-11{grid-column:auto/span 11}.grid .g-col-sm-12{grid-column:auto/span 12}.grid .g-start-sm-1{grid-column-start:1}.grid .g-start-sm-2{grid-column-start:2}.grid .g-start-sm-3{grid-column-start:3}.grid .g-start-sm-4{grid-column-start:4}.grid .g-start-sm-5{grid-column-start:5}.grid .g-start-sm-6{grid-column-start:6}.grid .g-start-sm-7{grid-column-start:7}.grid .g-start-sm-8{grid-column-start:8}.grid .g-start-sm-9{grid-column-start:9}.grid .g-start-sm-10{grid-column-start:10}.grid .g-start-sm-11{grid-column-start:11}}@media(min-width: 768px){.grid .g-col-md-1{grid-column:auto/span 1}.grid .g-col-md-2{grid-column:auto/span 2}.grid .g-col-md-3{grid-column:auto/span 3}.grid .g-col-md-4{grid-column:auto/span 4}.grid .g-col-md-5{grid-column:auto/span 5}.grid .g-col-md-6{grid-column:auto/span 6}.grid .g-col-md-7{grid-column:auto/span 7}.grid .g-col-md-8{grid-column:auto/span 8}.grid .g-col-md-9{grid-column:auto/span 9}.grid .g-col-md-10{grid-column:auto/span 10}.grid .g-col-md-11{grid-column:auto/span 11}.grid .g-col-md-12{grid-column:auto/span 12}.grid .g-start-md-1{grid-column-start:1}.grid .g-start-md-2{grid-column-start:2}.grid .g-start-md-3{grid-column-start:3}.grid .g-start-md-4{grid-column-start:4}.grid .g-start-md-5{grid-column-start:5}.grid .g-start-md-6{grid-column-start:6}.grid .g-start-md-7{grid-column-start:7}.grid .g-start-md-8{grid-column-start:8}.grid .g-start-md-9{grid-column-start:9}.grid .g-start-md-10{grid-column-start:10}.grid .g-start-md-11{grid-column-start:11}}@media(min-width: 992px){.grid .g-col-lg-1{grid-column:auto/span 1}.grid .g-col-lg-2{grid-column:auto/span 2}.grid .g-col-lg-3{grid-column:auto/span 3}.grid .g-col-lg-4{grid-column:auto/span 4}.grid .g-col-lg-5{grid-column:auto/span 5}.grid .g-col-lg-6{grid-column:auto/span 6}.grid .g-col-lg-7{grid-column:auto/span 7}.grid .g-col-lg-8{grid-column:auto/span 8}.grid .g-col-lg-9{grid-column:auto/span 9}.grid .g-col-lg-10{grid-column:auto/span 10}.grid .g-col-lg-11{grid-column:auto/span 11}.grid .g-col-lg-12{grid-column:auto/span 12}.grid .g-start-lg-1{grid-column-start:1}.grid .g-start-lg-2{grid-column-start:2}.grid .g-start-lg-3{grid-column-start:3}.grid .g-start-lg-4{grid-column-start:4}.grid .g-start-lg-5{grid-column-start:5}.grid .g-start-lg-6{grid-column-start:6}.grid .g-start-lg-7{grid-column-start:7}.grid .g-start-lg-8{grid-column-start:8}.grid .g-start-lg-9{grid-column-start:9}.grid .g-start-lg-10{grid-column-start:10}.grid .g-start-lg-11{grid-column-start:11}}@media(min-width: 1200px){.grid .g-col-xl-1{grid-column:auto/span 1}.grid .g-col-xl-2{grid-column:auto/span 2}.grid .g-col-xl-3{grid-column:auto/span 3}.grid .g-col-xl-4{grid-column:auto/span 4}.grid .g-col-xl-5{grid-column:auto/span 5}.grid .g-col-xl-6{grid-column:auto/span 6}.grid .g-col-xl-7{grid-column:auto/span 7}.grid .g-col-xl-8{grid-column:auto/span 8}.grid .g-col-xl-9{grid-column:auto/span 9}.grid .g-col-xl-10{grid-column:auto/span 10}.grid .g-col-xl-11{grid-column:auto/span 11}.grid .g-col-xl-12{grid-column:auto/span 12}.grid .g-start-xl-1{grid-column-start:1}.grid .g-start-xl-2{grid-column-start:2}.grid .g-start-xl-3{grid-column-start:3}.grid .g-start-xl-4{grid-column-start:4}.grid .g-start-xl-5{grid-column-start:5}.grid .g-start-xl-6{grid-column-start:6}.grid .g-start-xl-7{grid-column-start:7}.grid .g-start-xl-8{grid-column-start:8}.grid .g-start-xl-9{grid-column-start:9}.grid .g-start-xl-10{grid-column-start:10}.grid .g-start-xl-11{grid-column-start:11}}@media(min-width: 1400px){.grid .g-col-xxl-1{grid-column:auto/span 1}.grid .g-col-xxl-2{grid-column:auto/span 2}.grid .g-col-xxl-3{grid-column:auto/span 3}.grid .g-col-xxl-4{grid-column:auto/span 4}.grid .g-col-xxl-5{grid-column:auto/span 5}.grid .g-col-xxl-6{grid-column:auto/span 6}.grid .g-col-xxl-7{grid-column:auto/span 7}.grid .g-col-xxl-8{grid-column:auto/span 8}.grid .g-col-xxl-9{grid-column:auto/span 9}.grid .g-col-xxl-10{grid-column:auto/span 10}.grid .g-col-xxl-11{grid-column:auto/span 11}.grid .g-col-xxl-12{grid-column:auto/span 12}.grid .g-start-xxl-1{grid-column-start:1}.grid .g-start-xxl-2{grid-column-start:2}.grid .g-start-xxl-3{grid-column-start:3}.grid .g-start-xxl-4{grid-column-start:4}.grid .g-start-xxl-5{grid-column-start:5}.grid .g-start-xxl-6{grid-column-start:6}.grid .g-start-xxl-7{grid-column-start:7}.grid .g-start-xxl-8{grid-column-start:8}.grid .g-start-xxl-9{grid-column-start:9}.grid .g-start-xxl-10{grid-column-start:10}.grid .g-start-xxl-11{grid-column-start:11}}.table{--bs-table-color-type: initial;--bs-table-bg-type: initial;--bs-table-color-state: initial;--bs-table-bg-state: initial;--bs-table-color: #000000;--bs-table-bg: #FFFFFF;--bs-table-border-color: #dee2e6;--bs-table-accent-bg: transparent;--bs-table-striped-color: #000000;--bs-table-striped-bg: rgba(0, 0, 0, 0.05);--bs-table-active-color: #000000;--bs-table-active-bg: rgba(0, 0, 0, 0.1);--bs-table-hover-color: #000000;--bs-table-hover-bg: rgba(0, 0, 0, 0.075);width:100%;margin-bottom:1rem;vertical-align:top;border-color:var(--bs-table-border-color)}.table>:not(caption)>*>*{padding:.5rem .5rem;color:var(--bs-table-color-state, var(--bs-table-color-type, var(--bs-table-color)));background-color:var(--bs-table-bg);border-bottom-width:1px;box-shadow:inset 0 0 0 9999px var(--bs-table-bg-state, var(--bs-table-bg-type, var(--bs-table-accent-bg)))}.table>tbody{vertical-align:inherit}.table>thead{vertical-align:bottom}.table-group-divider{border-top:calc(1px*2) solid gray}.caption-top{caption-side:top}.table-sm>:not(caption)>*>*{padding:.25rem .25rem}.table-bordered>:not(caption)>*{border-width:1px 0}.table-bordered>:not(caption)>*>*{border-width:0 1px}.table-borderless>:not(caption)>*>*{border-bottom-width:0}.table-borderless>:not(:first-child){border-top-width:0}.table-striped>tbody>tr:nth-of-type(odd)>*{--bs-table-color-type: var(--bs-table-striped-color);--bs-table-bg-type: var(--bs-table-striped-bg)}.table-striped-columns>:not(caption)>tr>:nth-child(even){--bs-table-color-type: var(--bs-table-striped-color);--bs-table-bg-type: var(--bs-table-striped-bg)}.table-active{--bs-table-color-state: var(--bs-table-active-color);--bs-table-bg-state: var(--bs-table-active-bg)}.table-hover>tbody>tr:hover>*{--bs-table-color-state: var(--bs-table-hover-color);--bs-table-bg-state: var(--bs-table-hover-bg)}.table-primary{--bs-table-color: #000;--bs-table-bg: #fafafe;--bs-table-border-color: #e1e1e5;--bs-table-striped-bg: #eeeef1;--bs-table-striped-color: #000;--bs-table-active-bg: #e1e1e5;--bs-table-active-color: #000;--bs-table-hover-bg: #e7e7eb;--bs-table-hover-color: #000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-secondary{--bs-table-color: #000;--bs-table-bg: #d6d8d9;--bs-table-border-color: #c1c2c3;--bs-table-striped-bg: #cbcdce;--bs-table-striped-color: #000;--bs-table-active-bg: #c1c2c3;--bs-table-active-color: #000;--bs-table-hover-bg: #c6c8c9;--bs-table-hover-color: #000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-success{--bs-table-color: #000;--bs-table-bg: #d9f0d1;--bs-table-border-color: #c3d8bc;--bs-table-striped-bg: #cee4c7;--bs-table-striped-color: #000;--bs-table-active-bg: #c3d8bc;--bs-table-active-color: #000;--bs-table-hover-bg: #c9dec1;--bs-table-hover-color: #000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-info{--bs-table-color: #000;--bs-table-bg: #ebddf1;--bs-table-border-color: #d4c7d9;--bs-table-striped-bg: #dfd2e5;--bs-table-striped-color: #000;--bs-table-active-bg: #d4c7d9;--bs-table-active-color: #000;--bs-table-hover-bg: #d9ccdf;--bs-table-hover-color: #000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-warning{--bs-table-color: #000;--bs-table-bg: #ffe3d1;--bs-table-border-color: #e6ccbc;--bs-table-striped-bg: #f2d8c7;--bs-table-striped-color: #000;--bs-table-active-bg: #e6ccbc;--bs-table-active-color: #000;--bs-table-hover-bg: #ecd2c1;--bs-table-hover-color: #000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-danger{--bs-table-color: #000;--bs-table-bg: #ffccd7;--bs-table-border-color: #e6b8c2;--bs-table-striped-bg: #f2c2cc;--bs-table-striped-color: #000;--bs-table-active-bg: #e6b8c2;--bs-table-active-color: #000;--bs-table-hover-bg: #ecbdc7;--bs-table-hover-color: #000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-light{--bs-table-color: #000;--bs-table-bg: #f8f9fa;--bs-table-border-color: #dfe0e1;--bs-table-striped-bg: #ecedee;--bs-table-striped-color: #000;--bs-table-active-bg: #dfe0e1;--bs-table-active-color: #000;--bs-table-hover-bg: #e5e6e7;--bs-table-hover-color: #000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-dark{--bs-table-color: #fff;--bs-table-bg: #343a40;--bs-table-border-color: #484e53;--bs-table-striped-bg: #3e444a;--bs-table-striped-color: #fff;--bs-table-active-bg: #484e53;--bs-table-active-color: #fff;--bs-table-hover-bg: #43494e;--bs-table-hover-color: #fff;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-responsive{overflow-x:auto;-webkit-overflow-scrolling:touch}@media(max-width: 575.98px){.table-responsive-sm{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media(max-width: 767.98px){.table-responsive-md{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media(max-width: 991.98px){.table-responsive-lg{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media(max-width: 1199.98px){.table-responsive-xl{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media(max-width: 1399.98px){.table-responsive-xxl{overflow-x:auto;-webkit-overflow-scrolling:touch}}.form-label,.shiny-input-container .control-label{margin-bottom:.5rem}.col-form-label{padding-top:calc(0.375rem + 1px);padding-bottom:calc(0.375rem + 1px);margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(0.5rem + 1px);padding-bottom:calc(0.5rem + 1px);font-size:1.25rem}.col-form-label-sm{padding-top:calc(0.25rem + 1px);padding-bottom:calc(0.25rem + 1px);font-size:0.875rem}.form-text{margin-top:.25rem;font-size:0.875em;color:rgba(0,0,0,.75)}.form-control{display:block;width:100%;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#000;appearance:none;-webkit-appearance:none;-moz-appearance:none;-ms-appearance:none;-o-appearance:none;background-color:#fff;background-clip:padding-box;border:1px solid #dee2e6;border-radius:0;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media(prefers-reduced-motion: reduce){.form-control{transition:none}}.form-control[type=file]{overflow:hidden}.form-control[type=file]:not(:disabled):not([readonly]){cursor:pointer}.form-control:focus{color:#000;background-color:#fff;border-color:#f4f4fe;outline:0;box-shadow:0 0 0 .25rem rgba(232,232,252,.25)}.form-control::-webkit-date-and-time-value{min-width:85px;height:1.5em;margin:0}.form-control::-webkit-datetime-edit{display:block;padding:0}.form-control::placeholder{color:rgba(0,0,0,.75);opacity:1}.form-control:disabled{background-color:#e9ecef;opacity:1}.form-control::file-selector-button{padding:.375rem .75rem;margin:-0.375rem -0.75rem;margin-inline-end:.75rem;color:#000;background-color:#f8f9fa;pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:1px;border-radius:0;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media(prefers-reduced-motion: reduce){.form-control::file-selector-button{transition:none}}.form-control:hover:not(:disabled):not([readonly])::file-selector-button{background-color:#e9ecef}.form-control-plaintext{display:block;width:100%;padding:.375rem 0;margin-bottom:0;line-height:1.5;color:#000;background-color:rgba(0,0,0,0);border:solid rgba(0,0,0,0);border-width:1px 0}.form-control-plaintext:focus{outline:0}.form-control-plaintext.form-control-sm,.form-control-plaintext.form-control-lg{padding-right:0;padding-left:0}.form-control-sm{min-height:calc(1.5em + 0.5rem + calc(1px * 2));padding:.25rem .5rem;font-size:0.875rem}.form-control-sm::file-selector-button{padding:.25rem .5rem;margin:-0.25rem -0.5rem;margin-inline-end:.5rem}.form-control-lg{min-height:calc(1.5em + 1rem + calc(1px * 2));padding:.5rem 1rem;font-size:1.25rem}.form-control-lg::file-selector-button{padding:.5rem 1rem;margin:-0.5rem -1rem;margin-inline-end:1rem}textarea.form-control{min-height:calc(1.5em + 0.75rem + calc(1px * 2))}textarea.form-control-sm{min-height:calc(1.5em + 0.5rem + calc(1px * 2))}textarea.form-control-lg{min-height:calc(1.5em + 1rem + calc(1px * 2))}.form-control-color{width:3rem;height:calc(1.5em + 0.75rem + calc(1px * 2));padding:.375rem}.form-control-color:not(:disabled):not([readonly]){cursor:pointer}.form-control-color::-moz-color-swatch{border:0 !important}.form-control-color::-webkit-color-swatch{border:0 !important}.form-control-color.form-control-sm{height:calc(1.5em + 0.5rem + calc(1px * 2))}.form-control-color.form-control-lg{height:calc(1.5em + 1rem + calc(1px * 2))}.form-select{--bs-form-select-bg-img: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23343a40' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e");display:block;width:100%;padding:.375rem 2.25rem .375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#000;appearance:none;-webkit-appearance:none;-moz-appearance:none;-ms-appearance:none;-o-appearance:none;background-color:#fff;background-image:var(--bs-form-select-bg-img),var(--bs-form-select-bg-icon, none);background-repeat:no-repeat;background-position:right .75rem center;background-size:16px 12px;border:1px solid #dee2e6;border-radius:0;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media(prefers-reduced-motion: reduce){.form-select{transition:none}}.form-select:focus{border-color:#f4f4fe;outline:0;box-shadow:0 0 0 .25rem rgba(232,232,252,.25)}.form-select[multiple],.form-select[size]:not([size="1"]){padding-right:.75rem;background-image:none}.form-select:disabled{background-color:#e9ecef}.form-select:-moz-focusring{color:rgba(0,0,0,0);text-shadow:0 0 0 #000}.form-select-sm{padding-top:.25rem;padding-bottom:.25rem;padding-left:.5rem;font-size:0.875rem}.form-select-lg{padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem;font-size:1.25rem}[data-bs-theme=dark] .form-select{--bs-form-select-bg-img: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23dee2e6' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e")}.form-check,.shiny-input-container .checkbox,.shiny-input-container .radio{display:block;min-height:1.5rem;padding-left:0;margin-bottom:.125rem}.form-check .form-check-input,.form-check .shiny-input-container .checkbox input,.form-check .shiny-input-container .radio input,.shiny-input-container .checkbox .form-check-input,.shiny-input-container .checkbox .shiny-input-container .checkbox input,.shiny-input-container .checkbox .shiny-input-container .radio input,.shiny-input-container .radio .form-check-input,.shiny-input-container .radio .shiny-input-container .checkbox input,.shiny-input-container .radio .shiny-input-container .radio input{float:left;margin-left:0}.form-check-reverse{padding-right:0;padding-left:0;text-align:right}.form-check-reverse .form-check-input{float:right;margin-right:0;margin-left:0}.form-check-input,.shiny-input-container .checkbox input,.shiny-input-container .checkbox-inline input,.shiny-input-container .radio input,.shiny-input-container .radio-inline input{--bs-form-check-bg: #FFFFFF;width:1em;height:1em;margin-top:.25em;vertical-align:top;appearance:none;-webkit-appearance:none;-moz-appearance:none;-ms-appearance:none;-o-appearance:none;background-color:var(--bs-form-check-bg);background-image:var(--bs-form-check-bg-image);background-repeat:no-repeat;background-position:center;background-size:contain;border:1px solid #dee2e6;print-color-adjust:exact}.form-check-input[type=radio],.shiny-input-container .checkbox input[type=radio],.shiny-input-container .checkbox-inline input[type=radio],.shiny-input-container .radio input[type=radio],.shiny-input-container .radio-inline input[type=radio]{border-radius:50%}.form-check-input:active,.shiny-input-container .checkbox input:active,.shiny-input-container .checkbox-inline input:active,.shiny-input-container .radio input:active,.shiny-input-container .radio-inline input:active{filter:brightness(90%)}.form-check-input:focus,.shiny-input-container .checkbox input:focus,.shiny-input-container .checkbox-inline input:focus,.shiny-input-container .radio input:focus,.shiny-input-container .radio-inline input:focus{border-color:#f4f4fe;outline:0;box-shadow:0 0 0 .25rem rgba(232,232,252,.25)}.form-check-input:checked,.shiny-input-container .checkbox input:checked,.shiny-input-container .checkbox-inline input:checked,.shiny-input-container .radio input:checked,.shiny-input-container .radio-inline input:checked{background-color:#e8e8fc;border-color:#e8e8fc}.form-check-input:checked[type=checkbox],.shiny-input-container .checkbox input:checked[type=checkbox],.shiny-input-container .checkbox-inline input:checked[type=checkbox],.shiny-input-container .radio input:checked[type=checkbox],.shiny-input-container .radio-inline input:checked[type=checkbox]{--bs-form-check-bg-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'%3e%3cpath fill='none' stroke='%23000' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='m6 10 3 3 6-6'/%3e%3c/svg%3e")}.form-check-input:checked[type=radio],.shiny-input-container .checkbox input:checked[type=radio],.shiny-input-container .checkbox-inline input:checked[type=radio],.shiny-input-container .radio input:checked[type=radio],.shiny-input-container .radio-inline input:checked[type=radio]{--bs-form-check-bg-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='2' fill='%23000'/%3e%3c/svg%3e")}.form-check-input[type=checkbox]:indeterminate,.shiny-input-container .checkbox input[type=checkbox]:indeterminate,.shiny-input-container .checkbox-inline input[type=checkbox]:indeterminate,.shiny-input-container .radio input[type=checkbox]:indeterminate,.shiny-input-container .radio-inline input[type=checkbox]:indeterminate{background-color:#e8e8fc;border-color:#e8e8fc;--bs-form-check-bg-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'%3e%3cpath fill='none' stroke='%23000' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='M6 10h8'/%3e%3c/svg%3e")}.form-check-input:disabled,.shiny-input-container .checkbox input:disabled,.shiny-input-container .checkbox-inline input:disabled,.shiny-input-container .radio input:disabled,.shiny-input-container .radio-inline input:disabled{pointer-events:none;filter:none;opacity:.5}.form-check-input[disabled]~.form-check-label,.form-check-input[disabled]~span,.form-check-input:disabled~.form-check-label,.form-check-input:disabled~span,.shiny-input-container .checkbox input[disabled]~.form-check-label,.shiny-input-container .checkbox input[disabled]~span,.shiny-input-container .checkbox input:disabled~.form-check-label,.shiny-input-container .checkbox input:disabled~span,.shiny-input-container .checkbox-inline input[disabled]~.form-check-label,.shiny-input-container .checkbox-inline input[disabled]~span,.shiny-input-container .checkbox-inline input:disabled~.form-check-label,.shiny-input-container .checkbox-inline input:disabled~span,.shiny-input-container .radio input[disabled]~.form-check-label,.shiny-input-container .radio input[disabled]~span,.shiny-input-container .radio input:disabled~.form-check-label,.shiny-input-container .radio input:disabled~span,.shiny-input-container .radio-inline input[disabled]~.form-check-label,.shiny-input-container .radio-inline input[disabled]~span,.shiny-input-container .radio-inline input:disabled~.form-check-label,.shiny-input-container .radio-inline input:disabled~span{cursor:default;opacity:.5}.form-check-label,.shiny-input-container .checkbox label,.shiny-input-container .checkbox-inline label,.shiny-input-container .radio label,.shiny-input-container .radio-inline label{cursor:pointer}.form-switch{padding-left:2.5em}.form-switch .form-check-input{--bs-form-switch-bg: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='rgba%280, 0, 0, 0.25%29'/%3e%3c/svg%3e");width:2em;margin-left:-2.5em;background-image:var(--bs-form-switch-bg);background-position:left center;transition:background-position .15s ease-in-out}@media(prefers-reduced-motion: reduce){.form-switch .form-check-input{transition:none}}.form-switch .form-check-input:focus{--bs-form-switch-bg: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23f4f4fe'/%3e%3c/svg%3e")}.form-switch .form-check-input:checked{background-position:right center;--bs-form-switch-bg: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23000'/%3e%3c/svg%3e")}.form-switch.form-check-reverse{padding-right:2.5em;padding-left:0}.form-switch.form-check-reverse .form-check-input{margin-right:-2.5em;margin-left:0}.form-check-inline{display:inline-block;margin-right:1rem}.btn-check{position:absolute;clip:rect(0, 0, 0, 0);pointer-events:none}.btn-check[disabled]+.btn,.btn-check:disabled+.btn{pointer-events:none;filter:none;opacity:.65}[data-bs-theme=dark] .form-switch .form-check-input:not(:checked):not(:focus){--bs-form-switch-bg: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='rgba%28255, 255, 255, 0.25%29'/%3e%3c/svg%3e")}.form-range{width:100%;height:1.5rem;padding:0;appearance:none;-webkit-appearance:none;-moz-appearance:none;-ms-appearance:none;-o-appearance:none;background-color:rgba(0,0,0,0)}.form-range:focus{outline:0}.form-range:focus::-webkit-slider-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .25rem rgba(232,232,252,.25)}.form-range:focus::-moz-range-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .25rem rgba(232,232,252,.25)}.form-range::-moz-focus-outer{border:0}.form-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-0.25rem;appearance:none;-webkit-appearance:none;-moz-appearance:none;-ms-appearance:none;-o-appearance:none;background-color:#e8e8fc;border:0;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media(prefers-reduced-motion: reduce){.form-range::-webkit-slider-thumb{transition:none}}.form-range::-webkit-slider-thumb:active{background-color:#f8f8fe}.form-range::-webkit-slider-runnable-track{width:100%;height:.5rem;color:rgba(0,0,0,0);cursor:pointer;background-color:#f8f9fa;border-color:rgba(0,0,0,0)}.form-range::-moz-range-thumb{width:1rem;height:1rem;appearance:none;-webkit-appearance:none;-moz-appearance:none;-ms-appearance:none;-o-appearance:none;background-color:#e8e8fc;border:0;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media(prefers-reduced-motion: reduce){.form-range::-moz-range-thumb{transition:none}}.form-range::-moz-range-thumb:active{background-color:#f8f8fe}.form-range::-moz-range-track{width:100%;height:.5rem;color:rgba(0,0,0,0);cursor:pointer;background-color:#f8f9fa;border-color:rgba(0,0,0,0)}.form-range:disabled{pointer-events:none}.form-range:disabled::-webkit-slider-thumb{background-color:rgba(0,0,0,.75)}.form-range:disabled::-moz-range-thumb{background-color:rgba(0,0,0,.75)}.form-floating{position:relative}.form-floating>.form-control,.form-floating>.form-control-plaintext,.form-floating>.form-select{height:calc(3.5rem + calc(1px * 2));min-height:calc(3.5rem + calc(1px * 2));line-height:1.25}.form-floating>label{position:absolute;top:0;left:0;z-index:2;height:100%;padding:1rem .75rem;overflow:hidden;text-align:start;text-overflow:ellipsis;white-space:nowrap;pointer-events:none;border:1px solid rgba(0,0,0,0);transform-origin:0 0;transition:opacity .1s ease-in-out,transform .1s ease-in-out}@media(prefers-reduced-motion: reduce){.form-floating>label{transition:none}}.form-floating>.form-control,.form-floating>.form-control-plaintext{padding:1rem .75rem}.form-floating>.form-control::placeholder,.form-floating>.form-control-plaintext::placeholder{color:rgba(0,0,0,0)}.form-floating>.form-control:focus,.form-floating>.form-control:not(:placeholder-shown),.form-floating>.form-control-plaintext:focus,.form-floating>.form-control-plaintext:not(:placeholder-shown){padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control:-webkit-autofill,.form-floating>.form-control-plaintext:-webkit-autofill{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-select{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control:focus~label,.form-floating>.form-control:not(:placeholder-shown)~label,.form-floating>.form-control-plaintext~label,.form-floating>.form-select~label{color:rgba(var(--bs-body-color-rgb), 0.65);transform:scale(0.85) translateY(-0.5rem) translateX(0.15rem)}.form-floating>.form-control:focus~label::after,.form-floating>.form-control:not(:placeholder-shown)~label::after,.form-floating>.form-control-plaintext~label::after,.form-floating>.form-select~label::after{position:absolute;inset:1rem .375rem;z-index:-1;height:1.5em;content:"";background-color:#fff}.form-floating>.form-control:-webkit-autofill~label{color:rgba(var(--bs-body-color-rgb), 0.65);transform:scale(0.85) translateY(-0.5rem) translateX(0.15rem)}.form-floating>.form-control-plaintext~label{border-width:1px 0}.form-floating>:disabled~label,.form-floating>.form-control:disabled~label{color:#6c757d}.form-floating>:disabled~label::after,.form-floating>.form-control:disabled~label::after{background-color:#e9ecef}.input-group{position:relative;display:flex;display:-webkit-flex;flex-wrap:wrap;-webkit-flex-wrap:wrap;align-items:stretch;-webkit-align-items:stretch;width:100%}.input-group>.form-control,.input-group>.form-select,.input-group>.form-floating{position:relative;flex:1 1 auto;-webkit-flex:1 1 auto;width:1%;min-width:0}.input-group>.form-control:focus,.input-group>.form-select:focus,.input-group>.form-floating:focus-within{z-index:5}.input-group .btn{position:relative;z-index:2}.input-group .btn:focus{z-index:5}.input-group-text{display:flex;display:-webkit-flex;align-items:center;-webkit-align-items:center;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#000;text-align:center;white-space:nowrap;background-color:#f8f9fa;border:1px solid #dee2e6}.input-group-lg>.form-control,.input-group-lg>.form-select,.input-group-lg>.input-group-text,.input-group-lg>.btn{padding:.5rem 1rem;font-size:1.25rem}.input-group-sm>.form-control,.input-group-sm>.form-select,.input-group-sm>.input-group-text,.input-group-sm>.btn{padding:.25rem .5rem;font-size:0.875rem}.input-group-lg>.form-select,.input-group-sm>.form-select{padding-right:3rem}.input-group>:not(:first-child):not(.dropdown-menu):not(.valid-tooltip):not(.valid-feedback):not(.invalid-tooltip):not(.invalid-feedback){margin-left:calc(1px*-1)}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:0.875em;color:#3fb618}.valid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:0.875rem;color:#fff;background-color:#3fb618}.was-validated :valid~.valid-feedback,.was-validated :valid~.valid-tooltip,.is-valid~.valid-feedback,.is-valid~.valid-tooltip{display:block}.was-validated .form-control:valid,.form-control.is-valid{border-color:#3fb618;padding-right:calc(1.5em + 0.75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%233fb618' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(0.375em + 0.1875rem) center;background-size:calc(0.75em + 0.375rem) calc(0.75em + 0.375rem)}.was-validated .form-control:valid:focus,.form-control.is-valid:focus{border-color:#3fb618;box-shadow:0 0 0 .25rem rgba(63,182,24,.25)}.was-validated textarea.form-control:valid,textarea.form-control.is-valid{padding-right:calc(1.5em + 0.75rem);background-position:top calc(0.375em + 0.1875rem) right calc(0.375em + 0.1875rem)}.was-validated .form-select:valid,.form-select.is-valid{border-color:#3fb618}.was-validated .form-select:valid:not([multiple]):not([size]),.was-validated .form-select:valid:not([multiple])[size="1"],.form-select.is-valid:not([multiple]):not([size]),.form-select.is-valid:not([multiple])[size="1"]{--bs-form-select-bg-icon: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%233fb618' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");padding-right:4.125rem;background-position:right .75rem center,center right 2.25rem;background-size:16px 12px,calc(0.75em + 0.375rem) calc(0.75em + 0.375rem)}.was-validated .form-select:valid:focus,.form-select.is-valid:focus{border-color:#3fb618;box-shadow:0 0 0 .25rem rgba(63,182,24,.25)}.was-validated .form-control-color:valid,.form-control-color.is-valid{width:calc(3rem + calc(1.5em + 0.75rem))}.was-validated .form-check-input:valid,.form-check-input.is-valid{border-color:#3fb618}.was-validated .form-check-input:valid:checked,.form-check-input.is-valid:checked{background-color:#3fb618}.was-validated .form-check-input:valid:focus,.form-check-input.is-valid:focus{box-shadow:0 0 0 .25rem rgba(63,182,24,.25)}.was-validated .form-check-input:valid~.form-check-label,.form-check-input.is-valid~.form-check-label{color:#3fb618}.form-check-inline .form-check-input~.valid-feedback{margin-left:.5em}.was-validated .input-group>.form-control:not(:focus):valid,.input-group>.form-control:not(:focus).is-valid,.was-validated .input-group>.form-select:not(:focus):valid,.input-group>.form-select:not(:focus).is-valid,.was-validated .input-group>.form-floating:not(:focus-within):valid,.input-group>.form-floating:not(:focus-within).is-valid{z-index:3}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:0.875em;color:#ff0039}.invalid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:0.875rem;color:#fff;background-color:#ff0039}.was-validated :invalid~.invalid-feedback,.was-validated :invalid~.invalid-tooltip,.is-invalid~.invalid-feedback,.is-invalid~.invalid-tooltip{display:block}.was-validated .form-control:invalid,.form-control.is-invalid{border-color:#ff0039;padding-right:calc(1.5em + 0.75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23ff0039'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23ff0039' stroke='none'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(0.375em + 0.1875rem) center;background-size:calc(0.75em + 0.375rem) calc(0.75em + 0.375rem)}.was-validated .form-control:invalid:focus,.form-control.is-invalid:focus{border-color:#ff0039;box-shadow:0 0 0 .25rem rgba(255,0,57,.25)}.was-validated textarea.form-control:invalid,textarea.form-control.is-invalid{padding-right:calc(1.5em + 0.75rem);background-position:top calc(0.375em + 0.1875rem) right calc(0.375em + 0.1875rem)}.was-validated .form-select:invalid,.form-select.is-invalid{border-color:#ff0039}.was-validated .form-select:invalid:not([multiple]):not([size]),.was-validated .form-select:invalid:not([multiple])[size="1"],.form-select.is-invalid:not([multiple]):not([size]),.form-select.is-invalid:not([multiple])[size="1"]{--bs-form-select-bg-icon: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23ff0039'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23ff0039' stroke='none'/%3e%3c/svg%3e");padding-right:4.125rem;background-position:right .75rem center,center right 2.25rem;background-size:16px 12px,calc(0.75em + 0.375rem) calc(0.75em + 0.375rem)}.was-validated .form-select:invalid:focus,.form-select.is-invalid:focus{border-color:#ff0039;box-shadow:0 0 0 .25rem rgba(255,0,57,.25)}.was-validated .form-control-color:invalid,.form-control-color.is-invalid{width:calc(3rem + calc(1.5em + 0.75rem))}.was-validated .form-check-input:invalid,.form-check-input.is-invalid{border-color:#ff0039}.was-validated .form-check-input:invalid:checked,.form-check-input.is-invalid:checked{background-color:#ff0039}.was-validated .form-check-input:invalid:focus,.form-check-input.is-invalid:focus{box-shadow:0 0 0 .25rem rgba(255,0,57,.25)}.was-validated .form-check-input:invalid~.form-check-label,.form-check-input.is-invalid~.form-check-label{color:#ff0039}.form-check-inline .form-check-input~.invalid-feedback{margin-left:.5em}.was-validated .input-group>.form-control:not(:focus):invalid,.input-group>.form-control:not(:focus).is-invalid,.was-validated .input-group>.form-select:not(:focus):invalid,.input-group>.form-select:not(:focus).is-invalid,.was-validated .input-group>.form-floating:not(:focus-within):invalid,.input-group>.form-floating:not(:focus-within).is-invalid{z-index:4}.btn{--bs-btn-padding-x: 0.75rem;--bs-btn-padding-y: 0.375rem;--bs-btn-font-family: ;--bs-btn-font-size:1rem;--bs-btn-font-weight: 400;--bs-btn-line-height: 1.5;--bs-btn-color: #000000;--bs-btn-bg: transparent;--bs-btn-border-width: 1px;--bs-btn-border-color: transparent;--bs-btn-border-radius: 0.25rem;--bs-btn-hover-border-color: transparent;--bs-btn-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.15), 0 1px 1px rgba(0, 0, 0, 0.075);--bs-btn-disabled-opacity: 0.65;--bs-btn-focus-box-shadow: 0 0 0 0.25rem rgba(var(--bs-btn-focus-shadow-rgb), .5);display:inline-block;padding:var(--bs-btn-padding-y) var(--bs-btn-padding-x);font-family:var(--bs-btn-font-family);font-size:var(--bs-btn-font-size);font-weight:var(--bs-btn-font-weight);line-height:var(--bs-btn-line-height);color:var(--bs-btn-color);text-align:center;text-decoration:none;-webkit-text-decoration:none;-moz-text-decoration:none;-ms-text-decoration:none;-o-text-decoration:none;vertical-align:middle;cursor:pointer;user-select:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;-o-user-select:none;border:var(--bs-btn-border-width) solid var(--bs-btn-border-color);background-color:var(--bs-btn-bg);transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media(prefers-reduced-motion: reduce){.btn{transition:none}}.btn:hover{color:var(--bs-btn-hover-color);background-color:var(--bs-btn-hover-bg);border-color:var(--bs-btn-hover-border-color)}.btn-check+.btn:hover{color:var(--bs-btn-color);background-color:var(--bs-btn-bg);border-color:var(--bs-btn-border-color)}.btn:focus-visible{color:var(--bs-btn-hover-color);background-color:var(--bs-btn-hover-bg);border-color:var(--bs-btn-hover-border-color);outline:0;box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:focus-visible+.btn{border-color:var(--bs-btn-hover-border-color);outline:0;box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:checked+.btn,:not(.btn-check)+.btn:active,.btn:first-child:active,.btn.active,.btn.show{color:var(--bs-btn-active-color);background-color:var(--bs-btn-active-bg);border-color:var(--bs-btn-active-border-color)}.btn-check:checked+.btn:focus-visible,:not(.btn-check)+.btn:active:focus-visible,.btn:first-child:active:focus-visible,.btn.active:focus-visible,.btn.show:focus-visible{box-shadow:var(--bs-btn-focus-box-shadow)}.btn:disabled,.btn.disabled,fieldset:disabled .btn{color:var(--bs-btn-disabled-color);pointer-events:none;background-color:var(--bs-btn-disabled-bg);border-color:var(--bs-btn-disabled-border-color);opacity:var(--bs-btn-disabled-opacity)}.btn-default{--bs-btn-color: #fff;--bs-btn-bg: #343a40;--bs-btn-border-color: #343a40;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #2c3136;--bs-btn-hover-border-color: #2a2e33;--bs-btn-focus-shadow-rgb: 82, 88, 93;--bs-btn-active-color: #fff;--bs-btn-active-bg: #2a2e33;--bs-btn-active-border-color: #272c30;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #fff;--bs-btn-disabled-bg: #343a40;--bs-btn-disabled-border-color: #343a40}.btn-primary{--bs-btn-color: #000;--bs-btn-bg: #E8E8FC;--bs-btn-border-color: #E8E8FC;--bs-btn-hover-color: #000;--bs-btn-hover-bg: #ebebfc;--bs-btn-hover-border-color: #eaeafc;--bs-btn-focus-shadow-rgb: 197, 197, 214;--bs-btn-active-color: #000;--bs-btn-active-bg: #ededfd;--bs-btn-active-border-color: #eaeafc;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #000;--bs-btn-disabled-bg: #E8E8FC;--bs-btn-disabled-border-color: #E8E8FC}.btn-secondary{--bs-btn-color: #fff;--bs-btn-bg: #343a40;--bs-btn-border-color: #343a40;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #2c3136;--bs-btn-hover-border-color: #2a2e33;--bs-btn-focus-shadow-rgb: 82, 88, 93;--bs-btn-active-color: #fff;--bs-btn-active-bg: #2a2e33;--bs-btn-active-border-color: #272c30;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #fff;--bs-btn-disabled-bg: #343a40;--bs-btn-disabled-border-color: #343a40}.btn-success{--bs-btn-color: #fff;--bs-btn-bg: #3fb618;--bs-btn-border-color: #3fb618;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #369b14;--bs-btn-hover-border-color: #329213;--bs-btn-focus-shadow-rgb: 92, 193, 59;--bs-btn-active-color: #fff;--bs-btn-active-bg: #329213;--bs-btn-active-border-color: #2f8912;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #fff;--bs-btn-disabled-bg: #3fb618;--bs-btn-disabled-border-color: #3fb618}.btn-info{--bs-btn-color: #fff;--bs-btn-bg: #9954bb;--bs-btn-border-color: #9954bb;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #82479f;--bs-btn-hover-border-color: #7a4396;--bs-btn-focus-shadow-rgb: 168, 110, 197;--bs-btn-active-color: #fff;--bs-btn-active-bg: #7a4396;--bs-btn-active-border-color: #733f8c;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #fff;--bs-btn-disabled-bg: #9954bb;--bs-btn-disabled-border-color: #9954bb}.btn-warning{--bs-btn-color: #fff;--bs-btn-bg: #ff7518;--bs-btn-border-color: #ff7518;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #d96314;--bs-btn-hover-border-color: #cc5e13;--bs-btn-focus-shadow-rgb: 255, 138, 59;--bs-btn-active-color: #fff;--bs-btn-active-bg: #cc5e13;--bs-btn-active-border-color: #bf5812;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #fff;--bs-btn-disabled-bg: #ff7518;--bs-btn-disabled-border-color: #ff7518}.btn-danger{--bs-btn-color: #fff;--bs-btn-bg: #ff0039;--bs-btn-border-color: #ff0039;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #d90030;--bs-btn-hover-border-color: #cc002e;--bs-btn-focus-shadow-rgb: 255, 38, 87;--bs-btn-active-color: #fff;--bs-btn-active-bg: #cc002e;--bs-btn-active-border-color: #bf002b;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #fff;--bs-btn-disabled-bg: #ff0039;--bs-btn-disabled-border-color: #ff0039}.btn-light{--bs-btn-color: #000;--bs-btn-bg: #f8f9fa;--bs-btn-border-color: #f8f9fa;--bs-btn-hover-color: #000;--bs-btn-hover-bg: #d3d4d5;--bs-btn-hover-border-color: #c6c7c8;--bs-btn-focus-shadow-rgb: 211, 212, 213;--bs-btn-active-color: #000;--bs-btn-active-bg: #c6c7c8;--bs-btn-active-border-color: #babbbc;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #000;--bs-btn-disabled-bg: #f8f9fa;--bs-btn-disabled-border-color: #f8f9fa}.btn-dark{--bs-btn-color: #fff;--bs-btn-bg: #343a40;--bs-btn-border-color: #343a40;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #52585d;--bs-btn-hover-border-color: #484e53;--bs-btn-focus-shadow-rgb: 82, 88, 93;--bs-btn-active-color: #fff;--bs-btn-active-bg: #5d6166;--bs-btn-active-border-color: #484e53;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #fff;--bs-btn-disabled-bg: #343a40;--bs-btn-disabled-border-color: #343a40}.btn-outline-default{--bs-btn-color: #343a40;--bs-btn-border-color: #343a40;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #343a40;--bs-btn-hover-border-color: #343a40;--bs-btn-focus-shadow-rgb: 52, 58, 64;--bs-btn-active-color: #fff;--bs-btn-active-bg: #343a40;--bs-btn-active-border-color: #343a40;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #343a40;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #343a40;--bs-btn-bg: transparent;--bs-gradient: none}.btn-outline-primary{--bs-btn-color: #E8E8FC;--bs-btn-border-color: #E8E8FC;--bs-btn-hover-color: #000;--bs-btn-hover-bg: #E8E8FC;--bs-btn-hover-border-color: #E8E8FC;--bs-btn-focus-shadow-rgb: 232, 232, 252;--bs-btn-active-color: #000;--bs-btn-active-bg: #E8E8FC;--bs-btn-active-border-color: #E8E8FC;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #E8E8FC;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #E8E8FC;--bs-btn-bg: transparent;--bs-gradient: none}.btn-outline-secondary{--bs-btn-color: #343a40;--bs-btn-border-color: #343a40;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #343a40;--bs-btn-hover-border-color: #343a40;--bs-btn-focus-shadow-rgb: 52, 58, 64;--bs-btn-active-color: #fff;--bs-btn-active-bg: #343a40;--bs-btn-active-border-color: #343a40;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #343a40;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #343a40;--bs-btn-bg: transparent;--bs-gradient: none}.btn-outline-success{--bs-btn-color: #3fb618;--bs-btn-border-color: #3fb618;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #3fb618;--bs-btn-hover-border-color: #3fb618;--bs-btn-focus-shadow-rgb: 63, 182, 24;--bs-btn-active-color: #fff;--bs-btn-active-bg: #3fb618;--bs-btn-active-border-color: #3fb618;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #3fb618;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #3fb618;--bs-btn-bg: transparent;--bs-gradient: none}.btn-outline-info{--bs-btn-color: #9954bb;--bs-btn-border-color: #9954bb;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #9954bb;--bs-btn-hover-border-color: #9954bb;--bs-btn-focus-shadow-rgb: 153, 84, 187;--bs-btn-active-color: #fff;--bs-btn-active-bg: #9954bb;--bs-btn-active-border-color: #9954bb;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #9954bb;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #9954bb;--bs-btn-bg: transparent;--bs-gradient: none}.btn-outline-warning{--bs-btn-color: #ff7518;--bs-btn-border-color: #ff7518;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #ff7518;--bs-btn-hover-border-color: #ff7518;--bs-btn-focus-shadow-rgb: 255, 117, 24;--bs-btn-active-color: #fff;--bs-btn-active-bg: #ff7518;--bs-btn-active-border-color: #ff7518;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #ff7518;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #ff7518;--bs-btn-bg: transparent;--bs-gradient: none}.btn-outline-danger{--bs-btn-color: #ff0039;--bs-btn-border-color: #ff0039;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #ff0039;--bs-btn-hover-border-color: #ff0039;--bs-btn-focus-shadow-rgb: 255, 0, 57;--bs-btn-active-color: #fff;--bs-btn-active-bg: #ff0039;--bs-btn-active-border-color: #ff0039;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #ff0039;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #ff0039;--bs-btn-bg: transparent;--bs-gradient: none}.btn-outline-light{--bs-btn-color: #f8f9fa;--bs-btn-border-color: #f8f9fa;--bs-btn-hover-color: #000;--bs-btn-hover-bg: #f8f9fa;--bs-btn-hover-border-color: #f8f9fa;--bs-btn-focus-shadow-rgb: 248, 249, 250;--bs-btn-active-color: #000;--bs-btn-active-bg: #f8f9fa;--bs-btn-active-border-color: #f8f9fa;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #f8f9fa;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #f8f9fa;--bs-btn-bg: transparent;--bs-gradient: none}.btn-outline-dark{--bs-btn-color: #343a40;--bs-btn-border-color: #343a40;--bs-btn-hover-color: #fff;--bs-btn-hover-bg: #343a40;--bs-btn-hover-border-color: #343a40;--bs-btn-focus-shadow-rgb: 52, 58, 64;--bs-btn-active-color: #fff;--bs-btn-active-bg: #343a40;--bs-btn-active-border-color: #343a40;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #343a40;--bs-btn-disabled-bg: transparent;--bs-btn-disabled-border-color: #343a40;--bs-btn-bg: transparent;--bs-gradient: none}.btn-link{--bs-btn-font-weight: 400;--bs-btn-color: #4040BF;--bs-btn-bg: transparent;--bs-btn-border-color: transparent;--bs-btn-hover-color: #333399;--bs-btn-hover-border-color: transparent;--bs-btn-active-color: #333399;--bs-btn-active-border-color: transparent;--bs-btn-disabled-color: #6c757d;--bs-btn-disabled-border-color: transparent;--bs-btn-box-shadow: 0 0 0 #000;--bs-btn-focus-shadow-rgb: 93, 93, 201;text-decoration:underline;-webkit-text-decoration:underline;-moz-text-decoration:underline;-ms-text-decoration:underline;-o-text-decoration:underline}.btn-link:focus-visible{color:var(--bs-btn-color)}.btn-link:hover{color:var(--bs-btn-hover-color)}.btn-lg,.btn-group-lg>.btn{--bs-btn-padding-y: 0.5rem;--bs-btn-padding-x: 1rem;--bs-btn-font-size:1.25rem;--bs-btn-border-radius: 0.5rem}.btn-sm,.btn-group-sm>.btn{--bs-btn-padding-y: 0.25rem;--bs-btn-padding-x: 0.5rem;--bs-btn-font-size:0.875rem;--bs-btn-border-radius: 0.2em}.fade{transition:opacity .15s linear}@media(prefers-reduced-motion: reduce){.fade{transition:none}}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{height:0;overflow:hidden;transition:height .2s ease}@media(prefers-reduced-motion: reduce){.collapsing{transition:none}}.collapsing.collapse-horizontal{width:0;height:auto;transition:width .35s ease}@media(prefers-reduced-motion: reduce){.collapsing.collapse-horizontal{transition:none}}.dropup,.dropend,.dropdown,.dropstart,.dropup-center,.dropdown-center{position:relative}.dropdown-toggle{white-space:nowrap}.dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid;border-right:.3em solid rgba(0,0,0,0);border-bottom:0;border-left:.3em solid rgba(0,0,0,0)}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{--bs-dropdown-zindex: 1000;--bs-dropdown-min-width: 10rem;--bs-dropdown-padding-x: 0;--bs-dropdown-padding-y: 0.5rem;--bs-dropdown-spacer: 0.125rem;--bs-dropdown-font-size:1rem;--bs-dropdown-color: #000000;--bs-dropdown-bg: #FFFFFF;--bs-dropdown-border-color: rgba(0, 0, 0, 0.175);--bs-dropdown-border-radius: 0.25rem;--bs-dropdown-border-width: 1px;--bs-dropdown-inner-border-radius: calc(0.25rem - 1px);--bs-dropdown-divider-bg: rgba(0, 0, 0, 0.175);--bs-dropdown-divider-margin-y: 0.5rem;--bs-dropdown-box-shadow: 0 0.5rem 1rem rgba(0, 0, 0, 0.15);--bs-dropdown-link-color: #000000;--bs-dropdown-link-hover-color: #000000;--bs-dropdown-link-hover-bg: #f8f9fa;--bs-dropdown-link-active-color: #000;--bs-dropdown-link-active-bg: #E8E8FC;--bs-dropdown-link-disabled-color: rgba(0, 0, 0, 0.5);--bs-dropdown-item-padding-x: 1rem;--bs-dropdown-item-padding-y: 0.25rem;--bs-dropdown-header-color: #6c757d;--bs-dropdown-header-padding-x: 1rem;--bs-dropdown-header-padding-y: 0.5rem;position:absolute;z-index:var(--bs-dropdown-zindex);display:none;min-width:var(--bs-dropdown-min-width);padding:var(--bs-dropdown-padding-y) var(--bs-dropdown-padding-x);margin:0;font-size:var(--bs-dropdown-font-size);color:var(--bs-dropdown-color);text-align:left;list-style:none;background-color:var(--bs-dropdown-bg);background-clip:padding-box;border:var(--bs-dropdown-border-width) solid var(--bs-dropdown-border-color)}.dropdown-menu[data-bs-popper]{top:100%;left:0;margin-top:var(--bs-dropdown-spacer)}.dropdown-menu-start{--bs-position: start}.dropdown-menu-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-end{--bs-position: end}.dropdown-menu-end[data-bs-popper]{right:0;left:auto}@media(min-width: 576px){.dropdown-menu-sm-start{--bs-position: start}.dropdown-menu-sm-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-sm-end{--bs-position: end}.dropdown-menu-sm-end[data-bs-popper]{right:0;left:auto}}@media(min-width: 768px){.dropdown-menu-md-start{--bs-position: start}.dropdown-menu-md-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-md-end{--bs-position: end}.dropdown-menu-md-end[data-bs-popper]{right:0;left:auto}}@media(min-width: 992px){.dropdown-menu-lg-start{--bs-position: start}.dropdown-menu-lg-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-lg-end{--bs-position: end}.dropdown-menu-lg-end[data-bs-popper]{right:0;left:auto}}@media(min-width: 1200px){.dropdown-menu-xl-start{--bs-position: start}.dropdown-menu-xl-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-xl-end{--bs-position: end}.dropdown-menu-xl-end[data-bs-popper]{right:0;left:auto}}@media(min-width: 1400px){.dropdown-menu-xxl-start{--bs-position: start}.dropdown-menu-xxl-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-xxl-end{--bs-position: end}.dropdown-menu-xxl-end[data-bs-popper]{right:0;left:auto}}.dropup .dropdown-menu[data-bs-popper]{top:auto;bottom:100%;margin-top:0;margin-bottom:var(--bs-dropdown-spacer)}.dropup .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:0;border-right:.3em solid rgba(0,0,0,0);border-bottom:.3em solid;border-left:.3em solid rgba(0,0,0,0)}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropend .dropdown-menu[data-bs-popper]{top:0;right:auto;left:100%;margin-top:0;margin-left:var(--bs-dropdown-spacer)}.dropend .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid rgba(0,0,0,0);border-right:0;border-bottom:.3em solid rgba(0,0,0,0);border-left:.3em solid}.dropend .dropdown-toggle:empty::after{margin-left:0}.dropend .dropdown-toggle::after{vertical-align:0}.dropstart .dropdown-menu[data-bs-popper]{top:0;right:100%;left:auto;margin-top:0;margin-right:var(--bs-dropdown-spacer)}.dropstart .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:""}.dropstart .dropdown-toggle::after{display:none}.dropstart .dropdown-toggle::before{display:inline-block;margin-right:.255em;vertical-align:.255em;content:"";border-top:.3em solid rgba(0,0,0,0);border-right:.3em solid;border-bottom:.3em solid rgba(0,0,0,0)}.dropstart .dropdown-toggle:empty::after{margin-left:0}.dropstart .dropdown-toggle::before{vertical-align:0}.dropdown-divider{height:0;margin:var(--bs-dropdown-divider-margin-y) 0;overflow:hidden;border-top:1px solid var(--bs-dropdown-divider-bg);opacity:1}.dropdown-item{display:block;width:100%;padding:var(--bs-dropdown-item-padding-y) var(--bs-dropdown-item-padding-x);clear:both;font-weight:400;color:var(--bs-dropdown-link-color);text-align:inherit;text-decoration:none;-webkit-text-decoration:none;-moz-text-decoration:none;-ms-text-decoration:none;-o-text-decoration:none;white-space:nowrap;background-color:rgba(0,0,0,0);border:0}.dropdown-item:hover,.dropdown-item:focus{color:var(--bs-dropdown-link-hover-color);background-color:var(--bs-dropdown-link-hover-bg)}.dropdown-item.active,.dropdown-item:active{color:var(--bs-dropdown-link-active-color);text-decoration:none;background-color:var(--bs-dropdown-link-active-bg)}.dropdown-item.disabled,.dropdown-item:disabled{color:var(--bs-dropdown-link-disabled-color);pointer-events:none;background-color:rgba(0,0,0,0)}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:var(--bs-dropdown-header-padding-y) var(--bs-dropdown-header-padding-x);margin-bottom:0;font-size:0.875rem;color:var(--bs-dropdown-header-color);white-space:nowrap}.dropdown-item-text{display:block;padding:var(--bs-dropdown-item-padding-y) var(--bs-dropdown-item-padding-x);color:var(--bs-dropdown-link-color)}.dropdown-menu-dark{--bs-dropdown-color: #dee2e6;--bs-dropdown-bg: #343a40;--bs-dropdown-border-color: rgba(0, 0, 0, 0.175);--bs-dropdown-box-shadow: ;--bs-dropdown-link-color: #dee2e6;--bs-dropdown-link-hover-color: #fff;--bs-dropdown-divider-bg: rgba(0, 0, 0, 0.175);--bs-dropdown-link-hover-bg: rgba(255, 255, 255, 0.15);--bs-dropdown-link-active-color: #000;--bs-dropdown-link-active-bg: #E8E8FC;--bs-dropdown-link-disabled-color: #adb5bd;--bs-dropdown-header-color: #adb5bd}.btn-group,.btn-group-vertical{position:relative;display:inline-flex;vertical-align:middle}.btn-group>.btn,.btn-group-vertical>.btn{position:relative;flex:1 1 auto;-webkit-flex:1 1 auto}.btn-group>.btn-check:checked+.btn,.btn-group>.btn-check:focus+.btn,.btn-group>.btn:hover,.btn-group>.btn:focus,.btn-group>.btn:active,.btn-group>.btn.active,.btn-group-vertical>.btn-check:checked+.btn,.btn-group-vertical>.btn-check:focus+.btn,.btn-group-vertical>.btn:hover,.btn-group-vertical>.btn:focus,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn.active{z-index:1}.btn-toolbar{display:flex;display:-webkit-flex;flex-wrap:wrap;-webkit-flex-wrap:wrap;justify-content:flex-start;-webkit-justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group>:not(.btn-check:first-child)+.btn,.btn-group>.btn-group:not(:first-child){margin-left:calc(1px*-1)}.dropdown-toggle-split{padding-right:.5625rem;padding-left:.5625rem}.dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after,.dropend .dropdown-toggle-split::after{margin-left:0}.dropstart .dropdown-toggle-split::before{margin-right:0}.btn-sm+.dropdown-toggle-split,.btn-group-sm>.btn+.dropdown-toggle-split{padding-right:.375rem;padding-left:.375rem}.btn-lg+.dropdown-toggle-split,.btn-group-lg>.btn+.dropdown-toggle-split{padding-right:.75rem;padding-left:.75rem}.btn-group-vertical{flex-direction:column;-webkit-flex-direction:column;align-items:flex-start;-webkit-align-items:flex-start;justify-content:center;-webkit-justify-content:center}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group{width:100%}.btn-group-vertical>.btn:not(:first-child),.btn-group-vertical>.btn-group:not(:first-child){margin-top:calc(1px*-1)}.nav{--bs-nav-link-padding-x: 1rem;--bs-nav-link-padding-y: 0.5rem;--bs-nav-link-font-weight: ;--bs-nav-link-color: #4040BF;--bs-nav-link-hover-color: #333399;--bs-nav-link-disabled-color: rgba(0, 0, 0, 0.75);display:flex;display:-webkit-flex;flex-wrap:wrap;-webkit-flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:var(--bs-nav-link-padding-y) var(--bs-nav-link-padding-x);font-size:var(--bs-nav-link-font-size);font-weight:var(--bs-nav-link-font-weight);color:var(--bs-nav-link-color);text-decoration:none;-webkit-text-decoration:none;-moz-text-decoration:none;-ms-text-decoration:none;-o-text-decoration:none;background:none;border:0;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out}@media(prefers-reduced-motion: reduce){.nav-link{transition:none}}.nav-link:hover,.nav-link:focus{color:var(--bs-nav-link-hover-color)}.nav-link:focus-visible{outline:0;box-shadow:0 0 0 .25rem rgba(232,232,252,.25)}.nav-link.disabled,.nav-link:disabled{color:var(--bs-nav-link-disabled-color);pointer-events:none;cursor:default}.nav-tabs{--bs-nav-tabs-border-width: 1px;--bs-nav-tabs-border-color: #dee2e6;--bs-nav-tabs-border-radius: 0.25rem;--bs-nav-tabs-link-hover-border-color: #e9ecef #e9ecef #dee2e6;--bs-nav-tabs-link-active-color: #000;--bs-nav-tabs-link-active-bg: #FFFFFF;--bs-nav-tabs-link-active-border-color: #dee2e6 #dee2e6 #FFFFFF;border-bottom:var(--bs-nav-tabs-border-width) solid var(--bs-nav-tabs-border-color)}.nav-tabs .nav-link{margin-bottom:calc(-1*var(--bs-nav-tabs-border-width));border:var(--bs-nav-tabs-border-width) solid rgba(0,0,0,0)}.nav-tabs .nav-link:hover,.nav-tabs .nav-link:focus{isolation:isolate;border-color:var(--bs-nav-tabs-link-hover-border-color)}.nav-tabs .nav-link.active,.nav-tabs .nav-item.show .nav-link{color:var(--bs-nav-tabs-link-active-color);background-color:var(--bs-nav-tabs-link-active-bg);border-color:var(--bs-nav-tabs-link-active-border-color)}.nav-tabs .dropdown-menu{margin-top:calc(-1*var(--bs-nav-tabs-border-width))}.nav-pills{--bs-nav-pills-border-radius: 0.25rem;--bs-nav-pills-link-active-color: #000;--bs-nav-pills-link-active-bg: #E8E8FC}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:var(--bs-nav-pills-link-active-color);background-color:var(--bs-nav-pills-link-active-bg)}.nav-underline{--bs-nav-underline-gap: 1rem;--bs-nav-underline-border-width: 0.125rem;--bs-nav-underline-link-active-color: #000;gap:var(--bs-nav-underline-gap)}.nav-underline .nav-link{padding-right:0;padding-left:0;border-bottom:var(--bs-nav-underline-border-width) solid rgba(0,0,0,0)}.nav-underline .nav-link:hover,.nav-underline .nav-link:focus{border-bottom-color:currentcolor}.nav-underline .nav-link.active,.nav-underline .show>.nav-link{font-weight:700;color:var(--bs-nav-underline-link-active-color);border-bottom-color:currentcolor}.nav-fill>.nav-link,.nav-fill .nav-item{flex:1 1 auto;-webkit-flex:1 1 auto;text-align:center}.nav-justified>.nav-link,.nav-justified .nav-item{flex-basis:0;-webkit-flex-basis:0;flex-grow:1;-webkit-flex-grow:1;text-align:center}.nav-fill .nav-item .nav-link,.nav-justified .nav-item .nav-link{width:100%}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{--bs-navbar-padding-x: 0;--bs-navbar-padding-y: 0.5rem;--bs-navbar-color: #FFFFFF;--bs-navbar-hover-color: rgba(0, 0, 0, 0.8);--bs-navbar-disabled-color: rgba(255, 255, 255, 0.75);--bs-navbar-active-color: #000000;--bs-navbar-brand-padding-y: 0.3125rem;--bs-navbar-brand-margin-end: 1rem;--bs-navbar-brand-font-size: 1.25rem;--bs-navbar-brand-color: #FFFFFF;--bs-navbar-brand-hover-color: #000000;--bs-navbar-nav-link-padding-x: 0.5rem;--bs-navbar-toggler-padding-y: 0.25;--bs-navbar-toggler-padding-x: 0;--bs-navbar-toggler-font-size: 1.25rem;--bs-navbar-toggler-icon-bg: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='%23FFFFFF' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e");--bs-navbar-toggler-border-color: rgba(255, 255, 255, 0);--bs-navbar-toggler-border-radius: 0.25rem;--bs-navbar-toggler-focus-width: 0.25rem;--bs-navbar-toggler-transition: box-shadow 0.15s ease-in-out;position:relative;display:flex;display:-webkit-flex;flex-wrap:wrap;-webkit-flex-wrap:wrap;align-items:center;-webkit-align-items:center;justify-content:space-between;-webkit-justify-content:space-between;padding:var(--bs-navbar-padding-y) var(--bs-navbar-padding-x)}.navbar>.container,.navbar>.container-fluid,.navbar>.container-sm,.navbar>.container-md,.navbar>.container-lg,.navbar>.container-xl,.navbar>.container-xxl{display:flex;display:-webkit-flex;flex-wrap:inherit;-webkit-flex-wrap:inherit;align-items:center;-webkit-align-items:center;justify-content:space-between;-webkit-justify-content:space-between}.navbar-brand{padding-top:var(--bs-navbar-brand-padding-y);padding-bottom:var(--bs-navbar-brand-padding-y);margin-right:var(--bs-navbar-brand-margin-end);font-size:var(--bs-navbar-brand-font-size);color:var(--bs-navbar-brand-color);text-decoration:none;-webkit-text-decoration:none;-moz-text-decoration:none;-ms-text-decoration:none;-o-text-decoration:none;white-space:nowrap}.navbar-brand:hover,.navbar-brand:focus{color:var(--bs-navbar-brand-hover-color)}.navbar-nav{--bs-nav-link-padding-x: 0;--bs-nav-link-padding-y: 0.5rem;--bs-nav-link-font-weight: ;--bs-nav-link-color: var(--bs-navbar-color);--bs-nav-link-hover-color: var(--bs-navbar-hover-color);--bs-nav-link-disabled-color: var(--bs-navbar-disabled-color);display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link.active,.navbar-nav .nav-link.show{color:var(--bs-navbar-active-color)}.navbar-nav .dropdown-menu{position:static}.navbar-text{padding-top:.5rem;padding-bottom:.5rem;color:var(--bs-navbar-color)}.navbar-text a,.navbar-text a:hover,.navbar-text a:focus{color:var(--bs-navbar-active-color)}.navbar-collapse{flex-basis:100%;-webkit-flex-basis:100%;flex-grow:1;-webkit-flex-grow:1;align-items:center;-webkit-align-items:center}.navbar-toggler{padding:var(--bs-navbar-toggler-padding-y) var(--bs-navbar-toggler-padding-x);font-size:var(--bs-navbar-toggler-font-size);line-height:1;color:var(--bs-navbar-color);background-color:rgba(0,0,0,0);border:var(--bs-border-width) solid var(--bs-navbar-toggler-border-color);transition:var(--bs-navbar-toggler-transition)}@media(prefers-reduced-motion: reduce){.navbar-toggler{transition:none}}.navbar-toggler:hover{text-decoration:none}.navbar-toggler:focus{text-decoration:none;outline:0;box-shadow:0 0 0 var(--bs-navbar-toggler-focus-width)}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;background-image:var(--bs-navbar-toggler-icon-bg);background-repeat:no-repeat;background-position:center;background-size:100%}.navbar-nav-scroll{max-height:var(--bs-scroll-height, 75vh);overflow-y:auto}@media(min-width: 576px){.navbar-expand-sm{flex-wrap:nowrap;-webkit-flex-wrap:nowrap;justify-content:flex-start;-webkit-justify-content:flex-start}.navbar-expand-sm .navbar-nav{flex-direction:row;-webkit-flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-sm .navbar-nav-scroll{overflow:visible}.navbar-expand-sm .navbar-collapse{display:flex !important;display:-webkit-flex !important;flex-basis:auto;-webkit-flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}.navbar-expand-sm .offcanvas{position:static;z-index:auto;flex-grow:1;-webkit-flex-grow:1;width:auto !important;height:auto !important;visibility:visible !important;background-color:rgba(0,0,0,0) !important;border:0 !important;transform:none !important;transition:none}.navbar-expand-sm .offcanvas .offcanvas-header{display:none}.navbar-expand-sm .offcanvas .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible}}@media(min-width: 768px){.navbar-expand-md{flex-wrap:nowrap;-webkit-flex-wrap:nowrap;justify-content:flex-start;-webkit-justify-content:flex-start}.navbar-expand-md .navbar-nav{flex-direction:row;-webkit-flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-md .navbar-nav-scroll{overflow:visible}.navbar-expand-md .navbar-collapse{display:flex !important;display:-webkit-flex !important;flex-basis:auto;-webkit-flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}.navbar-expand-md .offcanvas{position:static;z-index:auto;flex-grow:1;-webkit-flex-grow:1;width:auto !important;height:auto !important;visibility:visible !important;background-color:rgba(0,0,0,0) !important;border:0 !important;transform:none !important;transition:none}.navbar-expand-md .offcanvas .offcanvas-header{display:none}.navbar-expand-md .offcanvas .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible}}@media(min-width: 992px){.navbar-expand-lg{flex-wrap:nowrap;-webkit-flex-wrap:nowrap;justify-content:flex-start;-webkit-justify-content:flex-start}.navbar-expand-lg .navbar-nav{flex-direction:row;-webkit-flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-lg .navbar-nav-scroll{overflow:visible}.navbar-expand-lg .navbar-collapse{display:flex !important;display:-webkit-flex !important;flex-basis:auto;-webkit-flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}.navbar-expand-lg .offcanvas{position:static;z-index:auto;flex-grow:1;-webkit-flex-grow:1;width:auto !important;height:auto !important;visibility:visible !important;background-color:rgba(0,0,0,0) !important;border:0 !important;transform:none !important;transition:none}.navbar-expand-lg .offcanvas .offcanvas-header{display:none}.navbar-expand-lg .offcanvas .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible}}@media(min-width: 1200px){.navbar-expand-xl{flex-wrap:nowrap;-webkit-flex-wrap:nowrap;justify-content:flex-start;-webkit-justify-content:flex-start}.navbar-expand-xl .navbar-nav{flex-direction:row;-webkit-flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-xl .navbar-nav-scroll{overflow:visible}.navbar-expand-xl .navbar-collapse{display:flex !important;display:-webkit-flex !important;flex-basis:auto;-webkit-flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}.navbar-expand-xl .offcanvas{position:static;z-index:auto;flex-grow:1;-webkit-flex-grow:1;width:auto !important;height:auto !important;visibility:visible !important;background-color:rgba(0,0,0,0) !important;border:0 !important;transform:none !important;transition:none}.navbar-expand-xl .offcanvas .offcanvas-header{display:none}.navbar-expand-xl .offcanvas .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible}}@media(min-width: 1400px){.navbar-expand-xxl{flex-wrap:nowrap;-webkit-flex-wrap:nowrap;justify-content:flex-start;-webkit-justify-content:flex-start}.navbar-expand-xxl .navbar-nav{flex-direction:row;-webkit-flex-direction:row}.navbar-expand-xxl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xxl .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-xxl .navbar-nav-scroll{overflow:visible}.navbar-expand-xxl .navbar-collapse{display:flex !important;display:-webkit-flex !important;flex-basis:auto;-webkit-flex-basis:auto}.navbar-expand-xxl .navbar-toggler{display:none}.navbar-expand-xxl .offcanvas{position:static;z-index:auto;flex-grow:1;-webkit-flex-grow:1;width:auto !important;height:auto !important;visibility:visible !important;background-color:rgba(0,0,0,0) !important;border:0 !important;transform:none !important;transition:none}.navbar-expand-xxl .offcanvas .offcanvas-header{display:none}.navbar-expand-xxl .offcanvas .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible}}.navbar-expand{flex-wrap:nowrap;-webkit-flex-wrap:nowrap;justify-content:flex-start;-webkit-justify-content:flex-start}.navbar-expand .navbar-nav{flex-direction:row;-webkit-flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand .navbar-nav-scroll{overflow:visible}.navbar-expand .navbar-collapse{display:flex !important;display:-webkit-flex !important;flex-basis:auto;-webkit-flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-expand .offcanvas{position:static;z-index:auto;flex-grow:1;-webkit-flex-grow:1;width:auto !important;height:auto !important;visibility:visible !important;background-color:rgba(0,0,0,0) !important;border:0 !important;transform:none !important;transition:none}.navbar-expand .offcanvas .offcanvas-header{display:none}.navbar-expand .offcanvas .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible}.navbar-dark,.navbar[data-bs-theme=dark]{--bs-navbar-color: #FFFFFF;--bs-navbar-hover-color: rgba(0, 0, 0, 0.8);--bs-navbar-disabled-color: rgba(255, 255, 255, 0.75);--bs-navbar-active-color: #000000;--bs-navbar-brand-color: #FFFFFF;--bs-navbar-brand-hover-color: #000000;--bs-navbar-toggler-border-color: rgba(255, 255, 255, 0);--bs-navbar-toggler-icon-bg: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='%23FFFFFF' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}[data-bs-theme=dark] .navbar-toggler-icon{--bs-navbar-toggler-icon-bg: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='%23FFFFFF' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.card{--bs-card-spacer-y: 1rem;--bs-card-spacer-x: 1rem;--bs-card-title-spacer-y: 0.5rem;--bs-card-title-color: ;--bs-card-subtitle-color: ;--bs-card-border-width: 1px;--bs-card-border-color: rgba(0, 0, 0, 0.175);--bs-card-border-radius: 0.25rem;--bs-card-box-shadow: ;--bs-card-inner-border-radius: calc(0.25rem - 1px);--bs-card-cap-padding-y: 0.5rem;--bs-card-cap-padding-x: 1rem;--bs-card-cap-bg: rgba(52, 58, 64, 0.25);--bs-card-cap-color: ;--bs-card-height: ;--bs-card-color: ;--bs-card-bg: #FFFFFF;--bs-card-img-overlay-padding: 1rem;--bs-card-group-margin: 0.75rem;position:relative;display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;min-width:0;height:var(--bs-card-height);color:var(--bs-body-color);word-wrap:break-word;background-color:var(--bs-card-bg);background-clip:border-box;border:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card>hr{margin-right:0;margin-left:0}.card>.list-group{border-top:inherit;border-bottom:inherit}.card>.list-group:first-child{border-top-width:0}.card>.list-group:last-child{border-bottom-width:0}.card>.card-header+.list-group,.card>.list-group+.card-footer{border-top:0}.card-body{flex:1 1 auto;-webkit-flex:1 1 auto;padding:var(--bs-card-spacer-y) var(--bs-card-spacer-x);color:var(--bs-card-color)}.card-title{margin-bottom:var(--bs-card-title-spacer-y);color:var(--bs-card-title-color)}.card-subtitle{margin-top:calc(-0.5*var(--bs-card-title-spacer-y));margin-bottom:0;color:var(--bs-card-subtitle-color)}.card-text:last-child{margin-bottom:0}.card-link+.card-link{margin-left:var(--bs-card-spacer-x)}.card-header{padding:var(--bs-card-cap-padding-y) var(--bs-card-cap-padding-x);margin-bottom:0;color:var(--bs-card-cap-color);background-color:var(--bs-card-cap-bg);border-bottom:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card-footer{padding:var(--bs-card-cap-padding-y) var(--bs-card-cap-padding-x);color:var(--bs-card-cap-color);background-color:var(--bs-card-cap-bg);border-top:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card-header-tabs{margin-right:calc(-0.5*var(--bs-card-cap-padding-x));margin-bottom:calc(-1*var(--bs-card-cap-padding-y));margin-left:calc(-0.5*var(--bs-card-cap-padding-x));border-bottom:0}.card-header-tabs .nav-link.active{background-color:var(--bs-card-bg);border-bottom-color:var(--bs-card-bg)}.card-header-pills{margin-right:calc(-0.5*var(--bs-card-cap-padding-x));margin-left:calc(-0.5*var(--bs-card-cap-padding-x))}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:var(--bs-card-img-overlay-padding)}.card-img,.card-img-top,.card-img-bottom{width:100%}.card-group>.card{margin-bottom:var(--bs-card-group-margin)}@media(min-width: 576px){.card-group{display:flex;display:-webkit-flex;flex-flow:row wrap;-webkit-flex-flow:row wrap}.card-group>.card{flex:1 0 0%;-webkit-flex:1 0 0%;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}}.accordion{--bs-accordion-color: #000000;--bs-accordion-bg: #FFFFFF;--bs-accordion-transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, border-radius 0.15s ease;--bs-accordion-border-color: #dee2e6;--bs-accordion-border-width: 1px;--bs-accordion-border-radius: 0.25rem;--bs-accordion-inner-border-radius: calc(0.25rem - 1px);--bs-accordion-btn-padding-x: 1.25rem;--bs-accordion-btn-padding-y: 1rem;--bs-accordion-btn-color: #000000;--bs-accordion-btn-bg: #FFFFFF;--bs-accordion-btn-icon: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23000000'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e");--bs-accordion-btn-icon-width: 1.25rem;--bs-accordion-btn-icon-transform: rotate(-180deg);--bs-accordion-btn-icon-transition: transform 0.2s ease-in-out;--bs-accordion-btn-active-icon: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%235d5d65'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e");--bs-accordion-btn-focus-border-color: #f4f4fe;--bs-accordion-btn-focus-box-shadow: 0 0 0 0.25rem rgba(232, 232, 252, 0.25);--bs-accordion-body-padding-x: 1.25rem;--bs-accordion-body-padding-y: 1rem;--bs-accordion-active-color: #5d5d65;--bs-accordion-active-bg: #fafafe}.accordion-button{position:relative;display:flex;display:-webkit-flex;align-items:center;-webkit-align-items:center;width:100%;padding:var(--bs-accordion-btn-padding-y) var(--bs-accordion-btn-padding-x);font-size:1rem;color:var(--bs-accordion-btn-color);text-align:left;background-color:var(--bs-accordion-btn-bg);border:0;overflow-anchor:none;transition:var(--bs-accordion-transition)}@media(prefers-reduced-motion: reduce){.accordion-button{transition:none}}.accordion-button:not(.collapsed){color:var(--bs-accordion-active-color);background-color:var(--bs-accordion-active-bg);box-shadow:inset 0 calc(-1*var(--bs-accordion-border-width)) 0 var(--bs-accordion-border-color)}.accordion-button:not(.collapsed)::after{background-image:var(--bs-accordion-btn-active-icon);transform:var(--bs-accordion-btn-icon-transform)}.accordion-button::after{flex-shrink:0;-webkit-flex-shrink:0;width:var(--bs-accordion-btn-icon-width);height:var(--bs-accordion-btn-icon-width);margin-left:auto;content:"";background-image:var(--bs-accordion-btn-icon);background-repeat:no-repeat;background-size:var(--bs-accordion-btn-icon-width);transition:var(--bs-accordion-btn-icon-transition)}@media(prefers-reduced-motion: reduce){.accordion-button::after{transition:none}}.accordion-button:hover{z-index:2}.accordion-button:focus{z-index:3;border-color:var(--bs-accordion-btn-focus-border-color);outline:0;box-shadow:var(--bs-accordion-btn-focus-box-shadow)}.accordion-header{margin-bottom:0}.accordion-item{color:var(--bs-accordion-color);background-color:var(--bs-accordion-bg);border:var(--bs-accordion-border-width) solid var(--bs-accordion-border-color)}.accordion-item:not(:first-of-type){border-top:0}.accordion-body{padding:var(--bs-accordion-body-padding-y) var(--bs-accordion-body-padding-x)}.accordion-flush .accordion-collapse{border-width:0}.accordion-flush .accordion-item{border-right:0;border-left:0}.accordion-flush .accordion-item:first-child{border-top:0}.accordion-flush .accordion-item:last-child{border-bottom:0}[data-bs-theme=dark] .accordion-button::after{--bs-accordion-btn-icon: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23f1f1fd'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e");--bs-accordion-btn-active-icon: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23f1f1fd'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e")}.breadcrumb{--bs-breadcrumb-padding-x: 0;--bs-breadcrumb-padding-y: 0;--bs-breadcrumb-margin-bottom: 1rem;--bs-breadcrumb-bg: ;--bs-breadcrumb-border-radius: ;--bs-breadcrumb-divider-color: rgba(0, 0, 0, 0.75);--bs-breadcrumb-item-padding-x: 0.5rem;--bs-breadcrumb-item-active-color: rgba(0, 0, 0, 0.75);display:flex;display:-webkit-flex;flex-wrap:wrap;-webkit-flex-wrap:wrap;padding:var(--bs-breadcrumb-padding-y) var(--bs-breadcrumb-padding-x);margin-bottom:var(--bs-breadcrumb-margin-bottom);font-size:var(--bs-breadcrumb-font-size);list-style:none;background-color:var(--bs-breadcrumb-bg)}.breadcrumb-item+.breadcrumb-item{padding-left:var(--bs-breadcrumb-item-padding-x)}.breadcrumb-item+.breadcrumb-item::before{float:left;padding-right:var(--bs-breadcrumb-item-padding-x);color:var(--bs-breadcrumb-divider-color);content:var(--bs-breadcrumb-divider, ">") /* rtl: var(--bs-breadcrumb-divider, ">") */}.breadcrumb-item.active{color:var(--bs-breadcrumb-item-active-color)}.pagination{--bs-pagination-padding-x: 0.75rem;--bs-pagination-padding-y: 0.375rem;--bs-pagination-font-size:1rem;--bs-pagination-color: #4040BF;--bs-pagination-bg: #FFFFFF;--bs-pagination-border-width: 1px;--bs-pagination-border-color: #dee2e6;--bs-pagination-border-radius: 0.25rem;--bs-pagination-hover-color: #333399;--bs-pagination-hover-bg: #f8f9fa;--bs-pagination-hover-border-color: #dee2e6;--bs-pagination-focus-color: #333399;--bs-pagination-focus-bg: #e9ecef;--bs-pagination-focus-box-shadow: 0 0 0 0.25rem rgba(232, 232, 252, 0.25);--bs-pagination-active-color: #000;--bs-pagination-active-bg: #E8E8FC;--bs-pagination-active-border-color: #E8E8FC;--bs-pagination-disabled-color: rgba(0, 0, 0, 0.75);--bs-pagination-disabled-bg: #e9ecef;--bs-pagination-disabled-border-color: #dee2e6;display:flex;display:-webkit-flex;padding-left:0;list-style:none}.page-link{position:relative;display:block;padding:var(--bs-pagination-padding-y) var(--bs-pagination-padding-x);font-size:var(--bs-pagination-font-size);color:var(--bs-pagination-color);text-decoration:none;-webkit-text-decoration:none;-moz-text-decoration:none;-ms-text-decoration:none;-o-text-decoration:none;background-color:var(--bs-pagination-bg);border:var(--bs-pagination-border-width) solid var(--bs-pagination-border-color);transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media(prefers-reduced-motion: reduce){.page-link{transition:none}}.page-link:hover{z-index:2;color:var(--bs-pagination-hover-color);background-color:var(--bs-pagination-hover-bg);border-color:var(--bs-pagination-hover-border-color)}.page-link:focus{z-index:3;color:var(--bs-pagination-focus-color);background-color:var(--bs-pagination-focus-bg);outline:0;box-shadow:var(--bs-pagination-focus-box-shadow)}.page-link.active,.active>.page-link{z-index:3;color:var(--bs-pagination-active-color);background-color:var(--bs-pagination-active-bg);border-color:var(--bs-pagination-active-border-color)}.page-link.disabled,.disabled>.page-link{color:var(--bs-pagination-disabled-color);pointer-events:none;background-color:var(--bs-pagination-disabled-bg);border-color:var(--bs-pagination-disabled-border-color)}.page-item:not(:first-child) .page-link{margin-left:calc(1px*-1)}.pagination-lg{--bs-pagination-padding-x: 1.5rem;--bs-pagination-padding-y: 0.75rem;--bs-pagination-font-size:1.25rem;--bs-pagination-border-radius: 0.5rem}.pagination-sm{--bs-pagination-padding-x: 0.5rem;--bs-pagination-padding-y: 0.25rem;--bs-pagination-font-size:0.875rem;--bs-pagination-border-radius: 0.2em}.badge{--bs-badge-padding-x: 0.65em;--bs-badge-padding-y: 0.35em;--bs-badge-font-size:0.75em;--bs-badge-font-weight: 700;--bs-badge-color: #fff;--bs-badge-border-radius: 0.25rem;display:inline-block;padding:var(--bs-badge-padding-y) var(--bs-badge-padding-x);font-size:var(--bs-badge-font-size);font-weight:var(--bs-badge-font-weight);line-height:1;color:var(--bs-badge-color);text-align:center;white-space:nowrap;vertical-align:baseline}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.alert{--bs-alert-bg: transparent;--bs-alert-padding-x: 1rem;--bs-alert-padding-y: 1rem;--bs-alert-margin-bottom: 1rem;--bs-alert-color: inherit;--bs-alert-border-color: transparent;--bs-alert-border: 0 solid var(--bs-alert-border-color);--bs-alert-border-radius: 0.25rem;--bs-alert-link-color: inherit;position:relative;padding:var(--bs-alert-padding-y) var(--bs-alert-padding-x);margin-bottom:var(--bs-alert-margin-bottom);color:var(--bs-alert-color);background-color:var(--bs-alert-bg);border:var(--bs-alert-border)}.alert-heading{color:inherit}.alert-link{font-weight:700;color:var(--bs-alert-link-color)}.alert-dismissible{padding-right:3rem}.alert-dismissible .btn-close{position:absolute;top:0;right:0;z-index:2;padding:1.25rem 1rem}.alert-default{--bs-alert-color: var(--bs-default-text-emphasis);--bs-alert-bg: var(--bs-default-bg-subtle);--bs-alert-border-color: var(--bs-default-border-subtle);--bs-alert-link-color: var(--bs-default-text-emphasis)}.alert-primary{--bs-alert-color: var(--bs-primary-text-emphasis);--bs-alert-bg: var(--bs-primary-bg-subtle);--bs-alert-border-color: var(--bs-primary-border-subtle);--bs-alert-link-color: var(--bs-primary-text-emphasis)}.alert-secondary{--bs-alert-color: var(--bs-secondary-text-emphasis);--bs-alert-bg: var(--bs-secondary-bg-subtle);--bs-alert-border-color: var(--bs-secondary-border-subtle);--bs-alert-link-color: var(--bs-secondary-text-emphasis)}.alert-success{--bs-alert-color: var(--bs-success-text-emphasis);--bs-alert-bg: var(--bs-success-bg-subtle);--bs-alert-border-color: var(--bs-success-border-subtle);--bs-alert-link-color: var(--bs-success-text-emphasis)}.alert-info{--bs-alert-color: var(--bs-info-text-emphasis);--bs-alert-bg: var(--bs-info-bg-subtle);--bs-alert-border-color: var(--bs-info-border-subtle);--bs-alert-link-color: var(--bs-info-text-emphasis)}.alert-warning{--bs-alert-color: var(--bs-warning-text-emphasis);--bs-alert-bg: var(--bs-warning-bg-subtle);--bs-alert-border-color: var(--bs-warning-border-subtle);--bs-alert-link-color: var(--bs-warning-text-emphasis)}.alert-danger{--bs-alert-color: var(--bs-danger-text-emphasis);--bs-alert-bg: var(--bs-danger-bg-subtle);--bs-alert-border-color: var(--bs-danger-border-subtle);--bs-alert-link-color: var(--bs-danger-text-emphasis)}.alert-light{--bs-alert-color: var(--bs-light-text-emphasis);--bs-alert-bg: var(--bs-light-bg-subtle);--bs-alert-border-color: var(--bs-light-border-subtle);--bs-alert-link-color: var(--bs-light-text-emphasis)}.alert-dark{--bs-alert-color: var(--bs-dark-text-emphasis);--bs-alert-bg: var(--bs-dark-bg-subtle);--bs-alert-border-color: var(--bs-dark-border-subtle);--bs-alert-link-color: var(--bs-dark-text-emphasis)}@keyframes progress-bar-stripes{0%{background-position-x:.5rem}}.progress,.progress-stacked{--bs-progress-height: 0.5rem;--bs-progress-font-size:0.75rem;--bs-progress-bg: #e9ecef;--bs-progress-border-radius: 0.25rem;--bs-progress-box-shadow: inset 0 1px 2px rgba(0, 0, 0, 0.075);--bs-progress-bar-color: #fff;--bs-progress-bar-bg: #E8E8FC;--bs-progress-bar-transition: width 0.6s ease;display:flex;display:-webkit-flex;height:var(--bs-progress-height);overflow:hidden;font-size:var(--bs-progress-font-size);background-color:var(--bs-progress-bg)}.progress-bar{display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;justify-content:center;-webkit-justify-content:center;overflow:hidden;color:var(--bs-progress-bar-color);text-align:center;white-space:nowrap;background-color:var(--bs-progress-bar-bg);transition:var(--bs-progress-bar-transition)}@media(prefers-reduced-motion: reduce){.progress-bar{transition:none}}.progress-bar-striped{background-image:linear-gradient(45deg, rgba(255, 255, 255, 0.15) 25%, transparent 25%, transparent 50%, rgba(255, 255, 255, 0.15) 50%, rgba(255, 255, 255, 0.15) 75%, transparent 75%, transparent);background-size:var(--bs-progress-height) var(--bs-progress-height)}.progress-stacked>.progress{overflow:visible}.progress-stacked>.progress>.progress-bar{width:100%}.progress-bar-animated{animation:1s linear infinite progress-bar-stripes}@media(prefers-reduced-motion: reduce){.progress-bar-animated{animation:none}}.list-group{--bs-list-group-color: #000000;--bs-list-group-bg: #FFFFFF;--bs-list-group-border-color: #dee2e6;--bs-list-group-border-width: 1px;--bs-list-group-border-radius: 0.25rem;--bs-list-group-item-padding-x: 1rem;--bs-list-group-item-padding-y: 0.5rem;--bs-list-group-action-color: rgba(0, 0, 0, 0.75);--bs-list-group-action-hover-color: #000;--bs-list-group-action-hover-bg: #f8f9fa;--bs-list-group-action-active-color: #000000;--bs-list-group-action-active-bg: #e9ecef;--bs-list-group-disabled-color: rgba(0, 0, 0, 0.75);--bs-list-group-disabled-bg: #FFFFFF;--bs-list-group-active-color: #000;--bs-list-group-active-bg: #E8E8FC;--bs-list-group-active-border-color: #E8E8FC;display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;padding-left:0;margin-bottom:0}.list-group-numbered{list-style-type:none;counter-reset:section}.list-group-numbered>.list-group-item::before{content:counters(section, ".") ". ";counter-increment:section}.list-group-item-action{width:100%;color:var(--bs-list-group-action-color);text-align:inherit}.list-group-item-action:hover,.list-group-item-action:focus{z-index:1;color:var(--bs-list-group-action-hover-color);text-decoration:none;background-color:var(--bs-list-group-action-hover-bg)}.list-group-item-action:active{color:var(--bs-list-group-action-active-color);background-color:var(--bs-list-group-action-active-bg)}.list-group-item{position:relative;display:block;padding:var(--bs-list-group-item-padding-y) var(--bs-list-group-item-padding-x);color:var(--bs-list-group-color);text-decoration:none;-webkit-text-decoration:none;-moz-text-decoration:none;-ms-text-decoration:none;-o-text-decoration:none;background-color:var(--bs-list-group-bg);border:var(--bs-list-group-border-width) solid var(--bs-list-group-border-color)}.list-group-item.disabled,.list-group-item:disabled{color:var(--bs-list-group-disabled-color);pointer-events:none;background-color:var(--bs-list-group-disabled-bg)}.list-group-item.active{z-index:2;color:var(--bs-list-group-active-color);background-color:var(--bs-list-group-active-bg);border-color:var(--bs-list-group-active-border-color)}.list-group-item+.list-group-item{border-top-width:0}.list-group-item+.list-group-item.active{margin-top:calc(-1*var(--bs-list-group-border-width));border-top-width:var(--bs-list-group-border-width)}.list-group-horizontal{flex-direction:row;-webkit-flex-direction:row}.list-group-horizontal>.list-group-item.active{margin-top:0}.list-group-horizontal>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal>.list-group-item+.list-group-item.active{margin-left:calc(-1*var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}@media(min-width: 576px){.list-group-horizontal-sm{flex-direction:row;-webkit-flex-direction:row}.list-group-horizontal-sm>.list-group-item.active{margin-top:0}.list-group-horizontal-sm>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-sm>.list-group-item+.list-group-item.active{margin-left:calc(-1*var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media(min-width: 768px){.list-group-horizontal-md{flex-direction:row;-webkit-flex-direction:row}.list-group-horizontal-md>.list-group-item.active{margin-top:0}.list-group-horizontal-md>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-md>.list-group-item+.list-group-item.active{margin-left:calc(-1*var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media(min-width: 992px){.list-group-horizontal-lg{flex-direction:row;-webkit-flex-direction:row}.list-group-horizontal-lg>.list-group-item.active{margin-top:0}.list-group-horizontal-lg>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-lg>.list-group-item+.list-group-item.active{margin-left:calc(-1*var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media(min-width: 1200px){.list-group-horizontal-xl{flex-direction:row;-webkit-flex-direction:row}.list-group-horizontal-xl>.list-group-item.active{margin-top:0}.list-group-horizontal-xl>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-xl>.list-group-item+.list-group-item.active{margin-left:calc(-1*var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media(min-width: 1400px){.list-group-horizontal-xxl{flex-direction:row;-webkit-flex-direction:row}.list-group-horizontal-xxl>.list-group-item.active{margin-top:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item.active{margin-left:calc(-1*var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}.list-group-flush>.list-group-item{border-width:0 0 var(--bs-list-group-border-width)}.list-group-flush>.list-group-item:last-child{border-bottom-width:0}.list-group-item-default{--bs-list-group-color: var(--bs-default-text-emphasis);--bs-list-group-bg: var(--bs-default-bg-subtle);--bs-list-group-border-color: var(--bs-default-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-default-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-default-border-subtle);--bs-list-group-active-color: var(--bs-default-bg-subtle);--bs-list-group-active-bg: var(--bs-default-text-emphasis);--bs-list-group-active-border-color: var(--bs-default-text-emphasis)}.list-group-item-primary{--bs-list-group-color: var(--bs-primary-text-emphasis);--bs-list-group-bg: var(--bs-primary-bg-subtle);--bs-list-group-border-color: var(--bs-primary-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-primary-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-primary-border-subtle);--bs-list-group-active-color: var(--bs-primary-bg-subtle);--bs-list-group-active-bg: var(--bs-primary-text-emphasis);--bs-list-group-active-border-color: var(--bs-primary-text-emphasis)}.list-group-item-secondary{--bs-list-group-color: var(--bs-secondary-text-emphasis);--bs-list-group-bg: var(--bs-secondary-bg-subtle);--bs-list-group-border-color: var(--bs-secondary-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-secondary-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-secondary-border-subtle);--bs-list-group-active-color: var(--bs-secondary-bg-subtle);--bs-list-group-active-bg: var(--bs-secondary-text-emphasis);--bs-list-group-active-border-color: var(--bs-secondary-text-emphasis)}.list-group-item-success{--bs-list-group-color: var(--bs-success-text-emphasis);--bs-list-group-bg: var(--bs-success-bg-subtle);--bs-list-group-border-color: var(--bs-success-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-success-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-success-border-subtle);--bs-list-group-active-color: var(--bs-success-bg-subtle);--bs-list-group-active-bg: var(--bs-success-text-emphasis);--bs-list-group-active-border-color: var(--bs-success-text-emphasis)}.list-group-item-info{--bs-list-group-color: var(--bs-info-text-emphasis);--bs-list-group-bg: var(--bs-info-bg-subtle);--bs-list-group-border-color: var(--bs-info-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-info-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-info-border-subtle);--bs-list-group-active-color: var(--bs-info-bg-subtle);--bs-list-group-active-bg: var(--bs-info-text-emphasis);--bs-list-group-active-border-color: var(--bs-info-text-emphasis)}.list-group-item-warning{--bs-list-group-color: var(--bs-warning-text-emphasis);--bs-list-group-bg: var(--bs-warning-bg-subtle);--bs-list-group-border-color: var(--bs-warning-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-warning-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-warning-border-subtle);--bs-list-group-active-color: var(--bs-warning-bg-subtle);--bs-list-group-active-bg: var(--bs-warning-text-emphasis);--bs-list-group-active-border-color: var(--bs-warning-text-emphasis)}.list-group-item-danger{--bs-list-group-color: var(--bs-danger-text-emphasis);--bs-list-group-bg: var(--bs-danger-bg-subtle);--bs-list-group-border-color: var(--bs-danger-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-danger-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-danger-border-subtle);--bs-list-group-active-color: var(--bs-danger-bg-subtle);--bs-list-group-active-bg: var(--bs-danger-text-emphasis);--bs-list-group-active-border-color: var(--bs-danger-text-emphasis)}.list-group-item-light{--bs-list-group-color: var(--bs-light-text-emphasis);--bs-list-group-bg: var(--bs-light-bg-subtle);--bs-list-group-border-color: var(--bs-light-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-light-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-light-border-subtle);--bs-list-group-active-color: var(--bs-light-bg-subtle);--bs-list-group-active-bg: var(--bs-light-text-emphasis);--bs-list-group-active-border-color: var(--bs-light-text-emphasis)}.list-group-item-dark{--bs-list-group-color: var(--bs-dark-text-emphasis);--bs-list-group-bg: var(--bs-dark-bg-subtle);--bs-list-group-border-color: var(--bs-dark-border-subtle);--bs-list-group-action-hover-color: var(--bs-emphasis-color);--bs-list-group-action-hover-bg: var(--bs-dark-border-subtle);--bs-list-group-action-active-color: var(--bs-emphasis-color);--bs-list-group-action-active-bg: var(--bs-dark-border-subtle);--bs-list-group-active-color: var(--bs-dark-bg-subtle);--bs-list-group-active-bg: var(--bs-dark-text-emphasis);--bs-list-group-active-border-color: var(--bs-dark-text-emphasis)}.btn-close{--bs-btn-close-color: #000;--bs-btn-close-bg: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23000'%3e%3cpath d='M.293.293a1 1 0 0 1 1.414 0L8 6.586 14.293.293a1 1 0 1 1 1.414 1.414L9.414 8l6.293 6.293a1 1 0 0 1-1.414 1.414L8 9.414l-6.293 6.293a1 1 0 0 1-1.414-1.414L6.586 8 .293 1.707a1 1 0 0 1 0-1.414z'/%3e%3c/svg%3e");--bs-btn-close-opacity: 0.5;--bs-btn-close-hover-opacity: 0.75;--bs-btn-close-focus-shadow: 0 0 0 0.25rem rgba(232, 232, 252, 0.25);--bs-btn-close-focus-opacity: 1;--bs-btn-close-disabled-opacity: 0.25;--bs-btn-close-white-filter: invert(1) grayscale(100%) brightness(200%);box-sizing:content-box;width:1em;height:1em;padding:.25em .25em;color:var(--bs-btn-close-color);background:rgba(0,0,0,0) var(--bs-btn-close-bg) center/1em auto no-repeat;border:0;opacity:var(--bs-btn-close-opacity)}.btn-close:hover{color:var(--bs-btn-close-color);text-decoration:none;opacity:var(--bs-btn-close-hover-opacity)}.btn-close:focus{outline:0;box-shadow:var(--bs-btn-close-focus-shadow);opacity:var(--bs-btn-close-focus-opacity)}.btn-close:disabled,.btn-close.disabled{pointer-events:none;user-select:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;-o-user-select:none;opacity:var(--bs-btn-close-disabled-opacity)}.btn-close-white{filter:var(--bs-btn-close-white-filter)}[data-bs-theme=dark] .btn-close{filter:var(--bs-btn-close-white-filter)}.toast{--bs-toast-zindex: 1090;--bs-toast-padding-x: 0.75rem;--bs-toast-padding-y: 0.5rem;--bs-toast-spacing: 1.5rem;--bs-toast-max-width: 350px;--bs-toast-font-size:0.875rem;--bs-toast-color: ;--bs-toast-bg: rgba(255, 255, 255, 0.85);--bs-toast-border-width: 1px;--bs-toast-border-color: rgba(0, 0, 0, 0.175);--bs-toast-border-radius: 0.25rem;--bs-toast-box-shadow: 0 0.5rem 1rem rgba(0, 0, 0, 0.15);--bs-toast-header-color: rgba(0, 0, 0, 0.75);--bs-toast-header-bg: rgba(255, 255, 255, 0.85);--bs-toast-header-border-color: rgba(0, 0, 0, 0.175);width:var(--bs-toast-max-width);max-width:100%;font-size:var(--bs-toast-font-size);color:var(--bs-toast-color);pointer-events:auto;background-color:var(--bs-toast-bg);background-clip:padding-box;border:var(--bs-toast-border-width) solid var(--bs-toast-border-color);box-shadow:var(--bs-toast-box-shadow)}.toast.showing{opacity:0}.toast:not(.show){display:none}.toast-container{--bs-toast-zindex: 1090;position:absolute;z-index:var(--bs-toast-zindex);width:max-content;width:-webkit-max-content;width:-moz-max-content;width:-ms-max-content;width:-o-max-content;max-width:100%;pointer-events:none}.toast-container>:not(:last-child){margin-bottom:var(--bs-toast-spacing)}.toast-header{display:flex;display:-webkit-flex;align-items:center;-webkit-align-items:center;padding:var(--bs-toast-padding-y) var(--bs-toast-padding-x);color:var(--bs-toast-header-color);background-color:var(--bs-toast-header-bg);background-clip:padding-box;border-bottom:var(--bs-toast-border-width) solid var(--bs-toast-header-border-color)}.toast-header .btn-close{margin-right:calc(-0.5*var(--bs-toast-padding-x));margin-left:var(--bs-toast-padding-x)}.toast-body{padding:var(--bs-toast-padding-x);word-wrap:break-word}.modal{--bs-modal-zindex: 1055;--bs-modal-width: 500px;--bs-modal-padding: 1rem;--bs-modal-margin: 0.5rem;--bs-modal-color: ;--bs-modal-bg: #FFFFFF;--bs-modal-border-color: rgba(0, 0, 0, 0.175);--bs-modal-border-width: 1px;--bs-modal-border-radius: 0.5rem;--bs-modal-box-shadow: 0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);--bs-modal-inner-border-radius: calc(0.5rem - 1px);--bs-modal-header-padding-x: 1rem;--bs-modal-header-padding-y: 1rem;--bs-modal-header-padding: 1rem 1rem;--bs-modal-header-border-color: #dee2e6;--bs-modal-header-border-width: 1px;--bs-modal-title-line-height: 1.5;--bs-modal-footer-gap: 0.5rem;--bs-modal-footer-bg: ;--bs-modal-footer-border-color: #dee2e6;--bs-modal-footer-border-width: 1px;position:fixed;top:0;left:0;z-index:var(--bs-modal-zindex);display:none;width:100%;height:100%;overflow-x:hidden;overflow-y:auto;outline:0}.modal-dialog{position:relative;width:auto;margin:var(--bs-modal-margin);pointer-events:none}.modal.fade .modal-dialog{transition:transform .3s ease-out;transform:translate(0, -50px)}@media(prefers-reduced-motion: reduce){.modal.fade .modal-dialog{transition:none}}.modal.show .modal-dialog{transform:none}.modal.modal-static .modal-dialog{transform:scale(1.02)}.modal-dialog-scrollable{height:calc(100% - var(--bs-modal-margin)*2)}.modal-dialog-scrollable .modal-content{max-height:100%;overflow:hidden}.modal-dialog-scrollable .modal-body{overflow-y:auto}.modal-dialog-centered{display:flex;display:-webkit-flex;align-items:center;-webkit-align-items:center;min-height:calc(100% - var(--bs-modal-margin)*2)}.modal-content{position:relative;display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;width:100%;color:var(--bs-modal-color);pointer-events:auto;background-color:var(--bs-modal-bg);background-clip:padding-box;border:var(--bs-modal-border-width) solid var(--bs-modal-border-color);outline:0}.modal-backdrop{--bs-backdrop-zindex: 1050;--bs-backdrop-bg: #000;--bs-backdrop-opacity: 0.5;position:fixed;top:0;left:0;z-index:var(--bs-backdrop-zindex);width:100vw;height:100vh;background-color:var(--bs-backdrop-bg)}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:var(--bs-backdrop-opacity)}.modal-header{display:flex;display:-webkit-flex;flex-shrink:0;-webkit-flex-shrink:0;align-items:center;-webkit-align-items:center;justify-content:space-between;-webkit-justify-content:space-between;padding:var(--bs-modal-header-padding);border-bottom:var(--bs-modal-header-border-width) solid var(--bs-modal-header-border-color)}.modal-header .btn-close{padding:calc(var(--bs-modal-header-padding-y)*.5) calc(var(--bs-modal-header-padding-x)*.5);margin:calc(-0.5*var(--bs-modal-header-padding-y)) calc(-0.5*var(--bs-modal-header-padding-x)) calc(-0.5*var(--bs-modal-header-padding-y)) auto}.modal-title{margin-bottom:0;line-height:var(--bs-modal-title-line-height)}.modal-body{position:relative;flex:1 1 auto;-webkit-flex:1 1 auto;padding:var(--bs-modal-padding)}.modal-footer{display:flex;display:-webkit-flex;flex-shrink:0;-webkit-flex-shrink:0;flex-wrap:wrap;-webkit-flex-wrap:wrap;align-items:center;-webkit-align-items:center;justify-content:flex-end;-webkit-justify-content:flex-end;padding:calc(var(--bs-modal-padding) - var(--bs-modal-footer-gap)*.5);background-color:var(--bs-modal-footer-bg);border-top:var(--bs-modal-footer-border-width) solid var(--bs-modal-footer-border-color)}.modal-footer>*{margin:calc(var(--bs-modal-footer-gap)*.5)}@media(min-width: 576px){.modal{--bs-modal-margin: 1.75rem;--bs-modal-box-shadow: 0 0.5rem 1rem rgba(0, 0, 0, 0.15)}.modal-dialog{max-width:var(--bs-modal-width);margin-right:auto;margin-left:auto}.modal-sm{--bs-modal-width: 300px}}@media(min-width: 992px){.modal-lg,.modal-xl{--bs-modal-width: 800px}}@media(min-width: 1200px){.modal-xl{--bs-modal-width: 1140px}}.modal-fullscreen{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen .modal-content{height:100%;border:0}.modal-fullscreen .modal-body{overflow-y:auto}@media(max-width: 575.98px){.modal-fullscreen-sm-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-sm-down .modal-content{height:100%;border:0}.modal-fullscreen-sm-down .modal-body{overflow-y:auto}}@media(max-width: 767.98px){.modal-fullscreen-md-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-md-down .modal-content{height:100%;border:0}.modal-fullscreen-md-down .modal-body{overflow-y:auto}}@media(max-width: 991.98px){.modal-fullscreen-lg-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-lg-down .modal-content{height:100%;border:0}.modal-fullscreen-lg-down .modal-body{overflow-y:auto}}@media(max-width: 1199.98px){.modal-fullscreen-xl-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-xl-down .modal-content{height:100%;border:0}.modal-fullscreen-xl-down .modal-body{overflow-y:auto}}@media(max-width: 1399.98px){.modal-fullscreen-xxl-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-xxl-down .modal-content{height:100%;border:0}.modal-fullscreen-xxl-down .modal-body{overflow-y:auto}}.tooltip{--bs-tooltip-zindex: 1080;--bs-tooltip-max-width: 200px;--bs-tooltip-padding-x: 0.5rem;--bs-tooltip-padding-y: 0.25rem;--bs-tooltip-margin: ;--bs-tooltip-font-size:0.875rem;--bs-tooltip-color: #FFFFFF;--bs-tooltip-bg: #000;--bs-tooltip-border-radius: 0.25rem;--bs-tooltip-opacity: 0.9;--bs-tooltip-arrow-width: 0.8rem;--bs-tooltip-arrow-height: 0.4rem;z-index:var(--bs-tooltip-zindex);display:block;margin:var(--bs-tooltip-margin);font-family:"Source Sans Pro",-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;white-space:normal;word-spacing:normal;line-break:auto;font-size:var(--bs-tooltip-font-size);word-wrap:break-word;opacity:0}.tooltip.show{opacity:var(--bs-tooltip-opacity)}.tooltip .tooltip-arrow{display:block;width:var(--bs-tooltip-arrow-width);height:var(--bs-tooltip-arrow-height)}.tooltip .tooltip-arrow::before{position:absolute;content:"";border-color:rgba(0,0,0,0);border-style:solid}.bs-tooltip-top .tooltip-arrow,.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow{bottom:calc(-1*var(--bs-tooltip-arrow-height))}.bs-tooltip-top .tooltip-arrow::before,.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow::before{top:-1px;border-width:var(--bs-tooltip-arrow-height) calc(var(--bs-tooltip-arrow-width)*.5) 0;border-top-color:var(--bs-tooltip-bg)}.bs-tooltip-end .tooltip-arrow,.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow{left:calc(-1*var(--bs-tooltip-arrow-height));width:var(--bs-tooltip-arrow-height);height:var(--bs-tooltip-arrow-width)}.bs-tooltip-end .tooltip-arrow::before,.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow::before{right:-1px;border-width:calc(var(--bs-tooltip-arrow-width)*.5) var(--bs-tooltip-arrow-height) calc(var(--bs-tooltip-arrow-width)*.5) 0;border-right-color:var(--bs-tooltip-bg)}.bs-tooltip-bottom .tooltip-arrow,.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow{top:calc(-1*var(--bs-tooltip-arrow-height))}.bs-tooltip-bottom .tooltip-arrow::before,.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow::before{bottom:-1px;border-width:0 calc(var(--bs-tooltip-arrow-width)*.5) var(--bs-tooltip-arrow-height);border-bottom-color:var(--bs-tooltip-bg)}.bs-tooltip-start .tooltip-arrow,.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow{right:calc(-1*var(--bs-tooltip-arrow-height));width:var(--bs-tooltip-arrow-height);height:var(--bs-tooltip-arrow-width)}.bs-tooltip-start .tooltip-arrow::before,.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow::before{left:-1px;border-width:calc(var(--bs-tooltip-arrow-width)*.5) 0 calc(var(--bs-tooltip-arrow-width)*.5) var(--bs-tooltip-arrow-height);border-left-color:var(--bs-tooltip-bg)}.tooltip-inner{max-width:var(--bs-tooltip-max-width);padding:var(--bs-tooltip-padding-y) var(--bs-tooltip-padding-x);color:var(--bs-tooltip-color);text-align:center;background-color:var(--bs-tooltip-bg)}.popover{--bs-popover-zindex: 1070;--bs-popover-max-width: 276px;--bs-popover-font-size:0.875rem;--bs-popover-bg: #FFFFFF;--bs-popover-border-width: 1px;--bs-popover-border-color: rgba(0, 0, 0, 0.175);--bs-popover-border-radius: 0.5rem;--bs-popover-inner-border-radius: calc(0.5rem - 1px);--bs-popover-box-shadow: 0 0.5rem 1rem rgba(0, 0, 0, 0.15);--bs-popover-header-padding-x: 1rem;--bs-popover-header-padding-y: 0.5rem;--bs-popover-header-font-size:1rem;--bs-popover-header-color: inherit;--bs-popover-header-bg: #e9ecef;--bs-popover-body-padding-x: 1rem;--bs-popover-body-padding-y: 1rem;--bs-popover-body-color: #000000;--bs-popover-arrow-width: 1rem;--bs-popover-arrow-height: 0.5rem;--bs-popover-arrow-border: var(--bs-popover-border-color);z-index:var(--bs-popover-zindex);display:block;max-width:var(--bs-popover-max-width);font-family:"Source Sans Pro",-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;white-space:normal;word-spacing:normal;line-break:auto;font-size:var(--bs-popover-font-size);word-wrap:break-word;background-color:var(--bs-popover-bg);background-clip:padding-box;border:var(--bs-popover-border-width) solid var(--bs-popover-border-color)}.popover .popover-arrow{display:block;width:var(--bs-popover-arrow-width);height:var(--bs-popover-arrow-height)}.popover .popover-arrow::before,.popover .popover-arrow::after{position:absolute;display:block;content:"";border-color:rgba(0,0,0,0);border-style:solid;border-width:0}.bs-popover-top>.popover-arrow,.bs-popover-auto[data-popper-placement^=top]>.popover-arrow{bottom:calc(-1*(var(--bs-popover-arrow-height)) - var(--bs-popover-border-width))}.bs-popover-top>.popover-arrow::before,.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before,.bs-popover-top>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after{border-width:var(--bs-popover-arrow-height) calc(var(--bs-popover-arrow-width)*.5) 0}.bs-popover-top>.popover-arrow::before,.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before{bottom:0;border-top-color:var(--bs-popover-arrow-border)}.bs-popover-top>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after{bottom:var(--bs-popover-border-width);border-top-color:var(--bs-popover-bg)}.bs-popover-end>.popover-arrow,.bs-popover-auto[data-popper-placement^=right]>.popover-arrow{left:calc(-1*(var(--bs-popover-arrow-height)) - var(--bs-popover-border-width));width:var(--bs-popover-arrow-height);height:var(--bs-popover-arrow-width)}.bs-popover-end>.popover-arrow::before,.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before,.bs-popover-end>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after{border-width:calc(var(--bs-popover-arrow-width)*.5) var(--bs-popover-arrow-height) calc(var(--bs-popover-arrow-width)*.5) 0}.bs-popover-end>.popover-arrow::before,.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before{left:0;border-right-color:var(--bs-popover-arrow-border)}.bs-popover-end>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after{left:var(--bs-popover-border-width);border-right-color:var(--bs-popover-bg)}.bs-popover-bottom>.popover-arrow,.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow{top:calc(-1*(var(--bs-popover-arrow-height)) - var(--bs-popover-border-width))}.bs-popover-bottom>.popover-arrow::before,.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before,.bs-popover-bottom>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after{border-width:0 calc(var(--bs-popover-arrow-width)*.5) var(--bs-popover-arrow-height)}.bs-popover-bottom>.popover-arrow::before,.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before{top:0;border-bottom-color:var(--bs-popover-arrow-border)}.bs-popover-bottom>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after{top:var(--bs-popover-border-width);border-bottom-color:var(--bs-popover-bg)}.bs-popover-bottom .popover-header::before,.bs-popover-auto[data-popper-placement^=bottom] .popover-header::before{position:absolute;top:0;left:50%;display:block;width:var(--bs-popover-arrow-width);margin-left:calc(-0.5*var(--bs-popover-arrow-width));content:"";border-bottom:var(--bs-popover-border-width) solid var(--bs-popover-header-bg)}.bs-popover-start>.popover-arrow,.bs-popover-auto[data-popper-placement^=left]>.popover-arrow{right:calc(-1*(var(--bs-popover-arrow-height)) - var(--bs-popover-border-width));width:var(--bs-popover-arrow-height);height:var(--bs-popover-arrow-width)}.bs-popover-start>.popover-arrow::before,.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before,.bs-popover-start>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after{border-width:calc(var(--bs-popover-arrow-width)*.5) 0 calc(var(--bs-popover-arrow-width)*.5) var(--bs-popover-arrow-height)}.bs-popover-start>.popover-arrow::before,.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before{right:0;border-left-color:var(--bs-popover-arrow-border)}.bs-popover-start>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after{right:var(--bs-popover-border-width);border-left-color:var(--bs-popover-bg)}.popover-header{padding:var(--bs-popover-header-padding-y) var(--bs-popover-header-padding-x);margin-bottom:0;font-size:var(--bs-popover-header-font-size);color:var(--bs-popover-header-color);background-color:var(--bs-popover-header-bg);border-bottom:var(--bs-popover-border-width) solid var(--bs-popover-border-color)}.popover-header:empty{display:none}.popover-body{padding:var(--bs-popover-body-padding-y) var(--bs-popover-body-padding-x);color:var(--bs-popover-body-color)}.carousel{position:relative}.carousel.pointer-event{touch-action:pan-y;-webkit-touch-action:pan-y;-moz-touch-action:pan-y;-ms-touch-action:pan-y;-o-touch-action:pan-y}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner::after{display:block;clear:both;content:""}.carousel-item{position:relative;display:none;float:left;width:100%;margin-right:-100%;backface-visibility:hidden;-webkit-backface-visibility:hidden;-moz-backface-visibility:hidden;-ms-backface-visibility:hidden;-o-backface-visibility:hidden;transition:transform .6s ease-in-out}@media(prefers-reduced-motion: reduce){.carousel-item{transition:none}}.carousel-item.active,.carousel-item-next,.carousel-item-prev{display:block}.carousel-item-next:not(.carousel-item-start),.active.carousel-item-end{transform:translateX(100%)}.carousel-item-prev:not(.carousel-item-end),.active.carousel-item-start{transform:translateX(-100%)}.carousel-fade .carousel-item{opacity:0;transition-property:opacity;transform:none}.carousel-fade .carousel-item.active,.carousel-fade .carousel-item-next.carousel-item-start,.carousel-fade .carousel-item-prev.carousel-item-end{z-index:1;opacity:1}.carousel-fade .active.carousel-item-start,.carousel-fade .active.carousel-item-end{z-index:0;opacity:0;transition:opacity 0s .6s}@media(prefers-reduced-motion: reduce){.carousel-fade .active.carousel-item-start,.carousel-fade .active.carousel-item-end{transition:none}}.carousel-control-prev,.carousel-control-next{position:absolute;top:0;bottom:0;z-index:1;display:flex;display:-webkit-flex;align-items:center;-webkit-align-items:center;justify-content:center;-webkit-justify-content:center;width:15%;padding:0;color:#fff;text-align:center;background:none;border:0;opacity:.5;transition:opacity .15s ease}@media(prefers-reduced-motion: reduce){.carousel-control-prev,.carousel-control-next{transition:none}}.carousel-control-prev:hover,.carousel-control-prev:focus,.carousel-control-next:hover,.carousel-control-next:focus{color:#fff;text-decoration:none;outline:0;opacity:.9}.carousel-control-prev{left:0}.carousel-control-next{right:0}.carousel-control-prev-icon,.carousel-control-next-icon{display:inline-block;width:2rem;height:2rem;background-repeat:no-repeat;background-position:50%;background-size:100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23fff'%3e%3cpath d='M11.354 1.646a.5.5 0 0 1 0 .708L5.707 8l5.647 5.646a.5.5 0 0 1-.708.708l-6-6a.5.5 0 0 1 0-.708l6-6a.5.5 0 0 1 .708 0z'/%3e%3c/svg%3e")}.carousel-control-next-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23fff'%3e%3cpath d='M4.646 1.646a.5.5 0 0 1 .708 0l6 6a.5.5 0 0 1 0 .708l-6 6a.5.5 0 0 1-.708-.708L10.293 8 4.646 2.354a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e")}.carousel-indicators{position:absolute;right:0;bottom:0;left:0;z-index:2;display:flex;display:-webkit-flex;justify-content:center;-webkit-justify-content:center;padding:0;margin-right:15%;margin-bottom:1rem;margin-left:15%}.carousel-indicators [data-bs-target]{box-sizing:content-box;flex:0 1 auto;-webkit-flex:0 1 auto;width:30px;height:3px;padding:0;margin-right:3px;margin-left:3px;text-indent:-999px;cursor:pointer;background-color:#fff;background-clip:padding-box;border:0;border-top:10px solid rgba(0,0,0,0);border-bottom:10px solid rgba(0,0,0,0);opacity:.5;transition:opacity .6s ease}@media(prefers-reduced-motion: reduce){.carousel-indicators [data-bs-target]{transition:none}}.carousel-indicators .active{opacity:1}.carousel-caption{position:absolute;right:15%;bottom:1.25rem;left:15%;padding-top:1.25rem;padding-bottom:1.25rem;color:#fff;text-align:center}.carousel-dark .carousel-control-prev-icon,.carousel-dark .carousel-control-next-icon{filter:invert(1) grayscale(100)}.carousel-dark .carousel-indicators [data-bs-target]{background-color:#000}.carousel-dark .carousel-caption{color:#000}[data-bs-theme=dark] .carousel .carousel-control-prev-icon,[data-bs-theme=dark] .carousel .carousel-control-next-icon,[data-bs-theme=dark].carousel .carousel-control-prev-icon,[data-bs-theme=dark].carousel .carousel-control-next-icon{filter:invert(1) grayscale(100)}[data-bs-theme=dark] .carousel .carousel-indicators [data-bs-target],[data-bs-theme=dark].carousel .carousel-indicators [data-bs-target]{background-color:#000}[data-bs-theme=dark] .carousel .carousel-caption,[data-bs-theme=dark].carousel .carousel-caption{color:#000}.spinner-grow,.spinner-border{display:inline-block;width:var(--bs-spinner-width);height:var(--bs-spinner-height);vertical-align:var(--bs-spinner-vertical-align);border-radius:50%;animation:var(--bs-spinner-animation-speed) linear infinite var(--bs-spinner-animation-name)}@keyframes spinner-border{to{transform:rotate(360deg) /* rtl:ignore */}}.spinner-border{--bs-spinner-width: 2rem;--bs-spinner-height: 2rem;--bs-spinner-vertical-align: -0.125em;--bs-spinner-border-width: 0.25em;--bs-spinner-animation-speed: 0.75s;--bs-spinner-animation-name: spinner-border;border:var(--bs-spinner-border-width) solid currentcolor;border-right-color:rgba(0,0,0,0)}.spinner-border-sm{--bs-spinner-width: 1rem;--bs-spinner-height: 1rem;--bs-spinner-border-width: 0.2em}@keyframes spinner-grow{0%{transform:scale(0)}50%{opacity:1;transform:none}}.spinner-grow{--bs-spinner-width: 2rem;--bs-spinner-height: 2rem;--bs-spinner-vertical-align: -0.125em;--bs-spinner-animation-speed: 0.75s;--bs-spinner-animation-name: spinner-grow;background-color:currentcolor;opacity:0}.spinner-grow-sm{--bs-spinner-width: 1rem;--bs-spinner-height: 1rem}@media(prefers-reduced-motion: reduce){.spinner-border,.spinner-grow{--bs-spinner-animation-speed: 1.5s}}.offcanvas,.offcanvas-xxl,.offcanvas-xl,.offcanvas-lg,.offcanvas-md,.offcanvas-sm{--bs-offcanvas-zindex: 1045;--bs-offcanvas-width: 400px;--bs-offcanvas-height: 30vh;--bs-offcanvas-padding-x: 1rem;--bs-offcanvas-padding-y: 1rem;--bs-offcanvas-color: #000000;--bs-offcanvas-bg: #FFFFFF;--bs-offcanvas-border-width: 1px;--bs-offcanvas-border-color: rgba(0, 0, 0, 0.175);--bs-offcanvas-box-shadow: 0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);--bs-offcanvas-transition: transform 0.3s ease-in-out;--bs-offcanvas-title-line-height: 1.5}@media(max-width: 575.98px){.offcanvas-sm{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media(max-width: 575.98px)and (prefers-reduced-motion: reduce){.offcanvas-sm{transition:none}}@media(max-width: 575.98px){.offcanvas-sm.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-sm.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-sm.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-sm.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-sm.showing,.offcanvas-sm.show:not(.hiding){transform:none}.offcanvas-sm.showing,.offcanvas-sm.hiding,.offcanvas-sm.show{visibility:visible}}@media(min-width: 576px){.offcanvas-sm{--bs-offcanvas-height: auto;--bs-offcanvas-border-width: 0;background-color:rgba(0,0,0,0) !important}.offcanvas-sm .offcanvas-header{display:none}.offcanvas-sm .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible;background-color:rgba(0,0,0,0) !important}}@media(max-width: 767.98px){.offcanvas-md{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media(max-width: 767.98px)and (prefers-reduced-motion: reduce){.offcanvas-md{transition:none}}@media(max-width: 767.98px){.offcanvas-md.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-md.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-md.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-md.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-md.showing,.offcanvas-md.show:not(.hiding){transform:none}.offcanvas-md.showing,.offcanvas-md.hiding,.offcanvas-md.show{visibility:visible}}@media(min-width: 768px){.offcanvas-md{--bs-offcanvas-height: auto;--bs-offcanvas-border-width: 0;background-color:rgba(0,0,0,0) !important}.offcanvas-md .offcanvas-header{display:none}.offcanvas-md .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible;background-color:rgba(0,0,0,0) !important}}@media(max-width: 991.98px){.offcanvas-lg{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media(max-width: 991.98px)and (prefers-reduced-motion: reduce){.offcanvas-lg{transition:none}}@media(max-width: 991.98px){.offcanvas-lg.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-lg.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-lg.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-lg.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-lg.showing,.offcanvas-lg.show:not(.hiding){transform:none}.offcanvas-lg.showing,.offcanvas-lg.hiding,.offcanvas-lg.show{visibility:visible}}@media(min-width: 992px){.offcanvas-lg{--bs-offcanvas-height: auto;--bs-offcanvas-border-width: 0;background-color:rgba(0,0,0,0) !important}.offcanvas-lg .offcanvas-header{display:none}.offcanvas-lg .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible;background-color:rgba(0,0,0,0) !important}}@media(max-width: 1199.98px){.offcanvas-xl{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media(max-width: 1199.98px)and (prefers-reduced-motion: reduce){.offcanvas-xl{transition:none}}@media(max-width: 1199.98px){.offcanvas-xl.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-xl.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-xl.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-xl.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-xl.showing,.offcanvas-xl.show:not(.hiding){transform:none}.offcanvas-xl.showing,.offcanvas-xl.hiding,.offcanvas-xl.show{visibility:visible}}@media(min-width: 1200px){.offcanvas-xl{--bs-offcanvas-height: auto;--bs-offcanvas-border-width: 0;background-color:rgba(0,0,0,0) !important}.offcanvas-xl .offcanvas-header{display:none}.offcanvas-xl .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible;background-color:rgba(0,0,0,0) !important}}@media(max-width: 1399.98px){.offcanvas-xxl{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media(max-width: 1399.98px)and (prefers-reduced-motion: reduce){.offcanvas-xxl{transition:none}}@media(max-width: 1399.98px){.offcanvas-xxl.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-xxl.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-xxl.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-xxl.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-xxl.showing,.offcanvas-xxl.show:not(.hiding){transform:none}.offcanvas-xxl.showing,.offcanvas-xxl.hiding,.offcanvas-xxl.show{visibility:visible}}@media(min-width: 1400px){.offcanvas-xxl{--bs-offcanvas-height: auto;--bs-offcanvas-border-width: 0;background-color:rgba(0,0,0,0) !important}.offcanvas-xxl .offcanvas-header{display:none}.offcanvas-xxl .offcanvas-body{display:flex;display:-webkit-flex;flex-grow:0;-webkit-flex-grow:0;padding:0;overflow-y:visible;background-color:rgba(0,0,0,0) !important}}.offcanvas{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;display:-webkit-flex;flex-direction:column;-webkit-flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}@media(prefers-reduced-motion: reduce){.offcanvas{transition:none}}.offcanvas.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas.showing,.offcanvas.show:not(.hiding){transform:none}.offcanvas.showing,.offcanvas.hiding,.offcanvas.show{visibility:visible}.offcanvas-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#000}.offcanvas-backdrop.fade{opacity:0}.offcanvas-backdrop.show{opacity:.5}.offcanvas-header{display:flex;display:-webkit-flex;align-items:center;-webkit-align-items:center;justify-content:space-between;-webkit-justify-content:space-between;padding:var(--bs-offcanvas-padding-y) var(--bs-offcanvas-padding-x)}.offcanvas-header .btn-close{padding:calc(var(--bs-offcanvas-padding-y)*.5) calc(var(--bs-offcanvas-padding-x)*.5);margin-top:calc(-0.5*var(--bs-offcanvas-padding-y));margin-right:calc(-0.5*var(--bs-offcanvas-padding-x));margin-bottom:calc(-0.5*var(--bs-offcanvas-padding-y))}.offcanvas-title{margin-bottom:0;line-height:var(--bs-offcanvas-title-line-height)}.offcanvas-body{flex-grow:1;-webkit-flex-grow:1;padding:var(--bs-offcanvas-padding-y) var(--bs-offcanvas-padding-x);overflow-y:auto}.placeholder{display:inline-block;min-height:1em;vertical-align:middle;cursor:wait;background-color:currentcolor;opacity:.5}.placeholder.btn::before{display:inline-block;content:""}.placeholder-xs{min-height:.6em}.placeholder-sm{min-height:.8em}.placeholder-lg{min-height:1.2em}.placeholder-glow .placeholder{animation:placeholder-glow 2s ease-in-out infinite}@keyframes placeholder-glow{50%{opacity:.2}}.placeholder-wave{mask-image:linear-gradient(130deg, #000 55%, rgba(0, 0, 0, 0.8) 75%, #000 95%);-webkit-mask-image:linear-gradient(130deg, #000 55%, rgba(0, 0, 0, 0.8) 75%, #000 95%);mask-size:200% 100%;-webkit-mask-size:200% 100%;animation:placeholder-wave 2s linear infinite}@keyframes placeholder-wave{100%{mask-position:-200% 0%;-webkit-mask-position:-200% 0%}}.clearfix::after{display:block;clear:both;content:""}.text-bg-default{color:#fff !important;background-color:RGBA(var(--bs-default-rgb), var(--bs-bg-opacity, 1)) !important}.text-bg-primary{color:#000 !important;background-color:RGBA(var(--bs-primary-rgb), var(--bs-bg-opacity, 1)) !important}.text-bg-secondary{color:#fff !important;background-color:RGBA(var(--bs-secondary-rgb), var(--bs-bg-opacity, 1)) !important}.text-bg-success{color:#fff !important;background-color:RGBA(var(--bs-success-rgb), var(--bs-bg-opacity, 1)) !important}.text-bg-info{color:#fff !important;background-color:RGBA(var(--bs-info-rgb), var(--bs-bg-opacity, 1)) !important}.text-bg-warning{color:#fff !important;background-color:RGBA(var(--bs-warning-rgb), var(--bs-bg-opacity, 1)) !important}.text-bg-danger{color:#fff !important;background-color:RGBA(var(--bs-danger-rgb), var(--bs-bg-opacity, 1)) !important}.text-bg-light{color:#000 !important;background-color:RGBA(var(--bs-light-rgb), var(--bs-bg-opacity, 1)) !important}.text-bg-dark{color:#fff !important;background-color:RGBA(var(--bs-dark-rgb), var(--bs-bg-opacity, 1)) !important}.link-default{color:RGBA(var(--bs-default-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-default-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-default:hover,.link-default:focus{color:RGBA(42, 46, 51, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(42, 46, 51, var(--bs-link-underline-opacity, 1)) !important}.link-primary{color:RGBA(var(--bs-primary-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-primary-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-primary:hover,.link-primary:focus{color:RGBA(237, 237, 253, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(237, 237, 253, var(--bs-link-underline-opacity, 1)) !important}.link-secondary{color:RGBA(var(--bs-secondary-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-secondary-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-secondary:hover,.link-secondary:focus{color:RGBA(42, 46, 51, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(42, 46, 51, var(--bs-link-underline-opacity, 1)) !important}.link-success{color:RGBA(var(--bs-success-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-success-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-success:hover,.link-success:focus{color:RGBA(50, 146, 19, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(50, 146, 19, var(--bs-link-underline-opacity, 1)) !important}.link-info{color:RGBA(var(--bs-info-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-info-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-info:hover,.link-info:focus{color:RGBA(122, 67, 150, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(122, 67, 150, var(--bs-link-underline-opacity, 1)) !important}.link-warning{color:RGBA(var(--bs-warning-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-warning-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-warning:hover,.link-warning:focus{color:RGBA(204, 94, 19, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(204, 94, 19, var(--bs-link-underline-opacity, 1)) !important}.link-danger{color:RGBA(var(--bs-danger-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-danger-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-danger:hover,.link-danger:focus{color:RGBA(204, 0, 46, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(204, 0, 46, var(--bs-link-underline-opacity, 1)) !important}.link-light{color:RGBA(var(--bs-light-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-light-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-light:hover,.link-light:focus{color:RGBA(249, 250, 251, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(249, 250, 251, var(--bs-link-underline-opacity, 1)) !important}.link-dark{color:RGBA(var(--bs-dark-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-dark-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-dark:hover,.link-dark:focus{color:RGBA(42, 46, 51, var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(42, 46, 51, var(--bs-link-underline-opacity, 1)) !important}.link-body-emphasis{color:RGBA(var(--bs-emphasis-color-rgb), var(--bs-link-opacity, 1)) !important;text-decoration-color:RGBA(var(--bs-emphasis-color-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-body-emphasis:hover,.link-body-emphasis:focus{color:RGBA(var(--bs-emphasis-color-rgb), var(--bs-link-opacity, 0.75)) !important;text-decoration-color:RGBA(var(--bs-emphasis-color-rgb), var(--bs-link-underline-opacity, 0.75)) !important}.focus-ring:focus{outline:0;box-shadow:var(--bs-focus-ring-x, 0) var(--bs-focus-ring-y, 0) var(--bs-focus-ring-blur, 0) var(--bs-focus-ring-width) var(--bs-focus-ring-color)}.icon-link{display:inline-flex;gap:.375rem;align-items:center;-webkit-align-items:center;text-decoration-color:rgba(var(--bs-link-color-rgb), var(--bs-link-opacity, 0.5));text-underline-offset:.25em;backface-visibility:hidden;-webkit-backface-visibility:hidden;-moz-backface-visibility:hidden;-ms-backface-visibility:hidden;-o-backface-visibility:hidden}.icon-link>.bi{flex-shrink:0;-webkit-flex-shrink:0;width:1em;height:1em;fill:currentcolor;transition:.2s ease-in-out transform}@media(prefers-reduced-motion: reduce){.icon-link>.bi{transition:none}}.icon-link-hover:hover>.bi,.icon-link-hover:focus-visible>.bi{transform:var(--bs-icon-link-transform, translate3d(0.25em, 0, 0))}.ratio{position:relative;width:100%}.ratio::before{display:block;padding-top:var(--bs-aspect-ratio);content:""}.ratio>*{position:absolute;top:0;left:0;width:100%;height:100%}.ratio-1x1{--bs-aspect-ratio: 100%}.ratio-4x3{--bs-aspect-ratio: 75%}.ratio-16x9{--bs-aspect-ratio: 56.25%}.ratio-21x9{--bs-aspect-ratio: 42.8571428571%}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}.sticky-top{position:sticky;top:0;z-index:1020}.sticky-bottom{position:sticky;bottom:0;z-index:1020}@media(min-width: 576px){.sticky-sm-top{position:sticky;top:0;z-index:1020}.sticky-sm-bottom{position:sticky;bottom:0;z-index:1020}}@media(min-width: 768px){.sticky-md-top{position:sticky;top:0;z-index:1020}.sticky-md-bottom{position:sticky;bottom:0;z-index:1020}}@media(min-width: 992px){.sticky-lg-top{position:sticky;top:0;z-index:1020}.sticky-lg-bottom{position:sticky;bottom:0;z-index:1020}}@media(min-width: 1200px){.sticky-xl-top{position:sticky;top:0;z-index:1020}.sticky-xl-bottom{position:sticky;bottom:0;z-index:1020}}@media(min-width: 1400px){.sticky-xxl-top{position:sticky;top:0;z-index:1020}.sticky-xxl-bottom{position:sticky;bottom:0;z-index:1020}}.hstack{display:flex;display:-webkit-flex;flex-direction:row;-webkit-flex-direction:row;align-items:center;-webkit-align-items:center;align-self:stretch;-webkit-align-self:stretch}.vstack{display:flex;display:-webkit-flex;flex:1 1 auto;-webkit-flex:1 1 auto;flex-direction:column;-webkit-flex-direction:column;align-self:stretch;-webkit-align-self:stretch}.visually-hidden,.visually-hidden-focusable:not(:focus):not(:focus-within){width:1px !important;height:1px !important;padding:0 !important;margin:-1px !important;overflow:hidden !important;clip:rect(0, 0, 0, 0) !important;white-space:nowrap !important;border:0 !important}.visually-hidden:not(caption),.visually-hidden-focusable:not(:focus):not(:focus-within):not(caption){position:absolute !important}.stretched-link::after{position:absolute;top:0;right:0;bottom:0;left:0;z-index:1;content:""}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.vr{display:inline-block;align-self:stretch;-webkit-align-self:stretch;width:1px;min-height:1em;background-color:currentcolor;opacity:.25}.align-baseline{vertical-align:baseline !important}.align-top{vertical-align:top !important}.align-middle{vertical-align:middle !important}.align-bottom{vertical-align:bottom !important}.align-text-bottom{vertical-align:text-bottom !important}.align-text-top{vertical-align:text-top !important}.float-start{float:left !important}.float-end{float:right !important}.float-none{float:none !important}.object-fit-contain{object-fit:contain !important}.object-fit-cover{object-fit:cover !important}.object-fit-fill{object-fit:fill !important}.object-fit-scale{object-fit:scale-down !important}.object-fit-none{object-fit:none !important}.opacity-0{opacity:0 !important}.opacity-25{opacity:.25 !important}.opacity-50{opacity:.5 !important}.opacity-75{opacity:.75 !important}.opacity-100{opacity:1 !important}.overflow-auto{overflow:auto !important}.overflow-hidden{overflow:hidden !important}.overflow-visible{overflow:visible !important}.overflow-scroll{overflow:scroll !important}.overflow-x-auto{overflow-x:auto !important}.overflow-x-hidden{overflow-x:hidden !important}.overflow-x-visible{overflow-x:visible !important}.overflow-x-scroll{overflow-x:scroll !important}.overflow-y-auto{overflow-y:auto !important}.overflow-y-hidden{overflow-y:hidden !important}.overflow-y-visible{overflow-y:visible !important}.overflow-y-scroll{overflow-y:scroll !important}.d-inline{display:inline !important}.d-inline-block{display:inline-block !important}.d-block{display:block !important}.d-grid{display:grid !important}.d-inline-grid{display:inline-grid !important}.d-table{display:table !important}.d-table-row{display:table-row !important}.d-table-cell{display:table-cell !important}.d-flex{display:flex !important}.d-inline-flex{display:inline-flex !important}.d-none{display:none !important}.shadow{box-shadow:0 .5rem 1rem rgba(0,0,0,.15) !important}.shadow-sm{box-shadow:0 .125rem .25rem rgba(0,0,0,.075) !important}.shadow-lg{box-shadow:0 1rem 3rem rgba(0,0,0,.175) !important}.shadow-none{box-shadow:none !important}.focus-ring-default{--bs-focus-ring-color: rgba(var(--bs-default-rgb), var(--bs-focus-ring-opacity))}.focus-ring-primary{--bs-focus-ring-color: rgba(var(--bs-primary-rgb), var(--bs-focus-ring-opacity))}.focus-ring-secondary{--bs-focus-ring-color: rgba(var(--bs-secondary-rgb), var(--bs-focus-ring-opacity))}.focus-ring-success{--bs-focus-ring-color: rgba(var(--bs-success-rgb), var(--bs-focus-ring-opacity))}.focus-ring-info{--bs-focus-ring-color: rgba(var(--bs-info-rgb), var(--bs-focus-ring-opacity))}.focus-ring-warning{--bs-focus-ring-color: rgba(var(--bs-warning-rgb), var(--bs-focus-ring-opacity))}.focus-ring-danger{--bs-focus-ring-color: rgba(var(--bs-danger-rgb), var(--bs-focus-ring-opacity))}.focus-ring-light{--bs-focus-ring-color: rgba(var(--bs-light-rgb), var(--bs-focus-ring-opacity))}.focus-ring-dark{--bs-focus-ring-color: rgba(var(--bs-dark-rgb), var(--bs-focus-ring-opacity))}.position-static{position:static !important}.position-relative{position:relative !important}.position-absolute{position:absolute !important}.position-fixed{position:fixed !important}.position-sticky{position:sticky !important}.top-0{top:0 !important}.top-50{top:50% !important}.top-100{top:100% !important}.bottom-0{bottom:0 !important}.bottom-50{bottom:50% !important}.bottom-100{bottom:100% !important}.start-0{left:0 !important}.start-50{left:50% !important}.start-100{left:100% !important}.end-0{right:0 !important}.end-50{right:50% !important}.end-100{right:100% !important}.translate-middle{transform:translate(-50%, -50%) !important}.translate-middle-x{transform:translateX(-50%) !important}.translate-middle-y{transform:translateY(-50%) !important}.border{border:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color) !important}.border-0{border:0 !important}.border-top{border-top:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color) !important}.border-top-0{border-top:0 !important}.border-end{border-right:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color) !important}.border-end-0{border-right:0 !important}.border-bottom{border-bottom:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color) !important}.border-bottom-0{border-bottom:0 !important}.border-start{border-left:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color) !important}.border-start-0{border-left:0 !important}.border-default{--bs-border-opacity: 1;border-color:rgba(var(--bs-default-rgb), var(--bs-border-opacity)) !important}.border-primary{--bs-border-opacity: 1;border-color:rgba(var(--bs-primary-rgb), var(--bs-border-opacity)) !important}.border-secondary{--bs-border-opacity: 1;border-color:rgba(var(--bs-secondary-rgb), var(--bs-border-opacity)) !important}.border-success{--bs-border-opacity: 1;border-color:rgba(var(--bs-success-rgb), var(--bs-border-opacity)) !important}.border-info{--bs-border-opacity: 1;border-color:rgba(var(--bs-info-rgb), var(--bs-border-opacity)) !important}.border-warning{--bs-border-opacity: 1;border-color:rgba(var(--bs-warning-rgb), var(--bs-border-opacity)) !important}.border-danger{--bs-border-opacity: 1;border-color:rgba(var(--bs-danger-rgb), var(--bs-border-opacity)) !important}.border-light{--bs-border-opacity: 1;border-color:rgba(var(--bs-light-rgb), var(--bs-border-opacity)) !important}.border-dark{--bs-border-opacity: 1;border-color:rgba(var(--bs-dark-rgb), var(--bs-border-opacity)) !important}.border-black{--bs-border-opacity: 1;border-color:rgba(var(--bs-black-rgb), var(--bs-border-opacity)) !important}.border-white{--bs-border-opacity: 1;border-color:rgba(var(--bs-white-rgb), var(--bs-border-opacity)) !important}.border-primary-subtle{border-color:var(--bs-primary-border-subtle) !important}.border-secondary-subtle{border-color:var(--bs-secondary-border-subtle) !important}.border-success-subtle{border-color:var(--bs-success-border-subtle) !important}.border-info-subtle{border-color:var(--bs-info-border-subtle) !important}.border-warning-subtle{border-color:var(--bs-warning-border-subtle) !important}.border-danger-subtle{border-color:var(--bs-danger-border-subtle) !important}.border-light-subtle{border-color:var(--bs-light-border-subtle) !important}.border-dark-subtle{border-color:var(--bs-dark-border-subtle) !important}.border-1{border-width:1px !important}.border-2{border-width:2px !important}.border-3{border-width:3px !important}.border-4{border-width:4px !important}.border-5{border-width:5px !important}.border-opacity-10{--bs-border-opacity: 0.1}.border-opacity-25{--bs-border-opacity: 0.25}.border-opacity-50{--bs-border-opacity: 0.5}.border-opacity-75{--bs-border-opacity: 0.75}.border-opacity-100{--bs-border-opacity: 1}.w-25{width:25% !important}.w-50{width:50% !important}.w-75{width:75% !important}.w-100{width:100% !important}.w-auto{width:auto !important}.mw-100{max-width:100% !important}.vw-100{width:100vw !important}.min-vw-100{min-width:100vw !important}.h-25{height:25% !important}.h-50{height:50% !important}.h-75{height:75% !important}.h-100{height:100% !important}.h-auto{height:auto !important}.mh-100{max-height:100% !important}.vh-100{height:100vh !important}.min-vh-100{min-height:100vh !important}.flex-fill{flex:1 1 auto !important}.flex-row{flex-direction:row !important}.flex-column{flex-direction:column !important}.flex-row-reverse{flex-direction:row-reverse !important}.flex-column-reverse{flex-direction:column-reverse !important}.flex-grow-0{flex-grow:0 !important}.flex-grow-1{flex-grow:1 !important}.flex-shrink-0{flex-shrink:0 !important}.flex-shrink-1{flex-shrink:1 !important}.flex-wrap{flex-wrap:wrap !important}.flex-nowrap{flex-wrap:nowrap !important}.flex-wrap-reverse{flex-wrap:wrap-reverse !important}.justify-content-start{justify-content:flex-start !important}.justify-content-end{justify-content:flex-end !important}.justify-content-center{justify-content:center !important}.justify-content-between{justify-content:space-between !important}.justify-content-around{justify-content:space-around !important}.justify-content-evenly{justify-content:space-evenly !important}.align-items-start{align-items:flex-start !important}.align-items-end{align-items:flex-end !important}.align-items-center{align-items:center !important}.align-items-baseline{align-items:baseline !important}.align-items-stretch{align-items:stretch !important}.align-content-start{align-content:flex-start !important}.align-content-end{align-content:flex-end !important}.align-content-center{align-content:center !important}.align-content-between{align-content:space-between !important}.align-content-around{align-content:space-around !important}.align-content-stretch{align-content:stretch !important}.align-self-auto{align-self:auto !important}.align-self-start{align-self:flex-start !important}.align-self-end{align-self:flex-end !important}.align-self-center{align-self:center !important}.align-self-baseline{align-self:baseline !important}.align-self-stretch{align-self:stretch !important}.order-first{order:-1 !important}.order-0{order:0 !important}.order-1{order:1 !important}.order-2{order:2 !important}.order-3{order:3 !important}.order-4{order:4 !important}.order-5{order:5 !important}.order-last{order:6 !important}.m-0{margin:0 !important}.m-1{margin:.25rem !important}.m-2{margin:.5rem !important}.m-3{margin:1rem !important}.m-4{margin:1.5rem !important}.m-5{margin:3rem !important}.m-auto{margin:auto !important}.mx-0{margin-right:0 !important;margin-left:0 !important}.mx-1{margin-right:.25rem !important;margin-left:.25rem !important}.mx-2{margin-right:.5rem !important;margin-left:.5rem !important}.mx-3{margin-right:1rem !important;margin-left:1rem !important}.mx-4{margin-right:1.5rem !important;margin-left:1.5rem !important}.mx-5{margin-right:3rem !important;margin-left:3rem !important}.mx-auto{margin-right:auto !important;margin-left:auto !important}.my-0{margin-top:0 !important;margin-bottom:0 !important}.my-1{margin-top:.25rem !important;margin-bottom:.25rem !important}.my-2{margin-top:.5rem !important;margin-bottom:.5rem !important}.my-3{margin-top:1rem !important;margin-bottom:1rem !important}.my-4{margin-top:1.5rem !important;margin-bottom:1.5rem !important}.my-5{margin-top:3rem !important;margin-bottom:3rem !important}.my-auto{margin-top:auto !important;margin-bottom:auto !important}.mt-0{margin-top:0 !important}.mt-1{margin-top:.25rem !important}.mt-2{margin-top:.5rem !important}.mt-3{margin-top:1rem !important}.mt-4{margin-top:1.5rem !important}.mt-5{margin-top:3rem !important}.mt-auto{margin-top:auto !important}.me-0{margin-right:0 !important}.me-1{margin-right:.25rem !important}.me-2{margin-right:.5rem !important}.me-3{margin-right:1rem !important}.me-4{margin-right:1.5rem !important}.me-5{margin-right:3rem !important}.me-auto{margin-right:auto !important}.mb-0{margin-bottom:0 !important}.mb-1{margin-bottom:.25rem !important}.mb-2{margin-bottom:.5rem !important}.mb-3{margin-bottom:1rem !important}.mb-4{margin-bottom:1.5rem !important}.mb-5{margin-bottom:3rem !important}.mb-auto{margin-bottom:auto !important}.ms-0{margin-left:0 !important}.ms-1{margin-left:.25rem !important}.ms-2{margin-left:.5rem !important}.ms-3{margin-left:1rem !important}.ms-4{margin-left:1.5rem !important}.ms-5{margin-left:3rem !important}.ms-auto{margin-left:auto !important}.p-0{padding:0 !important}.p-1{padding:.25rem !important}.p-2{padding:.5rem !important}.p-3{padding:1rem !important}.p-4{padding:1.5rem !important}.p-5{padding:3rem !important}.px-0{padding-right:0 !important;padding-left:0 !important}.px-1{padding-right:.25rem !important;padding-left:.25rem !important}.px-2{padding-right:.5rem !important;padding-left:.5rem !important}.px-3{padding-right:1rem !important;padding-left:1rem !important}.px-4{padding-right:1.5rem !important;padding-left:1.5rem !important}.px-5{padding-right:3rem !important;padding-left:3rem !important}.py-0{padding-top:0 !important;padding-bottom:0 !important}.py-1{padding-top:.25rem !important;padding-bottom:.25rem !important}.py-2{padding-top:.5rem !important;padding-bottom:.5rem !important}.py-3{padding-top:1rem !important;padding-bottom:1rem !important}.py-4{padding-top:1.5rem !important;padding-bottom:1.5rem !important}.py-5{padding-top:3rem !important;padding-bottom:3rem !important}.pt-0{padding-top:0 !important}.pt-1{padding-top:.25rem !important}.pt-2{padding-top:.5rem !important}.pt-3{padding-top:1rem !important}.pt-4{padding-top:1.5rem !important}.pt-5{padding-top:3rem !important}.pe-0{padding-right:0 !important}.pe-1{padding-right:.25rem !important}.pe-2{padding-right:.5rem !important}.pe-3{padding-right:1rem !important}.pe-4{padding-right:1.5rem !important}.pe-5{padding-right:3rem !important}.pb-0{padding-bottom:0 !important}.pb-1{padding-bottom:.25rem !important}.pb-2{padding-bottom:.5rem !important}.pb-3{padding-bottom:1rem !important}.pb-4{padding-bottom:1.5rem !important}.pb-5{padding-bottom:3rem !important}.ps-0{padding-left:0 !important}.ps-1{padding-left:.25rem !important}.ps-2{padding-left:.5rem !important}.ps-3{padding-left:1rem !important}.ps-4{padding-left:1.5rem !important}.ps-5{padding-left:3rem !important}.gap-0{gap:0 !important}.gap-1{gap:.25rem !important}.gap-2{gap:.5rem !important}.gap-3{gap:1rem !important}.gap-4{gap:1.5rem !important}.gap-5{gap:3rem !important}.row-gap-0{row-gap:0 !important}.row-gap-1{row-gap:.25rem !important}.row-gap-2{row-gap:.5rem !important}.row-gap-3{row-gap:1rem !important}.row-gap-4{row-gap:1.5rem !important}.row-gap-5{row-gap:3rem !important}.column-gap-0{column-gap:0 !important}.column-gap-1{column-gap:.25rem !important}.column-gap-2{column-gap:.5rem !important}.column-gap-3{column-gap:1rem !important}.column-gap-4{column-gap:1.5rem !important}.column-gap-5{column-gap:3rem !important}.font-monospace{font-family:var(--bs-font-monospace) !important}.fs-1{font-size:calc(1.325rem + 0.9vw) !important}.fs-2{font-size:calc(1.29rem + 0.48vw) !important}.fs-3{font-size:calc(1.27rem + 0.24vw) !important}.fs-4{font-size:1.25rem !important}.fs-5{font-size:1.1rem !important}.fs-6{font-size:1rem !important}.fst-italic{font-style:italic !important}.fst-normal{font-style:normal !important}.fw-lighter{font-weight:lighter !important}.fw-light{font-weight:300 !important}.fw-normal{font-weight:400 !important}.fw-medium{font-weight:500 !important}.fw-semibold{font-weight:600 !important}.fw-bold{font-weight:700 !important}.fw-bolder{font-weight:bolder !important}.lh-1{line-height:1 !important}.lh-sm{line-height:1.25 !important}.lh-base{line-height:1.5 !important}.lh-lg{line-height:2 !important}.text-start{text-align:left !important}.text-end{text-align:right !important}.text-center{text-align:center !important}.text-decoration-none{text-decoration:none !important}.text-decoration-underline{text-decoration:underline !important}.text-decoration-line-through{text-decoration:line-through !important}.text-lowercase{text-transform:lowercase !important}.text-uppercase{text-transform:uppercase !important}.text-capitalize{text-transform:capitalize !important}.text-wrap{white-space:normal !important}.text-nowrap{white-space:nowrap !important}.text-break{word-wrap:break-word !important;word-break:break-word !important}.text-default{--bs-text-opacity: 1;color:rgba(var(--bs-default-rgb), var(--bs-text-opacity)) !important}.text-primary{--bs-text-opacity: 1;color:rgba(var(--bs-primary-rgb), var(--bs-text-opacity)) !important}.text-secondary{--bs-text-opacity: 1;color:rgba(var(--bs-secondary-rgb), var(--bs-text-opacity)) !important}.text-success{--bs-text-opacity: 1;color:rgba(var(--bs-success-rgb), var(--bs-text-opacity)) !important}.text-info{--bs-text-opacity: 1;color:rgba(var(--bs-info-rgb), var(--bs-text-opacity)) !important}.text-warning{--bs-text-opacity: 1;color:rgba(var(--bs-warning-rgb), var(--bs-text-opacity)) !important}.text-danger{--bs-text-opacity: 1;color:rgba(var(--bs-danger-rgb), var(--bs-text-opacity)) !important}.text-light{--bs-text-opacity: 1;color:rgba(var(--bs-light-rgb), var(--bs-text-opacity)) !important}.text-dark{--bs-text-opacity: 1;color:rgba(var(--bs-dark-rgb), var(--bs-text-opacity)) !important}.text-black{--bs-text-opacity: 1;color:rgba(var(--bs-black-rgb), var(--bs-text-opacity)) !important}.text-white{--bs-text-opacity: 1;color:rgba(var(--bs-white-rgb), var(--bs-text-opacity)) !important}.text-body{--bs-text-opacity: 1;color:rgba(var(--bs-body-color-rgb), var(--bs-text-opacity)) !important}.text-muted{--bs-text-opacity: 1;color:var(--bs-secondary-color) !important}.text-black-50{--bs-text-opacity: 1;color:rgba(0,0,0,.5) !important}.text-white-50{--bs-text-opacity: 1;color:rgba(255,255,255,.5) !important}.text-body-secondary{--bs-text-opacity: 1;color:var(--bs-secondary-color) !important}.text-body-tertiary{--bs-text-opacity: 1;color:var(--bs-tertiary-color) !important}.text-body-emphasis{--bs-text-opacity: 1;color:var(--bs-emphasis-color) !important}.text-reset{--bs-text-opacity: 1;color:inherit !important}.text-opacity-25{--bs-text-opacity: 0.25}.text-opacity-50{--bs-text-opacity: 0.5}.text-opacity-75{--bs-text-opacity: 0.75}.text-opacity-100{--bs-text-opacity: 1}.text-primary-emphasis{color:var(--bs-primary-text-emphasis) !important}.text-secondary-emphasis{color:var(--bs-secondary-text-emphasis) !important}.text-success-emphasis{color:var(--bs-success-text-emphasis) !important}.text-info-emphasis{color:var(--bs-info-text-emphasis) !important}.text-warning-emphasis{color:var(--bs-warning-text-emphasis) !important}.text-danger-emphasis{color:var(--bs-danger-text-emphasis) !important}.text-light-emphasis{color:var(--bs-light-text-emphasis) !important}.text-dark-emphasis{color:var(--bs-dark-text-emphasis) !important}.link-opacity-10{--bs-link-opacity: 0.1}.link-opacity-10-hover:hover{--bs-link-opacity: 0.1}.link-opacity-25{--bs-link-opacity: 0.25}.link-opacity-25-hover:hover{--bs-link-opacity: 0.25}.link-opacity-50{--bs-link-opacity: 0.5}.link-opacity-50-hover:hover{--bs-link-opacity: 0.5}.link-opacity-75{--bs-link-opacity: 0.75}.link-opacity-75-hover:hover{--bs-link-opacity: 0.75}.link-opacity-100{--bs-link-opacity: 1}.link-opacity-100-hover:hover{--bs-link-opacity: 1}.link-offset-1{text-underline-offset:.125em !important}.link-offset-1-hover:hover{text-underline-offset:.125em !important}.link-offset-2{text-underline-offset:.25em !important}.link-offset-2-hover:hover{text-underline-offset:.25em !important}.link-offset-3{text-underline-offset:.375em !important}.link-offset-3-hover:hover{text-underline-offset:.375em !important}.link-underline-default{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-default-rgb), var(--bs-link-underline-opacity)) !important}.link-underline-primary{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-primary-rgb), var(--bs-link-underline-opacity)) !important}.link-underline-secondary{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-secondary-rgb), var(--bs-link-underline-opacity)) !important}.link-underline-success{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-success-rgb), var(--bs-link-underline-opacity)) !important}.link-underline-info{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-info-rgb), var(--bs-link-underline-opacity)) !important}.link-underline-warning{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-warning-rgb), var(--bs-link-underline-opacity)) !important}.link-underline-danger{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-danger-rgb), var(--bs-link-underline-opacity)) !important}.link-underline-light{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-light-rgb), var(--bs-link-underline-opacity)) !important}.link-underline-dark{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-dark-rgb), var(--bs-link-underline-opacity)) !important}.link-underline{--bs-link-underline-opacity: 1;text-decoration-color:rgba(var(--bs-link-color-rgb), var(--bs-link-underline-opacity, 1)) !important}.link-underline-opacity-0{--bs-link-underline-opacity: 0}.link-underline-opacity-0-hover:hover{--bs-link-underline-opacity: 0}.link-underline-opacity-10{--bs-link-underline-opacity: 0.1}.link-underline-opacity-10-hover:hover{--bs-link-underline-opacity: 0.1}.link-underline-opacity-25{--bs-link-underline-opacity: 0.25}.link-underline-opacity-25-hover:hover{--bs-link-underline-opacity: 0.25}.link-underline-opacity-50{--bs-link-underline-opacity: 0.5}.link-underline-opacity-50-hover:hover{--bs-link-underline-opacity: 0.5}.link-underline-opacity-75{--bs-link-underline-opacity: 0.75}.link-underline-opacity-75-hover:hover{--bs-link-underline-opacity: 0.75}.link-underline-opacity-100{--bs-link-underline-opacity: 1}.link-underline-opacity-100-hover:hover{--bs-link-underline-opacity: 1}.bg-default{--bs-bg-opacity: 1;background-color:rgba(var(--bs-default-rgb), var(--bs-bg-opacity)) !important}.bg-primary{--bs-bg-opacity: 1;background-color:rgba(var(--bs-primary-rgb), var(--bs-bg-opacity)) !important}.bg-secondary{--bs-bg-opacity: 1;background-color:rgba(var(--bs-secondary-rgb), var(--bs-bg-opacity)) !important}.bg-success{--bs-bg-opacity: 1;background-color:rgba(var(--bs-success-rgb), var(--bs-bg-opacity)) !important}.bg-info{--bs-bg-opacity: 1;background-color:rgba(var(--bs-info-rgb), var(--bs-bg-opacity)) !important}.bg-warning{--bs-bg-opacity: 1;background-color:rgba(var(--bs-warning-rgb), var(--bs-bg-opacity)) !important}.bg-danger{--bs-bg-opacity: 1;background-color:rgba(var(--bs-danger-rgb), var(--bs-bg-opacity)) !important}.bg-light{--bs-bg-opacity: 1;background-color:rgba(var(--bs-light-rgb), var(--bs-bg-opacity)) !important}.bg-dark{--bs-bg-opacity: 1;background-color:rgba(var(--bs-dark-rgb), var(--bs-bg-opacity)) !important}.bg-black{--bs-bg-opacity: 1;background-color:rgba(var(--bs-black-rgb), var(--bs-bg-opacity)) !important}.bg-white{--bs-bg-opacity: 1;background-color:rgba(var(--bs-white-rgb), var(--bs-bg-opacity)) !important}.bg-body{--bs-bg-opacity: 1;background-color:rgba(var(--bs-body-bg-rgb), var(--bs-bg-opacity)) !important}.bg-transparent{--bs-bg-opacity: 1;background-color:rgba(0,0,0,0) !important}.bg-body-secondary{--bs-bg-opacity: 1;background-color:rgba(var(--bs-secondary-bg-rgb), var(--bs-bg-opacity)) !important}.bg-body-tertiary{--bs-bg-opacity: 1;background-color:rgba(var(--bs-tertiary-bg-rgb), var(--bs-bg-opacity)) !important}.bg-opacity-10{--bs-bg-opacity: 0.1}.bg-opacity-25{--bs-bg-opacity: 0.25}.bg-opacity-50{--bs-bg-opacity: 0.5}.bg-opacity-75{--bs-bg-opacity: 0.75}.bg-opacity-100{--bs-bg-opacity: 1}.bg-primary-subtle{background-color:var(--bs-primary-bg-subtle) !important}.bg-secondary-subtle{background-color:var(--bs-secondary-bg-subtle) !important}.bg-success-subtle{background-color:var(--bs-success-bg-subtle) !important}.bg-info-subtle{background-color:var(--bs-info-bg-subtle) !important}.bg-warning-subtle{background-color:var(--bs-warning-bg-subtle) !important}.bg-danger-subtle{background-color:var(--bs-danger-bg-subtle) !important}.bg-light-subtle{background-color:var(--bs-light-bg-subtle) !important}.bg-dark-subtle{background-color:var(--bs-dark-bg-subtle) !important}.bg-gradient{background-image:var(--bs-gradient) !important}.user-select-all{user-select:all !important}.user-select-auto{user-select:auto !important}.user-select-none{user-select:none !important}.pe-none{pointer-events:none !important}.pe-auto{pointer-events:auto !important}.rounded{border-radius:var(--bs-border-radius) !important}.rounded-0{border-radius:0 !important}.rounded-1{border-radius:var(--bs-border-radius-sm) !important}.rounded-2{border-radius:var(--bs-border-radius) !important}.rounded-3{border-radius:var(--bs-border-radius-lg) !important}.rounded-4{border-radius:var(--bs-border-radius-xl) !important}.rounded-5{border-radius:var(--bs-border-radius-xxl) !important}.rounded-circle{border-radius:50% !important}.rounded-pill{border-radius:var(--bs-border-radius-pill) !important}.rounded-top{border-top-left-radius:var(--bs-border-radius) !important;border-top-right-radius:var(--bs-border-radius) !important}.rounded-top-0{border-top-left-radius:0 !important;border-top-right-radius:0 !important}.rounded-top-1{border-top-left-radius:var(--bs-border-radius-sm) !important;border-top-right-radius:var(--bs-border-radius-sm) !important}.rounded-top-2{border-top-left-radius:var(--bs-border-radius) !important;border-top-right-radius:var(--bs-border-radius) !important}.rounded-top-3{border-top-left-radius:var(--bs-border-radius-lg) !important;border-top-right-radius:var(--bs-border-radius-lg) !important}.rounded-top-4{border-top-left-radius:var(--bs-border-radius-xl) !important;border-top-right-radius:var(--bs-border-radius-xl) !important}.rounded-top-5{border-top-left-radius:var(--bs-border-radius-xxl) !important;border-top-right-radius:var(--bs-border-radius-xxl) !important}.rounded-top-circle{border-top-left-radius:50% !important;border-top-right-radius:50% !important}.rounded-top-pill{border-top-left-radius:var(--bs-border-radius-pill) !important;border-top-right-radius:var(--bs-border-radius-pill) !important}.rounded-end{border-top-right-radius:var(--bs-border-radius) !important;border-bottom-right-radius:var(--bs-border-radius) !important}.rounded-end-0{border-top-right-radius:0 !important;border-bottom-right-radius:0 !important}.rounded-end-1{border-top-right-radius:var(--bs-border-radius-sm) !important;border-bottom-right-radius:var(--bs-border-radius-sm) !important}.rounded-end-2{border-top-right-radius:var(--bs-border-radius) !important;border-bottom-right-radius:var(--bs-border-radius) !important}.rounded-end-3{border-top-right-radius:var(--bs-border-radius-lg) !important;border-bottom-right-radius:var(--bs-border-radius-lg) !important}.rounded-end-4{border-top-right-radius:var(--bs-border-radius-xl) !important;border-bottom-right-radius:var(--bs-border-radius-xl) !important}.rounded-end-5{border-top-right-radius:var(--bs-border-radius-xxl) !important;border-bottom-right-radius:var(--bs-border-radius-xxl) !important}.rounded-end-circle{border-top-right-radius:50% !important;border-bottom-right-radius:50% !important}.rounded-end-pill{border-top-right-radius:var(--bs-border-radius-pill) !important;border-bottom-right-radius:var(--bs-border-radius-pill) !important}.rounded-bottom{border-bottom-right-radius:var(--bs-border-radius) !important;border-bottom-left-radius:var(--bs-border-radius) !important}.rounded-bottom-0{border-bottom-right-radius:0 !important;border-bottom-left-radius:0 !important}.rounded-bottom-1{border-bottom-right-radius:var(--bs-border-radius-sm) !important;border-bottom-left-radius:var(--bs-border-radius-sm) !important}.rounded-bottom-2{border-bottom-right-radius:var(--bs-border-radius) !important;border-bottom-left-radius:var(--bs-border-radius) !important}.rounded-bottom-3{border-bottom-right-radius:var(--bs-border-radius-lg) !important;border-bottom-left-radius:var(--bs-border-radius-lg) !important}.rounded-bottom-4{border-bottom-right-radius:var(--bs-border-radius-xl) !important;border-bottom-left-radius:var(--bs-border-radius-xl) !important}.rounded-bottom-5{border-bottom-right-radius:var(--bs-border-radius-xxl) !important;border-bottom-left-radius:var(--bs-border-radius-xxl) !important}.rounded-bottom-circle{border-bottom-right-radius:50% !important;border-bottom-left-radius:50% !important}.rounded-bottom-pill{border-bottom-right-radius:var(--bs-border-radius-pill) !important;border-bottom-left-radius:var(--bs-border-radius-pill) !important}.rounded-start{border-bottom-left-radius:var(--bs-border-radius) !important;border-top-left-radius:var(--bs-border-radius) !important}.rounded-start-0{border-bottom-left-radius:0 !important;border-top-left-radius:0 !important}.rounded-start-1{border-bottom-left-radius:var(--bs-border-radius-sm) !important;border-top-left-radius:var(--bs-border-radius-sm) !important}.rounded-start-2{border-bottom-left-radius:var(--bs-border-radius) !important;border-top-left-radius:var(--bs-border-radius) !important}.rounded-start-3{border-bottom-left-radius:var(--bs-border-radius-lg) !important;border-top-left-radius:var(--bs-border-radius-lg) !important}.rounded-start-4{border-bottom-left-radius:var(--bs-border-radius-xl) !important;border-top-left-radius:var(--bs-border-radius-xl) !important}.rounded-start-5{border-bottom-left-radius:var(--bs-border-radius-xxl) !important;border-top-left-radius:var(--bs-border-radius-xxl) !important}.rounded-start-circle{border-bottom-left-radius:50% !important;border-top-left-radius:50% !important}.rounded-start-pill{border-bottom-left-radius:var(--bs-border-radius-pill) !important;border-top-left-radius:var(--bs-border-radius-pill) !important}.visible{visibility:visible !important}.invisible{visibility:hidden !important}.z-n1{z-index:-1 !important}.z-0{z-index:0 !important}.z-1{z-index:1 !important}.z-2{z-index:2 !important}.z-3{z-index:3 !important}@media(min-width: 576px){.float-sm-start{float:left !important}.float-sm-end{float:right !important}.float-sm-none{float:none !important}.object-fit-sm-contain{object-fit:contain !important}.object-fit-sm-cover{object-fit:cover !important}.object-fit-sm-fill{object-fit:fill !important}.object-fit-sm-scale{object-fit:scale-down !important}.object-fit-sm-none{object-fit:none !important}.d-sm-inline{display:inline !important}.d-sm-inline-block{display:inline-block !important}.d-sm-block{display:block !important}.d-sm-grid{display:grid !important}.d-sm-inline-grid{display:inline-grid !important}.d-sm-table{display:table !important}.d-sm-table-row{display:table-row !important}.d-sm-table-cell{display:table-cell !important}.d-sm-flex{display:flex !important}.d-sm-inline-flex{display:inline-flex !important}.d-sm-none{display:none !important}.flex-sm-fill{flex:1 1 auto !important}.flex-sm-row{flex-direction:row !important}.flex-sm-column{flex-direction:column !important}.flex-sm-row-reverse{flex-direction:row-reverse !important}.flex-sm-column-reverse{flex-direction:column-reverse !important}.flex-sm-grow-0{flex-grow:0 !important}.flex-sm-grow-1{flex-grow:1 !important}.flex-sm-shrink-0{flex-shrink:0 !important}.flex-sm-shrink-1{flex-shrink:1 !important}.flex-sm-wrap{flex-wrap:wrap !important}.flex-sm-nowrap{flex-wrap:nowrap !important}.flex-sm-wrap-reverse{flex-wrap:wrap-reverse !important}.justify-content-sm-start{justify-content:flex-start !important}.justify-content-sm-end{justify-content:flex-end !important}.justify-content-sm-center{justify-content:center !important}.justify-content-sm-between{justify-content:space-between !important}.justify-content-sm-around{justify-content:space-around !important}.justify-content-sm-evenly{justify-content:space-evenly !important}.align-items-sm-start{align-items:flex-start !important}.align-items-sm-end{align-items:flex-end !important}.align-items-sm-center{align-items:center !important}.align-items-sm-baseline{align-items:baseline !important}.align-items-sm-stretch{align-items:stretch !important}.align-content-sm-start{align-content:flex-start !important}.align-content-sm-end{align-content:flex-end !important}.align-content-sm-center{align-content:center !important}.align-content-sm-between{align-content:space-between !important}.align-content-sm-around{align-content:space-around !important}.align-content-sm-stretch{align-content:stretch !important}.align-self-sm-auto{align-self:auto !important}.align-self-sm-start{align-self:flex-start !important}.align-self-sm-end{align-self:flex-end !important}.align-self-sm-center{align-self:center !important}.align-self-sm-baseline{align-self:baseline !important}.align-self-sm-stretch{align-self:stretch !important}.order-sm-first{order:-1 !important}.order-sm-0{order:0 !important}.order-sm-1{order:1 !important}.order-sm-2{order:2 !important}.order-sm-3{order:3 !important}.order-sm-4{order:4 !important}.order-sm-5{order:5 !important}.order-sm-last{order:6 !important}.m-sm-0{margin:0 !important}.m-sm-1{margin:.25rem !important}.m-sm-2{margin:.5rem !important}.m-sm-3{margin:1rem !important}.m-sm-4{margin:1.5rem !important}.m-sm-5{margin:3rem !important}.m-sm-auto{margin:auto !important}.mx-sm-0{margin-right:0 !important;margin-left:0 !important}.mx-sm-1{margin-right:.25rem !important;margin-left:.25rem !important}.mx-sm-2{margin-right:.5rem !important;margin-left:.5rem !important}.mx-sm-3{margin-right:1rem !important;margin-left:1rem !important}.mx-sm-4{margin-right:1.5rem !important;margin-left:1.5rem !important}.mx-sm-5{margin-right:3rem !important;margin-left:3rem !important}.mx-sm-auto{margin-right:auto !important;margin-left:auto !important}.my-sm-0{margin-top:0 !important;margin-bottom:0 !important}.my-sm-1{margin-top:.25rem !important;margin-bottom:.25rem !important}.my-sm-2{margin-top:.5rem !important;margin-bottom:.5rem !important}.my-sm-3{margin-top:1rem !important;margin-bottom:1rem !important}.my-sm-4{margin-top:1.5rem !important;margin-bottom:1.5rem !important}.my-sm-5{margin-top:3rem !important;margin-bottom:3rem !important}.my-sm-auto{margin-top:auto !important;margin-bottom:auto !important}.mt-sm-0{margin-top:0 !important}.mt-sm-1{margin-top:.25rem !important}.mt-sm-2{margin-top:.5rem !important}.mt-sm-3{margin-top:1rem !important}.mt-sm-4{margin-top:1.5rem !important}.mt-sm-5{margin-top:3rem !important}.mt-sm-auto{margin-top:auto !important}.me-sm-0{margin-right:0 !important}.me-sm-1{margin-right:.25rem !important}.me-sm-2{margin-right:.5rem !important}.me-sm-3{margin-right:1rem !important}.me-sm-4{margin-right:1.5rem !important}.me-sm-5{margin-right:3rem !important}.me-sm-auto{margin-right:auto !important}.mb-sm-0{margin-bottom:0 !important}.mb-sm-1{margin-bottom:.25rem !important}.mb-sm-2{margin-bottom:.5rem !important}.mb-sm-3{margin-bottom:1rem !important}.mb-sm-4{margin-bottom:1.5rem !important}.mb-sm-5{margin-bottom:3rem !important}.mb-sm-auto{margin-bottom:auto !important}.ms-sm-0{margin-left:0 !important}.ms-sm-1{margin-left:.25rem !important}.ms-sm-2{margin-left:.5rem !important}.ms-sm-3{margin-left:1rem !important}.ms-sm-4{margin-left:1.5rem !important}.ms-sm-5{margin-left:3rem !important}.ms-sm-auto{margin-left:auto !important}.p-sm-0{padding:0 !important}.p-sm-1{padding:.25rem !important}.p-sm-2{padding:.5rem !important}.p-sm-3{padding:1rem !important}.p-sm-4{padding:1.5rem !important}.p-sm-5{padding:3rem !important}.px-sm-0{padding-right:0 !important;padding-left:0 !important}.px-sm-1{padding-right:.25rem !important;padding-left:.25rem !important}.px-sm-2{padding-right:.5rem !important;padding-left:.5rem !important}.px-sm-3{padding-right:1rem !important;padding-left:1rem !important}.px-sm-4{padding-right:1.5rem !important;padding-left:1.5rem !important}.px-sm-5{padding-right:3rem !important;padding-left:3rem !important}.py-sm-0{padding-top:0 !important;padding-bottom:0 !important}.py-sm-1{padding-top:.25rem !important;padding-bottom:.25rem !important}.py-sm-2{padding-top:.5rem !important;padding-bottom:.5rem !important}.py-sm-3{padding-top:1rem !important;padding-bottom:1rem !important}.py-sm-4{padding-top:1.5rem !important;padding-bottom:1.5rem !important}.py-sm-5{padding-top:3rem !important;padding-bottom:3rem !important}.pt-sm-0{padding-top:0 !important}.pt-sm-1{padding-top:.25rem !important}.pt-sm-2{padding-top:.5rem !important}.pt-sm-3{padding-top:1rem !important}.pt-sm-4{padding-top:1.5rem !important}.pt-sm-5{padding-top:3rem !important}.pe-sm-0{padding-right:0 !important}.pe-sm-1{padding-right:.25rem !important}.pe-sm-2{padding-right:.5rem !important}.pe-sm-3{padding-right:1rem !important}.pe-sm-4{padding-right:1.5rem !important}.pe-sm-5{padding-right:3rem !important}.pb-sm-0{padding-bottom:0 !important}.pb-sm-1{padding-bottom:.25rem !important}.pb-sm-2{padding-bottom:.5rem !important}.pb-sm-3{padding-bottom:1rem !important}.pb-sm-4{padding-bottom:1.5rem !important}.pb-sm-5{padding-bottom:3rem !important}.ps-sm-0{padding-left:0 !important}.ps-sm-1{padding-left:.25rem !important}.ps-sm-2{padding-left:.5rem !important}.ps-sm-3{padding-left:1rem !important}.ps-sm-4{padding-left:1.5rem !important}.ps-sm-5{padding-left:3rem !important}.gap-sm-0{gap:0 !important}.gap-sm-1{gap:.25rem !important}.gap-sm-2{gap:.5rem !important}.gap-sm-3{gap:1rem !important}.gap-sm-4{gap:1.5rem !important}.gap-sm-5{gap:3rem !important}.row-gap-sm-0{row-gap:0 !important}.row-gap-sm-1{row-gap:.25rem !important}.row-gap-sm-2{row-gap:.5rem !important}.row-gap-sm-3{row-gap:1rem !important}.row-gap-sm-4{row-gap:1.5rem !important}.row-gap-sm-5{row-gap:3rem !important}.column-gap-sm-0{column-gap:0 !important}.column-gap-sm-1{column-gap:.25rem !important}.column-gap-sm-2{column-gap:.5rem !important}.column-gap-sm-3{column-gap:1rem !important}.column-gap-sm-4{column-gap:1.5rem !important}.column-gap-sm-5{column-gap:3rem !important}.text-sm-start{text-align:left !important}.text-sm-end{text-align:right !important}.text-sm-center{text-align:center !important}}@media(min-width: 768px){.float-md-start{float:left !important}.float-md-end{float:right !important}.float-md-none{float:none !important}.object-fit-md-contain{object-fit:contain !important}.object-fit-md-cover{object-fit:cover !important}.object-fit-md-fill{object-fit:fill !important}.object-fit-md-scale{object-fit:scale-down !important}.object-fit-md-none{object-fit:none !important}.d-md-inline{display:inline !important}.d-md-inline-block{display:inline-block !important}.d-md-block{display:block !important}.d-md-grid{display:grid !important}.d-md-inline-grid{display:inline-grid !important}.d-md-table{display:table !important}.d-md-table-row{display:table-row !important}.d-md-table-cell{display:table-cell !important}.d-md-flex{display:flex !important}.d-md-inline-flex{display:inline-flex !important}.d-md-none{display:none !important}.flex-md-fill{flex:1 1 auto !important}.flex-md-row{flex-direction:row !important}.flex-md-column{flex-direction:column !important}.flex-md-row-reverse{flex-direction:row-reverse !important}.flex-md-column-reverse{flex-direction:column-reverse !important}.flex-md-grow-0{flex-grow:0 !important}.flex-md-grow-1{flex-grow:1 !important}.flex-md-shrink-0{flex-shrink:0 !important}.flex-md-shrink-1{flex-shrink:1 !important}.flex-md-wrap{flex-wrap:wrap !important}.flex-md-nowrap{flex-wrap:nowrap !important}.flex-md-wrap-reverse{flex-wrap:wrap-reverse !important}.justify-content-md-start{justify-content:flex-start !important}.justify-content-md-end{justify-content:flex-end !important}.justify-content-md-center{justify-content:center !important}.justify-content-md-between{justify-content:space-between !important}.justify-content-md-around{justify-content:space-around !important}.justify-content-md-evenly{justify-content:space-evenly !important}.align-items-md-start{align-items:flex-start !important}.align-items-md-end{align-items:flex-end !important}.align-items-md-center{align-items:center !important}.align-items-md-baseline{align-items:baseline !important}.align-items-md-stretch{align-items:stretch !important}.align-content-md-start{align-content:flex-start !important}.align-content-md-end{align-content:flex-end !important}.align-content-md-center{align-content:center !important}.align-content-md-between{align-content:space-between !important}.align-content-md-around{align-content:space-around !important}.align-content-md-stretch{align-content:stretch !important}.align-self-md-auto{align-self:auto !important}.align-self-md-start{align-self:flex-start !important}.align-self-md-end{align-self:flex-end !important}.align-self-md-center{align-self:center !important}.align-self-md-baseline{align-self:baseline !important}.align-self-md-stretch{align-self:stretch !important}.order-md-first{order:-1 !important}.order-md-0{order:0 !important}.order-md-1{order:1 !important}.order-md-2{order:2 !important}.order-md-3{order:3 !important}.order-md-4{order:4 !important}.order-md-5{order:5 !important}.order-md-last{order:6 !important}.m-md-0{margin:0 !important}.m-md-1{margin:.25rem !important}.m-md-2{margin:.5rem !important}.m-md-3{margin:1rem !important}.m-md-4{margin:1.5rem !important}.m-md-5{margin:3rem !important}.m-md-auto{margin:auto !important}.mx-md-0{margin-right:0 !important;margin-left:0 !important}.mx-md-1{margin-right:.25rem !important;margin-left:.25rem !important}.mx-md-2{margin-right:.5rem !important;margin-left:.5rem !important}.mx-md-3{margin-right:1rem !important;margin-left:1rem !important}.mx-md-4{margin-right:1.5rem !important;margin-left:1.5rem !important}.mx-md-5{margin-right:3rem !important;margin-left:3rem !important}.mx-md-auto{margin-right:auto !important;margin-left:auto !important}.my-md-0{margin-top:0 !important;margin-bottom:0 !important}.my-md-1{margin-top:.25rem !important;margin-bottom:.25rem !important}.my-md-2{margin-top:.5rem !important;margin-bottom:.5rem !important}.my-md-3{margin-top:1rem !important;margin-bottom:1rem !important}.my-md-4{margin-top:1.5rem !important;margin-bottom:1.5rem !important}.my-md-5{margin-top:3rem !important;margin-bottom:3rem !important}.my-md-auto{margin-top:auto !important;margin-bottom:auto !important}.mt-md-0{margin-top:0 !important}.mt-md-1{margin-top:.25rem !important}.mt-md-2{margin-top:.5rem !important}.mt-md-3{margin-top:1rem !important}.mt-md-4{margin-top:1.5rem !important}.mt-md-5{margin-top:3rem !important}.mt-md-auto{margin-top:auto !important}.me-md-0{margin-right:0 !important}.me-md-1{margin-right:.25rem !important}.me-md-2{margin-right:.5rem !important}.me-md-3{margin-right:1rem !important}.me-md-4{margin-right:1.5rem !important}.me-md-5{margin-right:3rem !important}.me-md-auto{margin-right:auto !important}.mb-md-0{margin-bottom:0 !important}.mb-md-1{margin-bottom:.25rem !important}.mb-md-2{margin-bottom:.5rem !important}.mb-md-3{margin-bottom:1rem !important}.mb-md-4{margin-bottom:1.5rem !important}.mb-md-5{margin-bottom:3rem !important}.mb-md-auto{margin-bottom:auto !important}.ms-md-0{margin-left:0 !important}.ms-md-1{margin-left:.25rem !important}.ms-md-2{margin-left:.5rem !important}.ms-md-3{margin-left:1rem !important}.ms-md-4{margin-left:1.5rem !important}.ms-md-5{margin-left:3rem !important}.ms-md-auto{margin-left:auto !important}.p-md-0{padding:0 !important}.p-md-1{padding:.25rem !important}.p-md-2{padding:.5rem !important}.p-md-3{padding:1rem !important}.p-md-4{padding:1.5rem !important}.p-md-5{padding:3rem !important}.px-md-0{padding-right:0 !important;padding-left:0 !important}.px-md-1{padding-right:.25rem !important;padding-left:.25rem !important}.px-md-2{padding-right:.5rem !important;padding-left:.5rem !important}.px-md-3{padding-right:1rem !important;padding-left:1rem !important}.px-md-4{padding-right:1.5rem !important;padding-left:1.5rem !important}.px-md-5{padding-right:3rem !important;padding-left:3rem !important}.py-md-0{padding-top:0 !important;padding-bottom:0 !important}.py-md-1{padding-top:.25rem !important;padding-bottom:.25rem !important}.py-md-2{padding-top:.5rem !important;padding-bottom:.5rem !important}.py-md-3{padding-top:1rem !important;padding-bottom:1rem !important}.py-md-4{padding-top:1.5rem !important;padding-bottom:1.5rem !important}.py-md-5{padding-top:3rem !important;padding-bottom:3rem !important}.pt-md-0{padding-top:0 !important}.pt-md-1{padding-top:.25rem !important}.pt-md-2{padding-top:.5rem !important}.pt-md-3{padding-top:1rem !important}.pt-md-4{padding-top:1.5rem !important}.pt-md-5{padding-top:3rem !important}.pe-md-0{padding-right:0 !important}.pe-md-1{padding-right:.25rem !important}.pe-md-2{padding-right:.5rem !important}.pe-md-3{padding-right:1rem !important}.pe-md-4{padding-right:1.5rem !important}.pe-md-5{padding-right:3rem !important}.pb-md-0{padding-bottom:0 !important}.pb-md-1{padding-bottom:.25rem !important}.pb-md-2{padding-bottom:.5rem !important}.pb-md-3{padding-bottom:1rem !important}.pb-md-4{padding-bottom:1.5rem !important}.pb-md-5{padding-bottom:3rem !important}.ps-md-0{padding-left:0 !important}.ps-md-1{padding-left:.25rem !important}.ps-md-2{padding-left:.5rem !important}.ps-md-3{padding-left:1rem !important}.ps-md-4{padding-left:1.5rem !important}.ps-md-5{padding-left:3rem !important}.gap-md-0{gap:0 !important}.gap-md-1{gap:.25rem !important}.gap-md-2{gap:.5rem !important}.gap-md-3{gap:1rem !important}.gap-md-4{gap:1.5rem !important}.gap-md-5{gap:3rem !important}.row-gap-md-0{row-gap:0 !important}.row-gap-md-1{row-gap:.25rem !important}.row-gap-md-2{row-gap:.5rem !important}.row-gap-md-3{row-gap:1rem !important}.row-gap-md-4{row-gap:1.5rem !important}.row-gap-md-5{row-gap:3rem !important}.column-gap-md-0{column-gap:0 !important}.column-gap-md-1{column-gap:.25rem !important}.column-gap-md-2{column-gap:.5rem !important}.column-gap-md-3{column-gap:1rem !important}.column-gap-md-4{column-gap:1.5rem !important}.column-gap-md-5{column-gap:3rem !important}.text-md-start{text-align:left !important}.text-md-end{text-align:right !important}.text-md-center{text-align:center !important}}@media(min-width: 992px){.float-lg-start{float:left !important}.float-lg-end{float:right !important}.float-lg-none{float:none !important}.object-fit-lg-contain{object-fit:contain !important}.object-fit-lg-cover{object-fit:cover !important}.object-fit-lg-fill{object-fit:fill !important}.object-fit-lg-scale{object-fit:scale-down !important}.object-fit-lg-none{object-fit:none !important}.d-lg-inline{display:inline !important}.d-lg-inline-block{display:inline-block !important}.d-lg-block{display:block !important}.d-lg-grid{display:grid !important}.d-lg-inline-grid{display:inline-grid !important}.d-lg-table{display:table !important}.d-lg-table-row{display:table-row !important}.d-lg-table-cell{display:table-cell !important}.d-lg-flex{display:flex !important}.d-lg-inline-flex{display:inline-flex !important}.d-lg-none{display:none !important}.flex-lg-fill{flex:1 1 auto !important}.flex-lg-row{flex-direction:row !important}.flex-lg-column{flex-direction:column !important}.flex-lg-row-reverse{flex-direction:row-reverse !important}.flex-lg-column-reverse{flex-direction:column-reverse !important}.flex-lg-grow-0{flex-grow:0 !important}.flex-lg-grow-1{flex-grow:1 !important}.flex-lg-shrink-0{flex-shrink:0 !important}.flex-lg-shrink-1{flex-shrink:1 !important}.flex-lg-wrap{flex-wrap:wrap !important}.flex-lg-nowrap{flex-wrap:nowrap !important}.flex-lg-wrap-reverse{flex-wrap:wrap-reverse !important}.justify-content-lg-start{justify-content:flex-start !important}.justify-content-lg-end{justify-content:flex-end !important}.justify-content-lg-center{justify-content:center !important}.justify-content-lg-between{justify-content:space-between !important}.justify-content-lg-around{justify-content:space-around !important}.justify-content-lg-evenly{justify-content:space-evenly !important}.align-items-lg-start{align-items:flex-start !important}.align-items-lg-end{align-items:flex-end !important}.align-items-lg-center{align-items:center !important}.align-items-lg-baseline{align-items:baseline !important}.align-items-lg-stretch{align-items:stretch !important}.align-content-lg-start{align-content:flex-start !important}.align-content-lg-end{align-content:flex-end !important}.align-content-lg-center{align-content:center !important}.align-content-lg-between{align-content:space-between !important}.align-content-lg-around{align-content:space-around !important}.align-content-lg-stretch{align-content:stretch !important}.align-self-lg-auto{align-self:auto !important}.align-self-lg-start{align-self:flex-start !important}.align-self-lg-end{align-self:flex-end !important}.align-self-lg-center{align-self:center !important}.align-self-lg-baseline{align-self:baseline !important}.align-self-lg-stretch{align-self:stretch !important}.order-lg-first{order:-1 !important}.order-lg-0{order:0 !important}.order-lg-1{order:1 !important}.order-lg-2{order:2 !important}.order-lg-3{order:3 !important}.order-lg-4{order:4 !important}.order-lg-5{order:5 !important}.order-lg-last{order:6 !important}.m-lg-0{margin:0 !important}.m-lg-1{margin:.25rem !important}.m-lg-2{margin:.5rem !important}.m-lg-3{margin:1rem !important}.m-lg-4{margin:1.5rem !important}.m-lg-5{margin:3rem !important}.m-lg-auto{margin:auto !important}.mx-lg-0{margin-right:0 !important;margin-left:0 !important}.mx-lg-1{margin-right:.25rem !important;margin-left:.25rem !important}.mx-lg-2{margin-right:.5rem !important;margin-left:.5rem !important}.mx-lg-3{margin-right:1rem !important;margin-left:1rem !important}.mx-lg-4{margin-right:1.5rem !important;margin-left:1.5rem !important}.mx-lg-5{margin-right:3rem !important;margin-left:3rem !important}.mx-lg-auto{margin-right:auto !important;margin-left:auto !important}.my-lg-0{margin-top:0 !important;margin-bottom:0 !important}.my-lg-1{margin-top:.25rem !important;margin-bottom:.25rem !important}.my-lg-2{margin-top:.5rem !important;margin-bottom:.5rem !important}.my-lg-3{margin-top:1rem !important;margin-bottom:1rem !important}.my-lg-4{margin-top:1.5rem !important;margin-bottom:1.5rem !important}.my-lg-5{margin-top:3rem !important;margin-bottom:3rem !important}.my-lg-auto{margin-top:auto !important;margin-bottom:auto !important}.mt-lg-0{margin-top:0 !important}.mt-lg-1{margin-top:.25rem !important}.mt-lg-2{margin-top:.5rem !important}.mt-lg-3{margin-top:1rem !important}.mt-lg-4{margin-top:1.5rem !important}.mt-lg-5{margin-top:3rem !important}.mt-lg-auto{margin-top:auto !important}.me-lg-0{margin-right:0 !important}.me-lg-1{margin-right:.25rem !important}.me-lg-2{margin-right:.5rem !important}.me-lg-3{margin-right:1rem !important}.me-lg-4{margin-right:1.5rem !important}.me-lg-5{margin-right:3rem !important}.me-lg-auto{margin-right:auto !important}.mb-lg-0{margin-bottom:0 !important}.mb-lg-1{margin-bottom:.25rem !important}.mb-lg-2{margin-bottom:.5rem !important}.mb-lg-3{margin-bottom:1rem !important}.mb-lg-4{margin-bottom:1.5rem !important}.mb-lg-5{margin-bottom:3rem !important}.mb-lg-auto{margin-bottom:auto !important}.ms-lg-0{margin-left:0 !important}.ms-lg-1{margin-left:.25rem !important}.ms-lg-2{margin-left:.5rem !important}.ms-lg-3{margin-left:1rem !important}.ms-lg-4{margin-left:1.5rem !important}.ms-lg-5{margin-left:3rem !important}.ms-lg-auto{margin-left:auto !important}.p-lg-0{padding:0 !important}.p-lg-1{padding:.25rem !important}.p-lg-2{padding:.5rem !important}.p-lg-3{padding:1rem !important}.p-lg-4{padding:1.5rem !important}.p-lg-5{padding:3rem !important}.px-lg-0{padding-right:0 !important;padding-left:0 !important}.px-lg-1{padding-right:.25rem !important;padding-left:.25rem !important}.px-lg-2{padding-right:.5rem !important;padding-left:.5rem !important}.px-lg-3{padding-right:1rem !important;padding-left:1rem !important}.px-lg-4{padding-right:1.5rem !important;padding-left:1.5rem !important}.px-lg-5{padding-right:3rem !important;padding-left:3rem !important}.py-lg-0{padding-top:0 !important;padding-bottom:0 !important}.py-lg-1{padding-top:.25rem !important;padding-bottom:.25rem !important}.py-lg-2{padding-top:.5rem !important;padding-bottom:.5rem !important}.py-lg-3{padding-top:1rem !important;padding-bottom:1rem !important}.py-lg-4{padding-top:1.5rem !important;padding-bottom:1.5rem !important}.py-lg-5{padding-top:3rem !important;padding-bottom:3rem !important}.pt-lg-0{padding-top:0 !important}.pt-lg-1{padding-top:.25rem !important}.pt-lg-2{padding-top:.5rem !important}.pt-lg-3{padding-top:1rem !important}.pt-lg-4{padding-top:1.5rem !important}.pt-lg-5{padding-top:3rem !important}.pe-lg-0{padding-right:0 !important}.pe-lg-1{padding-right:.25rem !important}.pe-lg-2{padding-right:.5rem !important}.pe-lg-3{padding-right:1rem !important}.pe-lg-4{padding-right:1.5rem !important}.pe-lg-5{padding-right:3rem !important}.pb-lg-0{padding-bottom:0 !important}.pb-lg-1{padding-bottom:.25rem !important}.pb-lg-2{padding-bottom:.5rem !important}.pb-lg-3{padding-bottom:1rem !important}.pb-lg-4{padding-bottom:1.5rem !important}.pb-lg-5{padding-bottom:3rem !important}.ps-lg-0{padding-left:0 !important}.ps-lg-1{padding-left:.25rem !important}.ps-lg-2{padding-left:.5rem !important}.ps-lg-3{padding-left:1rem !important}.ps-lg-4{padding-left:1.5rem !important}.ps-lg-5{padding-left:3rem !important}.gap-lg-0{gap:0 !important}.gap-lg-1{gap:.25rem !important}.gap-lg-2{gap:.5rem !important}.gap-lg-3{gap:1rem !important}.gap-lg-4{gap:1.5rem !important}.gap-lg-5{gap:3rem !important}.row-gap-lg-0{row-gap:0 !important}.row-gap-lg-1{row-gap:.25rem !important}.row-gap-lg-2{row-gap:.5rem !important}.row-gap-lg-3{row-gap:1rem !important}.row-gap-lg-4{row-gap:1.5rem !important}.row-gap-lg-5{row-gap:3rem !important}.column-gap-lg-0{column-gap:0 !important}.column-gap-lg-1{column-gap:.25rem !important}.column-gap-lg-2{column-gap:.5rem !important}.column-gap-lg-3{column-gap:1rem !important}.column-gap-lg-4{column-gap:1.5rem !important}.column-gap-lg-5{column-gap:3rem !important}.text-lg-start{text-align:left !important}.text-lg-end{text-align:right !important}.text-lg-center{text-align:center !important}}@media(min-width: 1200px){.float-xl-start{float:left !important}.float-xl-end{float:right !important}.float-xl-none{float:none !important}.object-fit-xl-contain{object-fit:contain !important}.object-fit-xl-cover{object-fit:cover !important}.object-fit-xl-fill{object-fit:fill !important}.object-fit-xl-scale{object-fit:scale-down !important}.object-fit-xl-none{object-fit:none !important}.d-xl-inline{display:inline !important}.d-xl-inline-block{display:inline-block !important}.d-xl-block{display:block !important}.d-xl-grid{display:grid !important}.d-xl-inline-grid{display:inline-grid !important}.d-xl-table{display:table !important}.d-xl-table-row{display:table-row !important}.d-xl-table-cell{display:table-cell !important}.d-xl-flex{display:flex !important}.d-xl-inline-flex{display:inline-flex !important}.d-xl-none{display:none !important}.flex-xl-fill{flex:1 1 auto !important}.flex-xl-row{flex-direction:row !important}.flex-xl-column{flex-direction:column !important}.flex-xl-row-reverse{flex-direction:row-reverse !important}.flex-xl-column-reverse{flex-direction:column-reverse !important}.flex-xl-grow-0{flex-grow:0 !important}.flex-xl-grow-1{flex-grow:1 !important}.flex-xl-shrink-0{flex-shrink:0 !important}.flex-xl-shrink-1{flex-shrink:1 !important}.flex-xl-wrap{flex-wrap:wrap !important}.flex-xl-nowrap{flex-wrap:nowrap !important}.flex-xl-wrap-reverse{flex-wrap:wrap-reverse !important}.justify-content-xl-start{justify-content:flex-start !important}.justify-content-xl-end{justify-content:flex-end !important}.justify-content-xl-center{justify-content:center !important}.justify-content-xl-between{justify-content:space-between !important}.justify-content-xl-around{justify-content:space-around !important}.justify-content-xl-evenly{justify-content:space-evenly !important}.align-items-xl-start{align-items:flex-start !important}.align-items-xl-end{align-items:flex-end !important}.align-items-xl-center{align-items:center !important}.align-items-xl-baseline{align-items:baseline !important}.align-items-xl-stretch{align-items:stretch !important}.align-content-xl-start{align-content:flex-start !important}.align-content-xl-end{align-content:flex-end !important}.align-content-xl-center{align-content:center !important}.align-content-xl-between{align-content:space-between !important}.align-content-xl-around{align-content:space-around !important}.align-content-xl-stretch{align-content:stretch !important}.align-self-xl-auto{align-self:auto !important}.align-self-xl-start{align-self:flex-start !important}.align-self-xl-end{align-self:flex-end !important}.align-self-xl-center{align-self:center !important}.align-self-xl-baseline{align-self:baseline !important}.align-self-xl-stretch{align-self:stretch !important}.order-xl-first{order:-1 !important}.order-xl-0{order:0 !important}.order-xl-1{order:1 !important}.order-xl-2{order:2 !important}.order-xl-3{order:3 !important}.order-xl-4{order:4 !important}.order-xl-5{order:5 !important}.order-xl-last{order:6 !important}.m-xl-0{margin:0 !important}.m-xl-1{margin:.25rem !important}.m-xl-2{margin:.5rem !important}.m-xl-3{margin:1rem !important}.m-xl-4{margin:1.5rem !important}.m-xl-5{margin:3rem !important}.m-xl-auto{margin:auto !important}.mx-xl-0{margin-right:0 !important;margin-left:0 !important}.mx-xl-1{margin-right:.25rem !important;margin-left:.25rem !important}.mx-xl-2{margin-right:.5rem !important;margin-left:.5rem !important}.mx-xl-3{margin-right:1rem !important;margin-left:1rem !important}.mx-xl-4{margin-right:1.5rem !important;margin-left:1.5rem !important}.mx-xl-5{margin-right:3rem !important;margin-left:3rem !important}.mx-xl-auto{margin-right:auto !important;margin-left:auto !important}.my-xl-0{margin-top:0 !important;margin-bottom:0 !important}.my-xl-1{margin-top:.25rem !important;margin-bottom:.25rem !important}.my-xl-2{margin-top:.5rem !important;margin-bottom:.5rem !important}.my-xl-3{margin-top:1rem !important;margin-bottom:1rem !important}.my-xl-4{margin-top:1.5rem !important;margin-bottom:1.5rem !important}.my-xl-5{margin-top:3rem !important;margin-bottom:3rem !important}.my-xl-auto{margin-top:auto !important;margin-bottom:auto !important}.mt-xl-0{margin-top:0 !important}.mt-xl-1{margin-top:.25rem !important}.mt-xl-2{margin-top:.5rem !important}.mt-xl-3{margin-top:1rem !important}.mt-xl-4{margin-top:1.5rem !important}.mt-xl-5{margin-top:3rem !important}.mt-xl-auto{margin-top:auto !important}.me-xl-0{margin-right:0 !important}.me-xl-1{margin-right:.25rem !important}.me-xl-2{margin-right:.5rem !important}.me-xl-3{margin-right:1rem !important}.me-xl-4{margin-right:1.5rem !important}.me-xl-5{margin-right:3rem !important}.me-xl-auto{margin-right:auto !important}.mb-xl-0{margin-bottom:0 !important}.mb-xl-1{margin-bottom:.25rem !important}.mb-xl-2{margin-bottom:.5rem !important}.mb-xl-3{margin-bottom:1rem !important}.mb-xl-4{margin-bottom:1.5rem !important}.mb-xl-5{margin-bottom:3rem !important}.mb-xl-auto{margin-bottom:auto !important}.ms-xl-0{margin-left:0 !important}.ms-xl-1{margin-left:.25rem !important}.ms-xl-2{margin-left:.5rem !important}.ms-xl-3{margin-left:1rem !important}.ms-xl-4{margin-left:1.5rem !important}.ms-xl-5{margin-left:3rem !important}.ms-xl-auto{margin-left:auto !important}.p-xl-0{padding:0 !important}.p-xl-1{padding:.25rem !important}.p-xl-2{padding:.5rem !important}.p-xl-3{padding:1rem !important}.p-xl-4{padding:1.5rem !important}.p-xl-5{padding:3rem !important}.px-xl-0{padding-right:0 !important;padding-left:0 !important}.px-xl-1{padding-right:.25rem !important;padding-left:.25rem !important}.px-xl-2{padding-right:.5rem !important;padding-left:.5rem !important}.px-xl-3{padding-right:1rem !important;padding-left:1rem !important}.px-xl-4{padding-right:1.5rem !important;padding-left:1.5rem !important}.px-xl-5{padding-right:3rem !important;padding-left:3rem !important}.py-xl-0{padding-top:0 !important;padding-bottom:0 !important}.py-xl-1{padding-top:.25rem !important;padding-bottom:.25rem !important}.py-xl-2{padding-top:.5rem !important;padding-bottom:.5rem !important}.py-xl-3{padding-top:1rem !important;padding-bottom:1rem !important}.py-xl-4{padding-top:1.5rem !important;padding-bottom:1.5rem !important}.py-xl-5{padding-top:3rem !important;padding-bottom:3rem !important}.pt-xl-0{padding-top:0 !important}.pt-xl-1{padding-top:.25rem !important}.pt-xl-2{padding-top:.5rem !important}.pt-xl-3{padding-top:1rem !important}.pt-xl-4{padding-top:1.5rem !important}.pt-xl-5{padding-top:3rem !important}.pe-xl-0{padding-right:0 !important}.pe-xl-1{padding-right:.25rem !important}.pe-xl-2{padding-right:.5rem !important}.pe-xl-3{padding-right:1rem !important}.pe-xl-4{padding-right:1.5rem !important}.pe-xl-5{padding-right:3rem !important}.pb-xl-0{padding-bottom:0 !important}.pb-xl-1{padding-bottom:.25rem !important}.pb-xl-2{padding-bottom:.5rem !important}.pb-xl-3{padding-bottom:1rem !important}.pb-xl-4{padding-bottom:1.5rem !important}.pb-xl-5{padding-bottom:3rem !important}.ps-xl-0{padding-left:0 !important}.ps-xl-1{padding-left:.25rem !important}.ps-xl-2{padding-left:.5rem !important}.ps-xl-3{padding-left:1rem !important}.ps-xl-4{padding-left:1.5rem !important}.ps-xl-5{padding-left:3rem !important}.gap-xl-0{gap:0 !important}.gap-xl-1{gap:.25rem !important}.gap-xl-2{gap:.5rem !important}.gap-xl-3{gap:1rem !important}.gap-xl-4{gap:1.5rem !important}.gap-xl-5{gap:3rem !important}.row-gap-xl-0{row-gap:0 !important}.row-gap-xl-1{row-gap:.25rem !important}.row-gap-xl-2{row-gap:.5rem !important}.row-gap-xl-3{row-gap:1rem !important}.row-gap-xl-4{row-gap:1.5rem !important}.row-gap-xl-5{row-gap:3rem !important}.column-gap-xl-0{column-gap:0 !important}.column-gap-xl-1{column-gap:.25rem !important}.column-gap-xl-2{column-gap:.5rem !important}.column-gap-xl-3{column-gap:1rem !important}.column-gap-xl-4{column-gap:1.5rem !important}.column-gap-xl-5{column-gap:3rem !important}.text-xl-start{text-align:left !important}.text-xl-end{text-align:right !important}.text-xl-center{text-align:center !important}}@media(min-width: 1400px){.float-xxl-start{float:left !important}.float-xxl-end{float:right !important}.float-xxl-none{float:none !important}.object-fit-xxl-contain{object-fit:contain !important}.object-fit-xxl-cover{object-fit:cover !important}.object-fit-xxl-fill{object-fit:fill !important}.object-fit-xxl-scale{object-fit:scale-down !important}.object-fit-xxl-none{object-fit:none !important}.d-xxl-inline{display:inline !important}.d-xxl-inline-block{display:inline-block !important}.d-xxl-block{display:block !important}.d-xxl-grid{display:grid !important}.d-xxl-inline-grid{display:inline-grid !important}.d-xxl-table{display:table !important}.d-xxl-table-row{display:table-row !important}.d-xxl-table-cell{display:table-cell !important}.d-xxl-flex{display:flex !important}.d-xxl-inline-flex{display:inline-flex !important}.d-xxl-none{display:none !important}.flex-xxl-fill{flex:1 1 auto !important}.flex-xxl-row{flex-direction:row !important}.flex-xxl-column{flex-direction:column !important}.flex-xxl-row-reverse{flex-direction:row-reverse !important}.flex-xxl-column-reverse{flex-direction:column-reverse !important}.flex-xxl-grow-0{flex-grow:0 !important}.flex-xxl-grow-1{flex-grow:1 !important}.flex-xxl-shrink-0{flex-shrink:0 !important}.flex-xxl-shrink-1{flex-shrink:1 !important}.flex-xxl-wrap{flex-wrap:wrap !important}.flex-xxl-nowrap{flex-wrap:nowrap !important}.flex-xxl-wrap-reverse{flex-wrap:wrap-reverse !important}.justify-content-xxl-start{justify-content:flex-start !important}.justify-content-xxl-end{justify-content:flex-end !important}.justify-content-xxl-center{justify-content:center !important}.justify-content-xxl-between{justify-content:space-between !important}.justify-content-xxl-around{justify-content:space-around !important}.justify-content-xxl-evenly{justify-content:space-evenly !important}.align-items-xxl-start{align-items:flex-start !important}.align-items-xxl-end{align-items:flex-end !important}.align-items-xxl-center{align-items:center !important}.align-items-xxl-baseline{align-items:baseline !important}.align-items-xxl-stretch{align-items:stretch !important}.align-content-xxl-start{align-content:flex-start !important}.align-content-xxl-end{align-content:flex-end !important}.align-content-xxl-center{align-content:center !important}.align-content-xxl-between{align-content:space-between !important}.align-content-xxl-around{align-content:space-around !important}.align-content-xxl-stretch{align-content:stretch !important}.align-self-xxl-auto{align-self:auto !important}.align-self-xxl-start{align-self:flex-start !important}.align-self-xxl-end{align-self:flex-end !important}.align-self-xxl-center{align-self:center !important}.align-self-xxl-baseline{align-self:baseline !important}.align-self-xxl-stretch{align-self:stretch !important}.order-xxl-first{order:-1 !important}.order-xxl-0{order:0 !important}.order-xxl-1{order:1 !important}.order-xxl-2{order:2 !important}.order-xxl-3{order:3 !important}.order-xxl-4{order:4 !important}.order-xxl-5{order:5 !important}.order-xxl-last{order:6 !important}.m-xxl-0{margin:0 !important}.m-xxl-1{margin:.25rem !important}.m-xxl-2{margin:.5rem !important}.m-xxl-3{margin:1rem !important}.m-xxl-4{margin:1.5rem !important}.m-xxl-5{margin:3rem !important}.m-xxl-auto{margin:auto !important}.mx-xxl-0{margin-right:0 !important;margin-left:0 !important}.mx-xxl-1{margin-right:.25rem !important;margin-left:.25rem !important}.mx-xxl-2{margin-right:.5rem !important;margin-left:.5rem !important}.mx-xxl-3{margin-right:1rem !important;margin-left:1rem !important}.mx-xxl-4{margin-right:1.5rem !important;margin-left:1.5rem !important}.mx-xxl-5{margin-right:3rem !important;margin-left:3rem !important}.mx-xxl-auto{margin-right:auto !important;margin-left:auto !important}.my-xxl-0{margin-top:0 !important;margin-bottom:0 !important}.my-xxl-1{margin-top:.25rem !important;margin-bottom:.25rem !important}.my-xxl-2{margin-top:.5rem !important;margin-bottom:.5rem !important}.my-xxl-3{margin-top:1rem !important;margin-bottom:1rem !important}.my-xxl-4{margin-top:1.5rem !important;margin-bottom:1.5rem !important}.my-xxl-5{margin-top:3rem !important;margin-bottom:3rem !important}.my-xxl-auto{margin-top:auto !important;margin-bottom:auto !important}.mt-xxl-0{margin-top:0 !important}.mt-xxl-1{margin-top:.25rem !important}.mt-xxl-2{margin-top:.5rem !important}.mt-xxl-3{margin-top:1rem !important}.mt-xxl-4{margin-top:1.5rem !important}.mt-xxl-5{margin-top:3rem !important}.mt-xxl-auto{margin-top:auto !important}.me-xxl-0{margin-right:0 !important}.me-xxl-1{margin-right:.25rem !important}.me-xxl-2{margin-right:.5rem !important}.me-xxl-3{margin-right:1rem !important}.me-xxl-4{margin-right:1.5rem !important}.me-xxl-5{margin-right:3rem !important}.me-xxl-auto{margin-right:auto !important}.mb-xxl-0{margin-bottom:0 !important}.mb-xxl-1{margin-bottom:.25rem !important}.mb-xxl-2{margin-bottom:.5rem !important}.mb-xxl-3{margin-bottom:1rem !important}.mb-xxl-4{margin-bottom:1.5rem !important}.mb-xxl-5{margin-bottom:3rem !important}.mb-xxl-auto{margin-bottom:auto !important}.ms-xxl-0{margin-left:0 !important}.ms-xxl-1{margin-left:.25rem !important}.ms-xxl-2{margin-left:.5rem !important}.ms-xxl-3{margin-left:1rem !important}.ms-xxl-4{margin-left:1.5rem !important}.ms-xxl-5{margin-left:3rem !important}.ms-xxl-auto{margin-left:auto !important}.p-xxl-0{padding:0 !important}.p-xxl-1{padding:.25rem !important}.p-xxl-2{padding:.5rem !important}.p-xxl-3{padding:1rem !important}.p-xxl-4{padding:1.5rem !important}.p-xxl-5{padding:3rem !important}.px-xxl-0{padding-right:0 !important;padding-left:0 !important}.px-xxl-1{padding-right:.25rem !important;padding-left:.25rem !important}.px-xxl-2{padding-right:.5rem !important;padding-left:.5rem !important}.px-xxl-3{padding-right:1rem !important;padding-left:1rem !important}.px-xxl-4{padding-right:1.5rem !important;padding-left:1.5rem !important}.px-xxl-5{padding-right:3rem !important;padding-left:3rem !important}.py-xxl-0{padding-top:0 !important;padding-bottom:0 !important}.py-xxl-1{padding-top:.25rem !important;padding-bottom:.25rem !important}.py-xxl-2{padding-top:.5rem !important;padding-bottom:.5rem !important}.py-xxl-3{padding-top:1rem !important;padding-bottom:1rem !important}.py-xxl-4{padding-top:1.5rem !important;padding-bottom:1.5rem !important}.py-xxl-5{padding-top:3rem !important;padding-bottom:3rem !important}.pt-xxl-0{padding-top:0 !important}.pt-xxl-1{padding-top:.25rem !important}.pt-xxl-2{padding-top:.5rem !important}.pt-xxl-3{padding-top:1rem !important}.pt-xxl-4{padding-top:1.5rem !important}.pt-xxl-5{padding-top:3rem !important}.pe-xxl-0{padding-right:0 !important}.pe-xxl-1{padding-right:.25rem !important}.pe-xxl-2{padding-right:.5rem !important}.pe-xxl-3{padding-right:1rem !important}.pe-xxl-4{padding-right:1.5rem !important}.pe-xxl-5{padding-right:3rem !important}.pb-xxl-0{padding-bottom:0 !important}.pb-xxl-1{padding-bottom:.25rem !important}.pb-xxl-2{padding-bottom:.5rem !important}.pb-xxl-3{padding-bottom:1rem !important}.pb-xxl-4{padding-bottom:1.5rem !important}.pb-xxl-5{padding-bottom:3rem !important}.ps-xxl-0{padding-left:0 !important}.ps-xxl-1{padding-left:.25rem !important}.ps-xxl-2{padding-left:.5rem !important}.ps-xxl-3{padding-left:1rem !important}.ps-xxl-4{padding-left:1.5rem !important}.ps-xxl-5{padding-left:3rem !important}.gap-xxl-0{gap:0 !important}.gap-xxl-1{gap:.25rem !important}.gap-xxl-2{gap:.5rem !important}.gap-xxl-3{gap:1rem !important}.gap-xxl-4{gap:1.5rem !important}.gap-xxl-5{gap:3rem !important}.row-gap-xxl-0{row-gap:0 !important}.row-gap-xxl-1{row-gap:.25rem !important}.row-gap-xxl-2{row-gap:.5rem !important}.row-gap-xxl-3{row-gap:1rem !important}.row-gap-xxl-4{row-gap:1.5rem !important}.row-gap-xxl-5{row-gap:3rem !important}.column-gap-xxl-0{column-gap:0 !important}.column-gap-xxl-1{column-gap:.25rem !important}.column-gap-xxl-2{column-gap:.5rem !important}.column-gap-xxl-3{column-gap:1rem !important}.column-gap-xxl-4{column-gap:1.5rem !important}.column-gap-xxl-5{column-gap:3rem !important}.text-xxl-start{text-align:left !important}.text-xxl-end{text-align:right !important}.text-xxl-center{text-align:center !important}}.bg-default{color:#fff}.bg-primary{color:#000}.bg-secondary{color:#fff}.bg-success{color:#fff}.bg-info{color:#fff}.bg-warning{color:#fff}.bg-danger{color:#fff}.bg-light{color:#000}.bg-dark{color:#fff}@media(min-width: 1200px){.fs-1{font-size:2rem !important}.fs-2{font-size:1.65rem !important}.fs-3{font-size:1.45rem !important}}@media print{.d-print-inline{display:inline !important}.d-print-inline-block{display:inline-block !important}.d-print-block{display:block !important}.d-print-grid{display:grid !important}.d-print-inline-grid{display:inline-grid !important}.d-print-table{display:table !important}.d-print-table-row{display:table-row !important}.d-print-table-cell{display:table-cell !important}.d-print-flex{display:flex !important}.d-print-inline-flex{display:inline-flex !important}.d-print-none{display:none !important}}:root{--bslib-spacer: 1rem;--bslib-mb-spacer: var(--bslib-spacer, 1rem)}.bslib-mb-spacing{margin-bottom:var(--bslib-mb-spacer)}.bslib-gap-spacing{gap:var(--bslib-mb-spacer)}.bslib-gap-spacing>.bslib-mb-spacing,.bslib-gap-spacing>.form-group,.bslib-gap-spacing>p,.bslib-gap-spacing>pre{margin-bottom:0}.html-fill-container>.html-fill-item.bslib-mb-spacing{margin-bottom:0}.tab-content>.tab-pane.html-fill-container{display:none}.tab-content>.active.html-fill-container{display:flex}.tab-content.html-fill-container{padding:0}.bg-blue{--bslib-color-bg: #2780e3;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-blue{--bslib-color-fg: #2780e3;color:var(--bslib-color-fg)}.bg-indigo{--bslib-color-bg: #6610f2;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-indigo{--bslib-color-fg: #6610f2;color:var(--bslib-color-fg)}.bg-purple{--bslib-color-bg: #613d7c;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-purple{--bslib-color-fg: #613d7c;color:var(--bslib-color-fg)}.bg-pink{--bslib-color-bg: #e83e8c;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-pink{--bslib-color-fg: #e83e8c;color:var(--bslib-color-fg)}.bg-red{--bslib-color-bg: #ff0039;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-red{--bslib-color-fg: #ff0039;color:var(--bslib-color-fg)}.bg-orange{--bslib-color-bg: #f0ad4e;--bslib-color-fg: #000;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-orange{--bslib-color-fg: #f0ad4e;color:var(--bslib-color-fg)}.bg-yellow{--bslib-color-bg: #ff7518;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-yellow{--bslib-color-fg: #ff7518;color:var(--bslib-color-fg)}.bg-green{--bslib-color-bg: #3fb618;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-green{--bslib-color-fg: #3fb618;color:var(--bslib-color-fg)}.bg-teal{--bslib-color-bg: #20c997;--bslib-color-fg: #000;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-teal{--bslib-color-fg: #20c997;color:var(--bslib-color-fg)}.bg-cyan{--bslib-color-bg: #9954bb;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-cyan{--bslib-color-fg: #9954bb;color:var(--bslib-color-fg)}.text-default{--bslib-color-fg: #343a40}.bg-default{--bslib-color-bg: #343a40;--bslib-color-fg: #fff}.text-primary{--bslib-color-fg: #E8E8FC}.bg-primary{--bslib-color-bg: #E8E8FC;--bslib-color-fg: #000}.text-secondary{--bslib-color-fg: #343a40}.bg-secondary{--bslib-color-bg: #343a40;--bslib-color-fg: #fff}.text-success{--bslib-color-fg: #3fb618}.bg-success{--bslib-color-bg: #3fb618;--bslib-color-fg: #fff}.text-info{--bslib-color-fg: #9954bb}.bg-info{--bslib-color-bg: #9954bb;--bslib-color-fg: #fff}.text-warning{--bslib-color-fg: #ff7518}.bg-warning{--bslib-color-bg: #ff7518;--bslib-color-fg: #fff}.text-danger{--bslib-color-fg: #ff0039}.bg-danger{--bslib-color-bg: #ff0039;--bslib-color-fg: #fff}.text-light{--bslib-color-fg: #f8f9fa}.bg-light{--bslib-color-bg: #f8f9fa;--bslib-color-fg: #000}.text-dark{--bslib-color-fg: #343a40}.bg-dark{--bslib-color-bg: #343a40;--bslib-color-fg: #fff}.bg-gradient-blue-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #4053e9;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #4053e9;color:#fff}.bg-gradient-blue-purple{--bslib-color-fg: #fff;--bslib-color-bg: #3e65ba;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #3e65ba;color:#fff}.bg-gradient-blue-pink{--bslib-color-fg: #fff;--bslib-color-bg: #7466c0;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #7466c0;color:#fff}.bg-gradient-blue-red{--bslib-color-fg: #fff;--bslib-color-bg: #7d4d9f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #7d4d9f;color:#fff}.bg-gradient-blue-orange{--bslib-color-fg: #fff;--bslib-color-bg: #7792a7;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #7792a7;color:#fff}.bg-gradient-blue-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #7d7c92;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #7d7c92;color:#fff}.bg-gradient-blue-green{--bslib-color-fg: #fff;--bslib-color-bg: #319692;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #319692;color:#fff}.bg-gradient-blue-teal{--bslib-color-fg: #fff;--bslib-color-bg: #249dc5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #249dc5;color:#fff}.bg-gradient-blue-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #556ed3;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #556ed3;color:#fff}.bg-gradient-indigo-blue{--bslib-color-fg: #fff;--bslib-color-bg: #4d3dec;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #4d3dec;color:#fff}.bg-gradient-indigo-purple{--bslib-color-fg: #fff;--bslib-color-bg: #6422c3;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #6422c3;color:#fff}.bg-gradient-indigo-pink{--bslib-color-fg: #fff;--bslib-color-bg: #9a22c9;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #9a22c9;color:#fff}.bg-gradient-indigo-red{--bslib-color-fg: #fff;--bslib-color-bg: #a30aa8;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #a30aa8;color:#fff}.bg-gradient-indigo-orange{--bslib-color-fg: #fff;--bslib-color-bg: #9d4fb0;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #9d4fb0;color:#fff}.bg-gradient-indigo-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #a3389b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #a3389b;color:#fff}.bg-gradient-indigo-green{--bslib-color-fg: #fff;--bslib-color-bg: #56529b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #56529b;color:#fff}.bg-gradient-indigo-teal{--bslib-color-fg: #fff;--bslib-color-bg: #4a5ace;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #4a5ace;color:#fff}.bg-gradient-indigo-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #7a2bdc;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #7a2bdc;color:#fff}.bg-gradient-purple-blue{--bslib-color-fg: #fff;--bslib-color-bg: #4a58a5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #4a58a5;color:#fff}.bg-gradient-purple-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #632bab;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #632bab;color:#fff}.bg-gradient-purple-pink{--bslib-color-fg: #fff;--bslib-color-bg: #973d82;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #973d82;color:#fff}.bg-gradient-purple-red{--bslib-color-fg: #fff;--bslib-color-bg: #a02561;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #a02561;color:#fff}.bg-gradient-purple-orange{--bslib-color-fg: #fff;--bslib-color-bg: #9a6a6a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #9a6a6a;color:#fff}.bg-gradient-purple-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #a05354;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #a05354;color:#fff}.bg-gradient-purple-green{--bslib-color-fg: #fff;--bslib-color-bg: #536d54;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #536d54;color:#fff}.bg-gradient-purple-teal{--bslib-color-fg: #fff;--bslib-color-bg: #477587;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #477587;color:#fff}.bg-gradient-purple-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #774695;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #774695;color:#fff}.bg-gradient-pink-blue{--bslib-color-fg: #fff;--bslib-color-bg: #9b58af;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #9b58af;color:#fff}.bg-gradient-pink-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #b42cb5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #b42cb5;color:#fff}.bg-gradient-pink-purple{--bslib-color-fg: #fff;--bslib-color-bg: #b23e86;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #b23e86;color:#fff}.bg-gradient-pink-red{--bslib-color-fg: #fff;--bslib-color-bg: #f1256b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #f1256b;color:#fff}.bg-gradient-pink-orange{--bslib-color-fg: #fff;--bslib-color-bg: #eb6a73;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #eb6a73;color:#fff}.bg-gradient-pink-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #f1545e;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #f1545e;color:#fff}.bg-gradient-pink-green{--bslib-color-fg: #fff;--bslib-color-bg: #a46e5e;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #a46e5e;color:#fff}.bg-gradient-pink-teal{--bslib-color-fg: #fff;--bslib-color-bg: #987690;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #987690;color:#fff}.bg-gradient-pink-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #c8479f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #c8479f;color:#fff}.bg-gradient-red-blue{--bslib-color-fg: #fff;--bslib-color-bg: #a9337d;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #a9337d;color:#fff}.bg-gradient-red-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #c20683;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #c20683;color:#fff}.bg-gradient-red-purple{--bslib-color-fg: #fff;--bslib-color-bg: #c01854;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #c01854;color:#fff}.bg-gradient-red-pink{--bslib-color-fg: #fff;--bslib-color-bg: #f6195a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #f6195a;color:#fff}.bg-gradient-red-orange{--bslib-color-fg: #fff;--bslib-color-bg: #f94541;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #f94541;color:#fff}.bg-gradient-red-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #ff2f2c;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #ff2f2c;color:#fff}.bg-gradient-red-green{--bslib-color-fg: #fff;--bslib-color-bg: #b2492c;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #b2492c;color:#fff}.bg-gradient-red-teal{--bslib-color-fg: #fff;--bslib-color-bg: #a6505f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #a6505f;color:#fff}.bg-gradient-red-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #d6226d;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #d6226d;color:#fff}.bg-gradient-orange-blue{--bslib-color-fg: #fff;--bslib-color-bg: #a09b8a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #a09b8a;color:#fff}.bg-gradient-orange-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #b96e90;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #b96e90;color:#fff}.bg-gradient-orange-purple{--bslib-color-fg: #fff;--bslib-color-bg: #b78060;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #b78060;color:#fff}.bg-gradient-orange-pink{--bslib-color-fg: #fff;--bslib-color-bg: #ed8167;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #ed8167;color:#fff}.bg-gradient-orange-red{--bslib-color-fg: #fff;--bslib-color-bg: #f66846;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #f66846;color:#fff}.bg-gradient-orange-yellow{--bslib-color-fg: #000;--bslib-color-bg: #f69738;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #f69738;color:#000}.bg-gradient-orange-green{--bslib-color-fg: #000;--bslib-color-bg: #a9b138;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #a9b138;color:#000}.bg-gradient-orange-teal{--bslib-color-fg: #000;--bslib-color-bg: #9db86b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #9db86b;color:#000}.bg-gradient-orange-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #cd897a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #cd897a;color:#fff}.bg-gradient-yellow-blue{--bslib-color-fg: #fff;--bslib-color-bg: #a97969;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #a97969;color:#fff}.bg-gradient-yellow-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #c24d6f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #c24d6f;color:#fff}.bg-gradient-yellow-purple{--bslib-color-fg: #fff;--bslib-color-bg: #c05f40;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #c05f40;color:#fff}.bg-gradient-yellow-pink{--bslib-color-fg: #fff;--bslib-color-bg: #f65f46;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #f65f46;color:#fff}.bg-gradient-yellow-red{--bslib-color-fg: #fff;--bslib-color-bg: #ff4625;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #ff4625;color:#fff}.bg-gradient-yellow-orange{--bslib-color-fg: #000;--bslib-color-bg: #f98b2e;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #f98b2e;color:#000}.bg-gradient-yellow-green{--bslib-color-fg: #fff;--bslib-color-bg: #b28f18;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #b28f18;color:#fff}.bg-gradient-yellow-teal{--bslib-color-fg: #fff;--bslib-color-bg: #a6974b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #a6974b;color:#fff}.bg-gradient-yellow-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #d66859;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #d66859;color:#fff}.bg-gradient-green-blue{--bslib-color-fg: #fff;--bslib-color-bg: #35a069;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #35a069;color:#fff}.bg-gradient-green-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #4f746f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #4f746f;color:#fff}.bg-gradient-green-purple{--bslib-color-fg: #fff;--bslib-color-bg: #4d8640;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #4d8640;color:#fff}.bg-gradient-green-pink{--bslib-color-fg: #fff;--bslib-color-bg: #838646;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #838646;color:#fff}.bg-gradient-green-red{--bslib-color-fg: #fff;--bslib-color-bg: #8c6d25;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #8c6d25;color:#fff}.bg-gradient-green-orange{--bslib-color-fg: #000;--bslib-color-bg: #86b22e;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #86b22e;color:#000}.bg-gradient-green-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #8c9c18;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #8c9c18;color:#fff}.bg-gradient-green-teal{--bslib-color-fg: #000;--bslib-color-bg: #33be4b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #33be4b;color:#000}.bg-gradient-green-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #638f59;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #638f59;color:#fff}.bg-gradient-teal-blue{--bslib-color-fg: #fff;--bslib-color-bg: #23acb5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #23acb5;color:#fff}.bg-gradient-teal-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #3c7fbb;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #3c7fbb;color:#fff}.bg-gradient-teal-purple{--bslib-color-fg: #fff;--bslib-color-bg: #3a918c;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #3a918c;color:#fff}.bg-gradient-teal-pink{--bslib-color-fg: #fff;--bslib-color-bg: #709193;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #709193;color:#fff}.bg-gradient-teal-red{--bslib-color-fg: #fff;--bslib-color-bg: #797971;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #797971;color:#fff}.bg-gradient-teal-orange{--bslib-color-fg: #000;--bslib-color-bg: #73be7a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #73be7a;color:#000}.bg-gradient-teal-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #79a764;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #79a764;color:#fff}.bg-gradient-teal-green{--bslib-color-fg: #000;--bslib-color-bg: #2cc164;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #2cc164;color:#000}.bg-gradient-teal-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #509aa5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #509aa5;color:#fff}.bg-gradient-cyan-blue{--bslib-color-fg: #fff;--bslib-color-bg: #6b66cb;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #6b66cb;color:#fff}.bg-gradient-cyan-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #8539d1;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #8539d1;color:#fff}.bg-gradient-cyan-purple{--bslib-color-fg: #fff;--bslib-color-bg: #834ba2;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #834ba2;color:#fff}.bg-gradient-cyan-pink{--bslib-color-fg: #fff;--bslib-color-bg: #b94ba8;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #b94ba8;color:#fff}.bg-gradient-cyan-red{--bslib-color-fg: #fff;--bslib-color-bg: #c23287;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #c23287;color:#fff}.bg-gradient-cyan-orange{--bslib-color-fg: #fff;--bslib-color-bg: #bc788f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #bc788f;color:#fff}.bg-gradient-cyan-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #c2617a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #c2617a;color:#fff}.bg-gradient-cyan-green{--bslib-color-fg: #fff;--bslib-color-bg: #757b7a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #757b7a;color:#fff}.bg-gradient-cyan-teal{--bslib-color-fg: #fff;--bslib-color-bg: #6983ad;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #6983ad;color:#fff}.bg-blue{--bslib-color-bg: #2780e3;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-blue{--bslib-color-fg: #2780e3;color:var(--bslib-color-fg)}.bg-indigo{--bslib-color-bg: #6610f2;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-indigo{--bslib-color-fg: #6610f2;color:var(--bslib-color-fg)}.bg-purple{--bslib-color-bg: #613d7c;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-purple{--bslib-color-fg: #613d7c;color:var(--bslib-color-fg)}.bg-pink{--bslib-color-bg: #e83e8c;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-pink{--bslib-color-fg: #e83e8c;color:var(--bslib-color-fg)}.bg-red{--bslib-color-bg: #ff0039;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-red{--bslib-color-fg: #ff0039;color:var(--bslib-color-fg)}.bg-orange{--bslib-color-bg: #f0ad4e;--bslib-color-fg: #000;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-orange{--bslib-color-fg: #f0ad4e;color:var(--bslib-color-fg)}.bg-yellow{--bslib-color-bg: #ff7518;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-yellow{--bslib-color-fg: #ff7518;color:var(--bslib-color-fg)}.bg-green{--bslib-color-bg: #3fb618;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-green{--bslib-color-fg: #3fb618;color:var(--bslib-color-fg)}.bg-teal{--bslib-color-bg: #20c997;--bslib-color-fg: #000;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-teal{--bslib-color-fg: #20c997;color:var(--bslib-color-fg)}.bg-cyan{--bslib-color-bg: #9954bb;--bslib-color-fg: #fff;background-color:var(--bslib-color-bg);color:var(--bslib-color-fg)}.text-cyan{--bslib-color-fg: #9954bb;color:var(--bslib-color-fg)}.text-default{--bslib-color-fg: #343a40}.bg-default{--bslib-color-bg: #343a40;--bslib-color-fg: #fff}.text-primary{--bslib-color-fg: #E8E8FC}.bg-primary{--bslib-color-bg: #E8E8FC;--bslib-color-fg: #000}.text-secondary{--bslib-color-fg: #343a40}.bg-secondary{--bslib-color-bg: #343a40;--bslib-color-fg: #fff}.text-success{--bslib-color-fg: #3fb618}.bg-success{--bslib-color-bg: #3fb618;--bslib-color-fg: #fff}.text-info{--bslib-color-fg: #9954bb}.bg-info{--bslib-color-bg: #9954bb;--bslib-color-fg: #fff}.text-warning{--bslib-color-fg: #ff7518}.bg-warning{--bslib-color-bg: #ff7518;--bslib-color-fg: #fff}.text-danger{--bslib-color-fg: #ff0039}.bg-danger{--bslib-color-bg: #ff0039;--bslib-color-fg: #fff}.text-light{--bslib-color-fg: #f8f9fa}.bg-light{--bslib-color-bg: #f8f9fa;--bslib-color-fg: #000}.text-dark{--bslib-color-fg: #343a40}.bg-dark{--bslib-color-bg: #343a40;--bslib-color-fg: #fff}.bg-gradient-blue-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #4053e9;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #4053e9;color:#fff}.bg-gradient-blue-purple{--bslib-color-fg: #fff;--bslib-color-bg: #3e65ba;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #3e65ba;color:#fff}.bg-gradient-blue-pink{--bslib-color-fg: #fff;--bslib-color-bg: #7466c0;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #7466c0;color:#fff}.bg-gradient-blue-red{--bslib-color-fg: #fff;--bslib-color-bg: #7d4d9f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #7d4d9f;color:#fff}.bg-gradient-blue-orange{--bslib-color-fg: #fff;--bslib-color-bg: #7792a7;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #7792a7;color:#fff}.bg-gradient-blue-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #7d7c92;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #7d7c92;color:#fff}.bg-gradient-blue-green{--bslib-color-fg: #fff;--bslib-color-bg: #319692;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #319692;color:#fff}.bg-gradient-blue-teal{--bslib-color-fg: #fff;--bslib-color-bg: #249dc5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #249dc5;color:#fff}.bg-gradient-blue-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #556ed3;background:linear-gradient(var(--bg-gradient-deg, 140deg), #2780e3 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #556ed3;color:#fff}.bg-gradient-indigo-blue{--bslib-color-fg: #fff;--bslib-color-bg: #4d3dec;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #4d3dec;color:#fff}.bg-gradient-indigo-purple{--bslib-color-fg: #fff;--bslib-color-bg: #6422c3;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #6422c3;color:#fff}.bg-gradient-indigo-pink{--bslib-color-fg: #fff;--bslib-color-bg: #9a22c9;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #9a22c9;color:#fff}.bg-gradient-indigo-red{--bslib-color-fg: #fff;--bslib-color-bg: #a30aa8;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #a30aa8;color:#fff}.bg-gradient-indigo-orange{--bslib-color-fg: #fff;--bslib-color-bg: #9d4fb0;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #9d4fb0;color:#fff}.bg-gradient-indigo-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #a3389b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #a3389b;color:#fff}.bg-gradient-indigo-green{--bslib-color-fg: #fff;--bslib-color-bg: #56529b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #56529b;color:#fff}.bg-gradient-indigo-teal{--bslib-color-fg: #fff;--bslib-color-bg: #4a5ace;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #4a5ace;color:#fff}.bg-gradient-indigo-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #7a2bdc;background:linear-gradient(var(--bg-gradient-deg, 140deg), #6610f2 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #7a2bdc;color:#fff}.bg-gradient-purple-blue{--bslib-color-fg: #fff;--bslib-color-bg: #4a58a5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #4a58a5;color:#fff}.bg-gradient-purple-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #632bab;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #632bab;color:#fff}.bg-gradient-purple-pink{--bslib-color-fg: #fff;--bslib-color-bg: #973d82;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #973d82;color:#fff}.bg-gradient-purple-red{--bslib-color-fg: #fff;--bslib-color-bg: #a02561;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #a02561;color:#fff}.bg-gradient-purple-orange{--bslib-color-fg: #fff;--bslib-color-bg: #9a6a6a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #9a6a6a;color:#fff}.bg-gradient-purple-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #a05354;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #a05354;color:#fff}.bg-gradient-purple-green{--bslib-color-fg: #fff;--bslib-color-bg: #536d54;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #536d54;color:#fff}.bg-gradient-purple-teal{--bslib-color-fg: #fff;--bslib-color-bg: #477587;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #477587;color:#fff}.bg-gradient-purple-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #774695;background:linear-gradient(var(--bg-gradient-deg, 140deg), #613d7c var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #774695;color:#fff}.bg-gradient-pink-blue{--bslib-color-fg: #fff;--bslib-color-bg: #9b58af;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #9b58af;color:#fff}.bg-gradient-pink-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #b42cb5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #b42cb5;color:#fff}.bg-gradient-pink-purple{--bslib-color-fg: #fff;--bslib-color-bg: #b23e86;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #b23e86;color:#fff}.bg-gradient-pink-red{--bslib-color-fg: #fff;--bslib-color-bg: #f1256b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #f1256b;color:#fff}.bg-gradient-pink-orange{--bslib-color-fg: #fff;--bslib-color-bg: #eb6a73;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #eb6a73;color:#fff}.bg-gradient-pink-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #f1545e;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #f1545e;color:#fff}.bg-gradient-pink-green{--bslib-color-fg: #fff;--bslib-color-bg: #a46e5e;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #a46e5e;color:#fff}.bg-gradient-pink-teal{--bslib-color-fg: #fff;--bslib-color-bg: #987690;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #987690;color:#fff}.bg-gradient-pink-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #c8479f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #e83e8c var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #c8479f;color:#fff}.bg-gradient-red-blue{--bslib-color-fg: #fff;--bslib-color-bg: #a9337d;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #a9337d;color:#fff}.bg-gradient-red-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #c20683;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #c20683;color:#fff}.bg-gradient-red-purple{--bslib-color-fg: #fff;--bslib-color-bg: #c01854;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #c01854;color:#fff}.bg-gradient-red-pink{--bslib-color-fg: #fff;--bslib-color-bg: #f6195a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #f6195a;color:#fff}.bg-gradient-red-orange{--bslib-color-fg: #fff;--bslib-color-bg: #f94541;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #f94541;color:#fff}.bg-gradient-red-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #ff2f2c;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #ff2f2c;color:#fff}.bg-gradient-red-green{--bslib-color-fg: #fff;--bslib-color-bg: #b2492c;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #b2492c;color:#fff}.bg-gradient-red-teal{--bslib-color-fg: #fff;--bslib-color-bg: #a6505f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #a6505f;color:#fff}.bg-gradient-red-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #d6226d;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff0039 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #d6226d;color:#fff}.bg-gradient-orange-blue{--bslib-color-fg: #fff;--bslib-color-bg: #a09b8a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #a09b8a;color:#fff}.bg-gradient-orange-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #b96e90;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #b96e90;color:#fff}.bg-gradient-orange-purple{--bslib-color-fg: #fff;--bslib-color-bg: #b78060;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #b78060;color:#fff}.bg-gradient-orange-pink{--bslib-color-fg: #fff;--bslib-color-bg: #ed8167;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #ed8167;color:#fff}.bg-gradient-orange-red{--bslib-color-fg: #fff;--bslib-color-bg: #f66846;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #f66846;color:#fff}.bg-gradient-orange-yellow{--bslib-color-fg: #000;--bslib-color-bg: #f69738;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #f69738;color:#000}.bg-gradient-orange-green{--bslib-color-fg: #000;--bslib-color-bg: #a9b138;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #a9b138;color:#000}.bg-gradient-orange-teal{--bslib-color-fg: #000;--bslib-color-bg: #9db86b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #9db86b;color:#000}.bg-gradient-orange-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #cd897a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #f0ad4e var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #cd897a;color:#fff}.bg-gradient-yellow-blue{--bslib-color-fg: #fff;--bslib-color-bg: #a97969;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #a97969;color:#fff}.bg-gradient-yellow-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #c24d6f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #c24d6f;color:#fff}.bg-gradient-yellow-purple{--bslib-color-fg: #fff;--bslib-color-bg: #c05f40;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #c05f40;color:#fff}.bg-gradient-yellow-pink{--bslib-color-fg: #fff;--bslib-color-bg: #f65f46;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #f65f46;color:#fff}.bg-gradient-yellow-red{--bslib-color-fg: #fff;--bslib-color-bg: #ff4625;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #ff4625;color:#fff}.bg-gradient-yellow-orange{--bslib-color-fg: #000;--bslib-color-bg: #f98b2e;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #f98b2e;color:#000}.bg-gradient-yellow-green{--bslib-color-fg: #fff;--bslib-color-bg: #b28f18;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #b28f18;color:#fff}.bg-gradient-yellow-teal{--bslib-color-fg: #fff;--bslib-color-bg: #a6974b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #a6974b;color:#fff}.bg-gradient-yellow-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #d66859;background:linear-gradient(var(--bg-gradient-deg, 140deg), #ff7518 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #d66859;color:#fff}.bg-gradient-green-blue{--bslib-color-fg: #fff;--bslib-color-bg: #35a069;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #35a069;color:#fff}.bg-gradient-green-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #4f746f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #4f746f;color:#fff}.bg-gradient-green-purple{--bslib-color-fg: #fff;--bslib-color-bg: #4d8640;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #4d8640;color:#fff}.bg-gradient-green-pink{--bslib-color-fg: #fff;--bslib-color-bg: #838646;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #838646;color:#fff}.bg-gradient-green-red{--bslib-color-fg: #fff;--bslib-color-bg: #8c6d25;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #8c6d25;color:#fff}.bg-gradient-green-orange{--bslib-color-fg: #000;--bslib-color-bg: #86b22e;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #86b22e;color:#000}.bg-gradient-green-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #8c9c18;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #8c9c18;color:#fff}.bg-gradient-green-teal{--bslib-color-fg: #000;--bslib-color-bg: #33be4b;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #33be4b;color:#000}.bg-gradient-green-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #638f59;background:linear-gradient(var(--bg-gradient-deg, 140deg), #3fb618 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #638f59;color:#fff}.bg-gradient-teal-blue{--bslib-color-fg: #fff;--bslib-color-bg: #23acb5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #23acb5;color:#fff}.bg-gradient-teal-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #3c7fbb;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #3c7fbb;color:#fff}.bg-gradient-teal-purple{--bslib-color-fg: #fff;--bslib-color-bg: #3a918c;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #3a918c;color:#fff}.bg-gradient-teal-pink{--bslib-color-fg: #fff;--bslib-color-bg: #709193;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #709193;color:#fff}.bg-gradient-teal-red{--bslib-color-fg: #fff;--bslib-color-bg: #797971;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #797971;color:#fff}.bg-gradient-teal-orange{--bslib-color-fg: #000;--bslib-color-bg: #73be7a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #73be7a;color:#000}.bg-gradient-teal-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #79a764;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #79a764;color:#fff}.bg-gradient-teal-green{--bslib-color-fg: #000;--bslib-color-bg: #2cc164;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #2cc164;color:#000}.bg-gradient-teal-cyan{--bslib-color-fg: #fff;--bslib-color-bg: #509aa5;background:linear-gradient(var(--bg-gradient-deg, 140deg), #20c997 var(--bg-gradient-start, 36%), #9954bb var(--bg-gradient-end, 180%)) #509aa5;color:#fff}.bg-gradient-cyan-blue{--bslib-color-fg: #fff;--bslib-color-bg: #6b66cb;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #2780e3 var(--bg-gradient-end, 180%)) #6b66cb;color:#fff}.bg-gradient-cyan-indigo{--bslib-color-fg: #fff;--bslib-color-bg: #8539d1;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #6610f2 var(--bg-gradient-end, 180%)) #8539d1;color:#fff}.bg-gradient-cyan-purple{--bslib-color-fg: #fff;--bslib-color-bg: #834ba2;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #613d7c var(--bg-gradient-end, 180%)) #834ba2;color:#fff}.bg-gradient-cyan-pink{--bslib-color-fg: #fff;--bslib-color-bg: #b94ba8;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #e83e8c var(--bg-gradient-end, 180%)) #b94ba8;color:#fff}.bg-gradient-cyan-red{--bslib-color-fg: #fff;--bslib-color-bg: #c23287;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #ff0039 var(--bg-gradient-end, 180%)) #c23287;color:#fff}.bg-gradient-cyan-orange{--bslib-color-fg: #fff;--bslib-color-bg: #bc788f;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #f0ad4e var(--bg-gradient-end, 180%)) #bc788f;color:#fff}.bg-gradient-cyan-yellow{--bslib-color-fg: #fff;--bslib-color-bg: #c2617a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #ff7518 var(--bg-gradient-end, 180%)) #c2617a;color:#fff}.bg-gradient-cyan-green{--bslib-color-fg: #fff;--bslib-color-bg: #757b7a;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #3fb618 var(--bg-gradient-end, 180%)) #757b7a;color:#fff}.bg-gradient-cyan-teal{--bslib-color-fg: #fff;--bslib-color-bg: #6983ad;background:linear-gradient(var(--bg-gradient-deg, 140deg), #9954bb var(--bg-gradient-start, 36%), #20c997 var(--bg-gradient-end, 180%)) #6983ad;color:#fff}:root{--bslib-spacer: 1rem;--bslib-mb-spacer: var(--bslib-spacer, 1rem)}.bslib-mb-spacing{margin-bottom:var(--bslib-mb-spacer)}.bslib-gap-spacing{gap:var(--bslib-mb-spacer)}.bslib-gap-spacing>.bslib-mb-spacing,.bslib-gap-spacing>.form-group,.bslib-gap-spacing>p,.bslib-gap-spacing>pre{margin-bottom:0}.html-fill-container>.html-fill-item.bslib-mb-spacing{margin-bottom:0}.tab-content>.tab-pane.html-fill-container{display:none}.tab-content>.active.html-fill-container{display:flex}.tab-content.html-fill-container{padding:0}html{height:100%}.bslib-page-fill{width:100%;height:100%;margin:0;padding:var(--bslib-spacer, 1rem);gap:var(--bslib-spacer, 1rem)}@media(max-width: 575.98px){.bslib-page-fill{height:var(--bslib-page-fill-mobile-height, auto)}}.bslib-grid{display:grid !important;gap:var(--bslib-spacer, 1rem);height:var(--bslib-grid-height)}.bslib-grid.grid{grid-template-columns:repeat(var(--bs-columns, 12), minmax(0, 1fr));grid-template-rows:unset;grid-auto-rows:var(--bslib-grid--row-heights);--bslib-grid--row-heights--xs: unset;--bslib-grid--row-heights--sm: unset;--bslib-grid--row-heights--md: unset;--bslib-grid--row-heights--lg: unset;--bslib-grid--row-heights--xl: unset;--bslib-grid--row-heights--xxl: unset}.bslib-grid.grid.bslib-grid--row-heights--xs{--bslib-grid--row-heights: var(--bslib-grid--row-heights--xs)}@media(min-width: 576px){.bslib-grid.grid.bslib-grid--row-heights--sm{--bslib-grid--row-heights: var(--bslib-grid--row-heights--sm)}}@media(min-width: 768px){.bslib-grid.grid.bslib-grid--row-heights--md{--bslib-grid--row-heights: var(--bslib-grid--row-heights--md)}}@media(min-width: 992px){.bslib-grid.grid.bslib-grid--row-heights--lg{--bslib-grid--row-heights: var(--bslib-grid--row-heights--lg)}}@media(min-width: 1200px){.bslib-grid.grid.bslib-grid--row-heights--xl{--bslib-grid--row-heights: var(--bslib-grid--row-heights--xl)}}@media(min-width: 1400px){.bslib-grid.grid.bslib-grid--row-heights--xxl{--bslib-grid--row-heights: var(--bslib-grid--row-heights--xxl)}}.bslib-grid>*>.shiny-input-container{width:100%}.bslib-grid-item{grid-column:auto/span 1}@media(max-width: 767.98px){.bslib-grid-item{grid-column:1/-1}}@media(max-width: 575.98px){.bslib-grid{grid-template-columns:1fr !important;height:var(--bslib-grid-height-mobile)}.bslib-grid.grid{height:unset !important;grid-auto-rows:var(--bslib-grid--row-heights--xs, auto)}}.navbar+.container-fluid:has(>.tab-content>.tab-pane.active.html-fill-container),.navbar+.container-sm:has(>.tab-content>.tab-pane.active.html-fill-container),.navbar+.container-md:has(>.tab-content>.tab-pane.active.html-fill-container),.navbar+.container-lg:has(>.tab-content>.tab-pane.active.html-fill-container),.navbar+.container-xl:has(>.tab-content>.tab-pane.active.html-fill-container),.navbar+.container-xxl:has(>.tab-content>.tab-pane.active.html-fill-container){padding-left:0;padding-right:0}.navbar+.container-fluid>.tab-content>.tab-pane.active.html-fill-container,.navbar+.container-sm>.tab-content>.tab-pane.active.html-fill-container,.navbar+.container-md>.tab-content>.tab-pane.active.html-fill-container,.navbar+.container-lg>.tab-content>.tab-pane.active.html-fill-container,.navbar+.container-xl>.tab-content>.tab-pane.active.html-fill-container,.navbar+.container-xxl>.tab-content>.tab-pane.active.html-fill-container{padding:var(--bslib-spacer, 1rem);gap:var(--bslib-spacer, 1rem)}.navbar+.container-fluid>.tab-content>.tab-pane.active.html-fill-container:has(>.bslib-sidebar-layout:only-child),.navbar+.container-sm>.tab-content>.tab-pane.active.html-fill-container:has(>.bslib-sidebar-layout:only-child),.navbar+.container-md>.tab-content>.tab-pane.active.html-fill-container:has(>.bslib-sidebar-layout:only-child),.navbar+.container-lg>.tab-content>.tab-pane.active.html-fill-container:has(>.bslib-sidebar-layout:only-child),.navbar+.container-xl>.tab-content>.tab-pane.active.html-fill-container:has(>.bslib-sidebar-layout:only-child),.navbar+.container-xxl>.tab-content>.tab-pane.active.html-fill-container:has(>.bslib-sidebar-layout:only-child){padding:0}.navbar+.container-fluid>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border=true]),.navbar+.container-sm>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border=true]),.navbar+.container-md>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border=true]),.navbar+.container-lg>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border=true]),.navbar+.container-xl>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border=true]),.navbar+.container-xxl>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border=true]){border-left:none;border-right:none;border-bottom:none}.navbar+.container-fluid>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border-radius=true]),.navbar+.container-sm>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border-radius=true]),.navbar+.container-md>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border-radius=true]),.navbar+.container-lg>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border-radius=true]),.navbar+.container-xl>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border-radius=true]),.navbar+.container-xxl>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border-radius=true]){border-radius:0}.navbar+div>.bslib-sidebar-layout{border-top:var(--bslib-sidebar-border)}.bslib-card{overflow:auto}.bslib-card .card-body+.card-body{padding-top:0}.bslib-card .card-body{overflow:auto}.bslib-card .card-body p{margin-top:0}.bslib-card .card-body p:last-child{margin-bottom:0}.bslib-card .card-body{max-height:var(--bslib-card-body-max-height, none)}.bslib-card[data-full-screen=true]>.card-body{max-height:var(--bslib-card-body-max-height-full-screen, none)}.bslib-card .card-header .form-group{margin-bottom:0}.bslib-card .card-header .selectize-control{margin-bottom:0}.bslib-card .card-header .selectize-control .item{margin-right:1.15rem}.bslib-card .card-footer{margin-top:auto}.bslib-card .bslib-navs-card-title{display:flex;flex-wrap:wrap;justify-content:space-between;align-items:center}.bslib-card .bslib-navs-card-title .nav{margin-left:auto}.bslib-card .bslib-sidebar-layout:not([data-bslib-sidebar-border=true]){border:none}.bslib-card .bslib-sidebar-layout:not([data-bslib-sidebar-border-radius=true]){border-top-left-radius:0;border-top-right-radius:0}[data-full-screen=true]{position:fixed;inset:3.5rem 1rem 1rem;height:auto !important;max-height:none !important;width:auto !important;z-index:1070}.bslib-full-screen-enter{display:none;position:absolute;bottom:var(--bslib-full-screen-enter-bottom, 0.2rem);right:var(--bslib-full-screen-enter-right, 0);top:var(--bslib-full-screen-enter-top);left:var(--bslib-full-screen-enter-left);color:var(--bslib-color-fg, var(--bs-card-color));background-color:var(--bslib-color-bg, var(--bs-card-bg, var(--bs-body-bg)));border:var(--bs-card-border-width) solid var(--bslib-color-fg, var(--bs-card-border-color));box-shadow:0 2px 4px rgba(0,0,0,.15);margin:.2rem .4rem;padding:.55rem !important;font-size:.8rem;cursor:pointer;opacity:.7;z-index:1070}.bslib-full-screen-enter:hover{opacity:1}.card[data-full-screen=false]:hover>*>.bslib-full-screen-enter{display:block}.bslib-has-full-screen .card:hover>*>.bslib-full-screen-enter{display:none}@media(max-width: 575.98px){.bslib-full-screen-enter{display:none !important}}.bslib-full-screen-exit{position:relative;top:1.35rem;font-size:.9rem;cursor:pointer;text-decoration:none;display:flex;float:right;margin-right:2.15rem;align-items:center;color:rgba(var(--bs-body-bg-rgb), 0.8)}.bslib-full-screen-exit:hover{color:rgba(var(--bs-body-bg-rgb), 1)}.bslib-full-screen-exit svg{margin-left:.5rem;font-size:1.5rem}#bslib-full-screen-overlay{position:fixed;inset:0;background-color:rgba(var(--bs-body-color-rgb), 0.6);backdrop-filter:blur(2px);-webkit-backdrop-filter:blur(2px);z-index:1069;animation:bslib-full-screen-overlay-enter 400ms cubic-bezier(0.6, 0.02, 0.65, 1) forwards}@keyframes bslib-full-screen-overlay-enter{0%{opacity:0}100%{opacity:1}}.accordion .accordion-header{font-size:calc(1.29rem + 0.48vw);margin-top:0;margin-bottom:.5rem;font-weight:400;line-height:1.2;color:var(--bs-heading-color);margin-bottom:0}@media(min-width: 1200px){.accordion .accordion-header{font-size:1.65rem}}.accordion .accordion-icon:not(:empty){margin-right:.75rem;display:flex}.accordion .accordion-button:not(.collapsed){box-shadow:none}.accordion .accordion-button:not(.collapsed):focus{box-shadow:var(--bs-accordion-btn-focus-box-shadow)}.bslib-sidebar-layout{--bslib-sidebar-transition-duration: 500ms;--bslib-sidebar-transition-easing-x: cubic-bezier(0.8, 0.78, 0.22, 1.07);--bslib-sidebar-border: var(--bs-card-border-width, 1px) solid var(--bs-card-border-color, rgba(0, 0, 0, 0.175));--bslib-sidebar-border-radius: var(--bs-border-radius);--bslib-sidebar-vert-border: var(--bs-card-border-width, 1px) solid var(--bs-card-border-color, rgba(0, 0, 0, 0.175));--bslib-sidebar-bg: rgba(var(--bs-emphasis-color-rgb, 0, 0, 0), 0.05);--bslib-sidebar-fg: var(--bs-emphasis-color, black);--bslib-sidebar-main-fg: var(--bs-card-color, var(--bs-body-color));--bslib-sidebar-main-bg: var(--bs-card-bg, var(--bs-body-bg));--bslib-sidebar-toggle-bg: rgba(var(--bs-emphasis-color-rgb, 0, 0, 0), 0.1);--bslib-sidebar-padding: calc(var(--bslib-spacer) * 1.5);--bslib-sidebar-icon-size: var(--bslib-spacer, 1rem);--bslib-sidebar-icon-button-size: calc(var(--bslib-sidebar-icon-size, 1rem) * 2);--bslib-sidebar-padding-icon: calc(var(--bslib-sidebar-icon-button-size, 2rem) * 1.5);--bslib-collapse-toggle-border-radius: var(--bs-border-radius, 0.25rem);--bslib-collapse-toggle-transform: 0deg;--bslib-sidebar-toggle-transition-easing: cubic-bezier(1, 0, 0, 1);--bslib-collapse-toggle-right-transform: 180deg;--bslib-sidebar-column-main: minmax(0, 1fr);display:grid !important;grid-template-columns:min(100% - var(--bslib-sidebar-icon-size),var(--bslib-sidebar-width, 250px)) var(--bslib-sidebar-column-main);position:relative;transition:grid-template-columns ease-in-out var(--bslib-sidebar-transition-duration);border:var(--bslib-sidebar-border);border-radius:var(--bslib-sidebar-border-radius)}@media(prefers-reduced-motion: reduce){.bslib-sidebar-layout{transition:none}}.bslib-sidebar-layout[data-bslib-sidebar-border=false]{border:none}.bslib-sidebar-layout[data-bslib-sidebar-border-radius=false]{border-radius:initial}.bslib-sidebar-layout>.main,.bslib-sidebar-layout>.sidebar{grid-row:1/2;border-radius:inherit;overflow:auto}.bslib-sidebar-layout>.main{grid-column:2/3;border-top-left-radius:0;border-bottom-left-radius:0;padding:var(--bslib-sidebar-padding);transition:padding var(--bslib-sidebar-transition-easing-x) var(--bslib-sidebar-transition-duration);color:var(--bslib-sidebar-main-fg);background-color:var(--bslib-sidebar-main-bg)}.bslib-sidebar-layout>.sidebar{grid-column:1/2;width:100%;height:100%;border-right:var(--bslib-sidebar-vert-border);border-top-right-radius:0;border-bottom-right-radius:0;color:var(--bslib-sidebar-fg);background-color:var(--bslib-sidebar-bg);backdrop-filter:blur(5px)}.bslib-sidebar-layout>.sidebar>.sidebar-content{display:flex;flex-direction:column;gap:var(--bslib-spacer, 1rem);padding:var(--bslib-sidebar-padding);padding-top:var(--bslib-sidebar-padding-icon)}.bslib-sidebar-layout>.sidebar>.sidebar-content>:last-child:not(.sidebar-title){margin-bottom:0}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion{margin-left:calc(-1*var(--bslib-sidebar-padding));margin-right:calc(-1*var(--bslib-sidebar-padding))}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:last-child{margin-bottom:calc(-1*var(--bslib-sidebar-padding))}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:not(:last-child){margin-bottom:1rem}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion .accordion-body{display:flex;flex-direction:column}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:not(:first-child) .accordion-item:first-child{border-top:var(--bs-accordion-border-width) solid var(--bs-accordion-border-color)}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:not(:last-child) .accordion-item:last-child{border-bottom:var(--bs-accordion-border-width) solid var(--bs-accordion-border-color)}.bslib-sidebar-layout>.sidebar>.sidebar-content.has-accordion>.sidebar-title{border-bottom:none;padding-bottom:0}.bslib-sidebar-layout>.sidebar .shiny-input-container{width:100%}.bslib-sidebar-layout[data-bslib-sidebar-open=always]>.sidebar>.sidebar-content{padding-top:var(--bslib-sidebar-padding)}.bslib-sidebar-layout>.collapse-toggle{grid-row:1/2;grid-column:1/2;display:inline-flex;align-items:center;position:absolute;right:calc(var(--bslib-sidebar-icon-size));top:calc(var(--bslib-sidebar-icon-size, 1rem)/2);border:none;border-radius:var(--bslib-collapse-toggle-border-radius);height:var(--bslib-sidebar-icon-button-size, 2rem);width:var(--bslib-sidebar-icon-button-size, 2rem);display:flex;align-items:center;justify-content:center;padding:0;color:var(--bslib-sidebar-fg);background-color:unset;transition:color var(--bslib-sidebar-transition-easing-x) var(--bslib-sidebar-transition-duration),top var(--bslib-sidebar-transition-easing-x) var(--bslib-sidebar-transition-duration),right var(--bslib-sidebar-transition-easing-x) var(--bslib-sidebar-transition-duration),left var(--bslib-sidebar-transition-easing-x) var(--bslib-sidebar-transition-duration)}.bslib-sidebar-layout>.collapse-toggle:hover{background-color:var(--bslib-sidebar-toggle-bg)}.bslib-sidebar-layout>.collapse-toggle>.collapse-icon{opacity:.8;width:var(--bslib-sidebar-icon-size);height:var(--bslib-sidebar-icon-size);transform:rotateY(var(--bslib-collapse-toggle-transform));transition:transform var(--bslib-sidebar-toggle-transition-easing) var(--bslib-sidebar-transition-duration)}.bslib-sidebar-layout>.collapse-toggle:hover>.collapse-icon{opacity:1}.bslib-sidebar-layout .sidebar-title{font-size:1.25rem;line-height:1.25;margin-top:0;margin-bottom:1rem;padding-bottom:1rem;border-bottom:var(--bslib-sidebar-border)}.bslib-sidebar-layout.sidebar-right{grid-template-columns:var(--bslib-sidebar-column-main) min(100% - var(--bslib-sidebar-icon-size),var(--bslib-sidebar-width, 250px))}.bslib-sidebar-layout.sidebar-right>.main{grid-column:1/2;border-top-right-radius:0;border-bottom-right-radius:0;border-top-left-radius:inherit;border-bottom-left-radius:inherit}.bslib-sidebar-layout.sidebar-right>.sidebar{grid-column:2/3;border-right:none;border-left:var(--bslib-sidebar-vert-border);border-top-left-radius:0;border-bottom-left-radius:0}.bslib-sidebar-layout.sidebar-right>.collapse-toggle{grid-column:2/3;left:var(--bslib-sidebar-icon-size);right:unset;border:var(--bslib-collapse-toggle-border)}.bslib-sidebar-layout.sidebar-right>.collapse-toggle>.collapse-icon{transform:rotateY(var(--bslib-collapse-toggle-right-transform))}.bslib-sidebar-layout.sidebar-collapsed{--bslib-collapse-toggle-transform: 180deg;--bslib-collapse-toggle-right-transform: 0deg;--bslib-sidebar-vert-border: none;grid-template-columns:0 minmax(0, 1fr)}.bslib-sidebar-layout.sidebar-collapsed.sidebar-right{grid-template-columns:minmax(0, 1fr) 0}.bslib-sidebar-layout.sidebar-collapsed:not(.transitioning)>.sidebar>*{display:none}.bslib-sidebar-layout.sidebar-collapsed>.main{border-radius:inherit}.bslib-sidebar-layout.sidebar-collapsed:not(.sidebar-right)>.main{padding-left:var(--bslib-sidebar-padding-icon)}.bslib-sidebar-layout.sidebar-collapsed.sidebar-right>.main{padding-right:var(--bslib-sidebar-padding-icon)}.bslib-sidebar-layout.sidebar-collapsed>.collapse-toggle{color:var(--bslib-sidebar-main-fg);top:calc(var(--bslib-sidebar-overlap-counter, 0)*(var(--bslib-sidebar-icon-size) + var(--bslib-sidebar-padding)) + var(--bslib-sidebar-icon-size, 1rem)/2);right:calc(-2.5*var(--bslib-sidebar-icon-size) - var(--bs-card-border-width, 1px))}.bslib-sidebar-layout.sidebar-collapsed.sidebar-right>.collapse-toggle{left:calc(-2.5*var(--bslib-sidebar-icon-size) - var(--bs-card-border-width, 1px));right:unset}@media(min-width: 576px){.bslib-sidebar-layout.transitioning>.sidebar>.sidebar-content{display:none}}@media(max-width: 575.98px){.bslib-sidebar-layout[data-bslib-sidebar-open=desktop]{--bslib-sidebar-js-init-collapsed: true}.bslib-sidebar-layout>.sidebar,.bslib-sidebar-layout.sidebar-right>.sidebar{border:none}.bslib-sidebar-layout>.main,.bslib-sidebar-layout.sidebar-right>.main{grid-column:1/3}.bslib-sidebar-layout[data-bslib-sidebar-open=always]{display:block !important}.bslib-sidebar-layout[data-bslib-sidebar-open=always]>.sidebar{max-height:var(--bslib-sidebar-max-height-mobile);overflow-y:auto;border-top:var(--bslib-sidebar-vert-border)}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]){grid-template-columns:100% 0}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]):not(.sidebar-collapsed)>.sidebar{z-index:1}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]):not(.sidebar-collapsed)>.collapse-toggle{z-index:1}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]).sidebar-right{grid-template-columns:0 100%}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]).sidebar-collapsed{grid-template-columns:0 100%}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]).sidebar-collapsed.sidebar-right{grid-template-columns:100% 0}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]):not(.sidebar-right)>.main{padding-left:var(--bslib-sidebar-padding-icon)}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]).sidebar-right>.main{padding-right:var(--bslib-sidebar-padding-icon)}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always])>.main{opacity:0;transition:opacity var(--bslib-sidebar-transition-easing-x) var(--bslib-sidebar-transition-duration)}.bslib-sidebar-layout:not([data-bslib-sidebar-open=always]).sidebar-collapsed>.main{opacity:1}}:root{--bslib-value-box-shadow: none;--bslib-value-box-border-width-auto-yes: var(--bslib-value-box-border-width-baseline);--bslib-value-box-border-width-auto-no: 0;--bslib-value-box-border-width-baseline: 1px}.bslib-value-box{border-width:var(--bslib-value-box-border-width-auto-no, var(--bslib-value-box-border-width-baseline));container-name:bslib-value-box;container-type:inline-size}.bslib-value-box.card{box-shadow:var(--bslib-value-box-shadow)}.bslib-value-box.border-auto{border-width:var(--bslib-value-box-border-width-auto-yes, var(--bslib-value-box-border-width-baseline))}.bslib-value-box.default{--bslib-value-box-bg-default: var(--bs-card-bg, #FFFFFF);--bslib-value-box-border-color-default: var(--bs-card-border-color, rgba(0, 0, 0, 0.175));color:var(--bslib-value-box-color);background-color:var(--bslib-value-box-bg, var(--bslib-value-box-bg-default));border-color:var(--bslib-value-box-border-color, var(--bslib-value-box-border-color-default))}.bslib-value-box .value-box-grid{display:grid;grid-template-areas:"left right";align-items:center;overflow:hidden}.bslib-value-box .value-box-showcase{height:100%;max-height:var(---bslib-value-box-showcase-max-h, 100%)}.bslib-value-box .value-box-showcase,.bslib-value-box .value-box-showcase>.html-fill-item{width:100%}.bslib-value-box[data-full-screen=true] .value-box-showcase{max-height:var(---bslib-value-box-showcase-max-h-fs, 100%)}@media screen and (min-width: 575.98px){@container bslib-value-box (max-width: 300px){.bslib-value-box:not(.showcase-bottom) .value-box-grid{grid-template-columns:1fr !important;grid-template-rows:auto auto;grid-template-areas:"top" "bottom"}.bslib-value-box:not(.showcase-bottom) .value-box-grid .value-box-showcase{grid-area:top !important}.bslib-value-box:not(.showcase-bottom) .value-box-grid .value-box-area{grid-area:bottom !important;justify-content:end}}}.bslib-value-box .value-box-area{justify-content:center;padding:1.5rem 1rem;font-size:.9rem;font-weight:500}.bslib-value-box .value-box-area *{margin-bottom:0;margin-top:0}.bslib-value-box .value-box-title{font-size:1rem;margin-top:0;margin-bottom:.5rem;font-weight:400;line-height:1.2}.bslib-value-box .value-box-title:empty::after{content:" "}.bslib-value-box .value-box-value{font-size:calc(1.29rem + 0.48vw);margin-top:0;margin-bottom:.5rem;font-weight:400;line-height:1.2}@media(min-width: 1200px){.bslib-value-box .value-box-value{font-size:1.65rem}}.bslib-value-box .value-box-value:empty::after{content:" "}.bslib-value-box .value-box-showcase{align-items:center;justify-content:center;margin-top:auto;margin-bottom:auto;padding:1rem}.bslib-value-box .value-box-showcase .bi,.bslib-value-box .value-box-showcase .fa,.bslib-value-box .value-box-showcase .fab,.bslib-value-box .value-box-showcase .fas,.bslib-value-box .value-box-showcase .far{opacity:.85;min-width:50px;max-width:125%}.bslib-value-box .value-box-showcase .bi,.bslib-value-box .value-box-showcase .fa,.bslib-value-box .value-box-showcase .fab,.bslib-value-box .value-box-showcase .fas,.bslib-value-box .value-box-showcase .far{font-size:4rem}.bslib-value-box.showcase-top-right .value-box-grid{grid-template-columns:1fr var(---bslib-value-box-showcase-w, 50%)}.bslib-value-box.showcase-top-right .value-box-grid .value-box-showcase{grid-area:right;margin-left:auto;align-self:start;align-items:end;padding-left:0;padding-bottom:0}.bslib-value-box.showcase-top-right .value-box-grid .value-box-area{grid-area:left;align-self:end}.bslib-value-box.showcase-top-right[data-full-screen=true] .value-box-grid{grid-template-columns:auto var(---bslib-value-box-showcase-w-fs, 1fr)}.bslib-value-box.showcase-top-right[data-full-screen=true] .value-box-grid>div{align-self:center}.bslib-value-box.showcase-top-right:not([data-full-screen=true]) .value-box-showcase{margin-top:0}@container bslib-value-box (max-width: 300px){.bslib-value-box.showcase-top-right:not([data-full-screen=true]) .value-box-grid .value-box-showcase{padding-left:1rem}}.bslib-value-box.showcase-left-center .value-box-grid{grid-template-columns:var(---bslib-value-box-showcase-w, 30%) auto}.bslib-value-box.showcase-left-center[data-full-screen=true] .value-box-grid{grid-template-columns:var(---bslib-value-box-showcase-w-fs, 1fr) auto}.bslib-value-box.showcase-left-center:not([data-fill-screen=true]) .value-box-grid .value-box-showcase{grid-area:left}.bslib-value-box.showcase-left-center:not([data-fill-screen=true]) .value-box-grid .value-box-area{grid-area:right}.bslib-value-box.showcase-bottom .value-box-grid{grid-template-columns:1fr;grid-template-rows:1fr var(---bslib-value-box-showcase-h, auto);grid-template-areas:"top" "bottom";overflow:hidden}.bslib-value-box.showcase-bottom .value-box-grid .value-box-showcase{grid-area:bottom;padding:0;margin:0}.bslib-value-box.showcase-bottom .value-box-grid .value-box-area{grid-area:top}.bslib-value-box.showcase-bottom[data-full-screen=true] .value-box-grid{grid-template-rows:1fr var(---bslib-value-box-showcase-h-fs, 2fr)}.bslib-value-box.showcase-bottom[data-full-screen=true] .value-box-grid .value-box-showcase{padding:1rem}[data-bs-theme=dark] .bslib-value-box{--bslib-value-box-shadow: 0 0.5rem 1rem rgb(0 0 0 / 50%)}:root{--bslib-page-sidebar-title-bg: #3CDD8C;--bslib-page-sidebar-title-color: #000}.bslib-page-title{background-color:var(--bslib-page-sidebar-title-bg);color:var(--bslib-page-sidebar-title-color);font-size:1.25rem;font-weight:300;padding:var(--bslib-spacer, 1rem);padding-left:1.5rem;margin-bottom:0;border-bottom:1px solid #dee2e6}@media(min-width: 576px){.nav:not(.nav-hidden){display:flex !important;display:-webkit-flex !important}.nav:not(.nav-hidden):not(.nav-stacked):not(.flex-column){float:none !important}.nav:not(.nav-hidden):not(.nav-stacked):not(.flex-column)>.bslib-nav-spacer{margin-left:auto !important}.nav:not(.nav-hidden):not(.nav-stacked):not(.flex-column)>.form-inline{margin-top:auto;margin-bottom:auto}.nav:not(.nav-hidden).nav-stacked{flex-direction:column;-webkit-flex-direction:column;height:100%}.nav:not(.nav-hidden).nav-stacked>.bslib-nav-spacer{margin-top:auto !important}}.html-fill-container{display:flex;flex-direction:column;min-height:0;min-width:0}.html-fill-container>.html-fill-item{flex:1 1 auto;min-height:0;min-width:0}.html-fill-container>:not(.html-fill-item){flex:0 0 auto}.quarto-container{min-height:calc(100vh - 132px)}body.hypothesis-enabled #quarto-header{margin-right:16px}footer.footer .nav-footer,#quarto-header>nav{padding-left:1em;padding-right:1em}footer.footer div.nav-footer p:first-child{margin-top:0}footer.footer div.nav-footer p:last-child{margin-bottom:0}#quarto-content>*{padding-top:14px}#quarto-content>#quarto-sidebar-glass{padding-top:0px}@media(max-width: 991.98px){#quarto-content>*{padding-top:0}#quarto-content .subtitle{padding-top:14px}#quarto-content section:first-of-type h2:first-of-type,#quarto-content section:first-of-type .h2:first-of-type{margin-top:1rem}}.headroom-target,header.headroom{will-change:transform;transition:position 200ms linear;transition:all 200ms linear}header.headroom--pinned{transform:translateY(0%)}header.headroom--unpinned{transform:translateY(-100%)}.navbar-container{width:100%}.navbar-brand{overflow:hidden;text-overflow:ellipsis}.navbar-brand-container{max-width:calc(100% - 115px);min-width:0;display:flex;align-items:center}@media(min-width: 992px){.navbar-brand-container{margin-right:1em}}.navbar-brand.navbar-brand-logo{margin-right:4px;display:inline-flex}.navbar-toggler{flex-basis:content;flex-shrink:0}.navbar .navbar-brand-container{order:2}.navbar .navbar-toggler{order:1}.navbar .navbar-container>.navbar-nav{order:20}.navbar .navbar-container>.navbar-brand-container{margin-left:0 !important;margin-right:0 !important}.navbar .navbar-collapse{order:20}.navbar #quarto-search{order:4;margin-left:auto}.navbar .navbar-toggler{margin-right:.5em}.navbar-collapse .quarto-navbar-tools{margin-left:.5em}.navbar-logo{max-height:24px;width:auto;padding-right:4px}nav .nav-item:not(.compact){padding-top:1px}nav .nav-link i,nav .dropdown-item i{padding-right:1px}.navbar-expand-lg .navbar-nav .nav-link{padding-left:.6rem;padding-right:.6rem}nav .nav-item.compact .nav-link{padding-left:.5rem;padding-right:.5rem;font-size:1.1rem}.navbar .quarto-navbar-tools{order:3}.navbar .quarto-navbar-tools div.dropdown{display:inline-block}.navbar .quarto-navbar-tools .quarto-navigation-tool{color:#fff}.navbar .quarto-navbar-tools .quarto-navigation-tool:hover{color:#000}.navbar-nav .dropdown-menu{min-width:220px;font-size:.9rem}.navbar .navbar-nav .nav-link.dropdown-toggle::after{opacity:.75;vertical-align:.175em}.navbar ul.dropdown-menu{padding-top:0;padding-bottom:0}.navbar .dropdown-header{text-transform:uppercase;font-size:.8rem;padding:0 .5rem}.navbar .dropdown-item{padding:.4rem .5rem}.navbar .dropdown-item>i.bi{margin-left:.1rem;margin-right:.25em}.sidebar #quarto-search{margin-top:-1px}.sidebar #quarto-search svg.aa-SubmitIcon{width:16px;height:16px}.sidebar-navigation a{color:inherit}.sidebar-title{margin-top:.25rem;padding-bottom:.5rem;font-size:1.3rem;line-height:1.6rem;visibility:visible}.sidebar-title>a{font-size:inherit;text-decoration:none}.sidebar-title .sidebar-tools-main{margin-top:-6px}@media(max-width: 991.98px){#quarto-sidebar div.sidebar-header{padding-top:.2em}}.sidebar-header-stacked .sidebar-title{margin-top:.6rem}.sidebar-logo{max-width:90%;padding-bottom:.5rem}.sidebar-logo-link{text-decoration:none}.sidebar-navigation li a{text-decoration:none}.sidebar-navigation .quarto-navigation-tool{opacity:.7;font-size:.875rem}#quarto-sidebar>nav>.sidebar-tools-main{margin-left:14px}.sidebar-tools-main{display:inline-flex;margin-left:0px;order:2}.sidebar-tools-main:not(.tools-wide){vertical-align:middle}.sidebar-navigation .quarto-navigation-tool.dropdown-toggle::after{display:none}.sidebar.sidebar-navigation>*{padding-top:1em}.sidebar-item{margin-bottom:.2em;line-height:1rem;margin-top:.4rem}.sidebar-section{padding-left:.5em;padding-bottom:.2em}.sidebar-item .sidebar-item-container{display:flex;justify-content:space-between;cursor:pointer}.sidebar-item-toggle:hover{cursor:pointer}.sidebar-item .sidebar-item-toggle .bi{font-size:.7rem;text-align:center}.sidebar-item .sidebar-item-toggle .bi-chevron-right::before{transition:transform 200ms ease}.sidebar-item .sidebar-item-toggle[aria-expanded=false] .bi-chevron-right::before{transform:none}.sidebar-item .sidebar-item-toggle[aria-expanded=true] .bi-chevron-right::before{transform:rotate(90deg)}.sidebar-item-text{width:100%}.sidebar-navigation .sidebar-divider{margin-left:0;margin-right:0;margin-top:.5rem;margin-bottom:.5rem}@media(max-width: 991.98px){.quarto-secondary-nav{display:block}.quarto-secondary-nav button.quarto-search-button{padding-right:0em;padding-left:2em}.quarto-secondary-nav button.quarto-btn-toggle{margin-left:-0.75rem;margin-right:.15rem}.quarto-secondary-nav nav.quarto-title-breadcrumbs{display:none}.quarto-secondary-nav nav.quarto-page-breadcrumbs{display:flex;align-items:center;padding-right:1em;margin-left:-0.25em}.quarto-secondary-nav nav.quarto-page-breadcrumbs a{text-decoration:none}.quarto-secondary-nav nav.quarto-page-breadcrumbs ol.breadcrumb{margin-bottom:0}}@media(min-width: 992px){.quarto-secondary-nav{display:none}}.quarto-title-breadcrumbs .breadcrumb{margin-bottom:.5em;font-size:.9rem}.quarto-title-breadcrumbs .breadcrumb li:last-of-type a{color:#6c757d}.quarto-secondary-nav .quarto-btn-toggle{color:#595959}.quarto-secondary-nav[aria-expanded=false] .quarto-btn-toggle .bi-chevron-right::before{transform:none}.quarto-secondary-nav[aria-expanded=true] .quarto-btn-toggle .bi-chevron-right::before{transform:rotate(90deg)}.quarto-secondary-nav .quarto-btn-toggle .bi-chevron-right::before{transition:transform 200ms ease}.quarto-secondary-nav{cursor:pointer}.no-decor{text-decoration:none}.quarto-secondary-nav-title{margin-top:.3em;color:#595959;padding-top:4px}.quarto-secondary-nav nav.quarto-page-breadcrumbs{color:#595959}.quarto-secondary-nav nav.quarto-page-breadcrumbs a{color:#595959}.quarto-secondary-nav nav.quarto-page-breadcrumbs a:hover{color:rgba(64,64,191,.8)}.quarto-secondary-nav nav.quarto-page-breadcrumbs .breadcrumb-item::before{color:#8c8c8c}.breadcrumb-item{line-height:1.2rem}div.sidebar-item-container{color:#595959}div.sidebar-item-container:hover,div.sidebar-item-container:focus{color:rgba(64,64,191,.8)}div.sidebar-item-container.disabled{color:rgba(89,89,89,.75)}div.sidebar-item-container .active,div.sidebar-item-container .show>.nav-link,div.sidebar-item-container .sidebar-link>code{color:#4040bf}div.sidebar.sidebar-navigation.rollup.quarto-sidebar-toggle-contents,nav.sidebar.sidebar-navigation:not(.rollup){background-color:#fff}@media(max-width: 991.98px){.sidebar-navigation .sidebar-item a,.nav-page .nav-page-text,.sidebar-navigation{font-size:1rem}.sidebar-navigation ul.sidebar-section.depth1 .sidebar-section-item{font-size:1.1rem}.sidebar-logo{display:none}.sidebar.sidebar-navigation{position:static;border-bottom:1px solid #dee2e6}.sidebar.sidebar-navigation.collapsing{position:fixed;z-index:1000}.sidebar.sidebar-navigation.show{position:fixed;z-index:1000}.sidebar.sidebar-navigation{min-height:100%}nav.quarto-secondary-nav{background-color:#fff;border-bottom:1px solid #dee2e6}.quarto-banner nav.quarto-secondary-nav{background-color:#3cdd8c;color:#fff;border-top:1px solid #dee2e6}.sidebar .sidebar-footer{visibility:visible;padding-top:1rem;position:inherit}.sidebar-tools-collapse{display:block}}#quarto-sidebar{transition:width .15s ease-in}#quarto-sidebar>*{padding-right:1em}@media(max-width: 991.98px){#quarto-sidebar .sidebar-menu-container{white-space:nowrap;min-width:225px}#quarto-sidebar.show{transition:width .15s ease-out}}@media(min-width: 992px){#quarto-sidebar{display:flex;flex-direction:column}.nav-page .nav-page-text,.sidebar-navigation .sidebar-section .sidebar-item{font-size:.875rem}.sidebar-navigation .sidebar-item{font-size:.925rem}.sidebar.sidebar-navigation{display:block;position:sticky}.sidebar-search{width:100%}.sidebar .sidebar-footer{visibility:visible}}@media(min-width: 992px){#quarto-sidebar-glass{display:none}}@media(max-width: 991.98px){#quarto-sidebar-glass{position:fixed;top:0;bottom:0;left:0;right:0;background-color:rgba(255,255,255,0);transition:background-color .15s ease-in;z-index:-1}#quarto-sidebar-glass.collapsing{z-index:1000}#quarto-sidebar-glass.show{transition:background-color .15s ease-out;background-color:rgba(102,102,102,.4);z-index:1000}}.sidebar .sidebar-footer{padding:.5rem 1rem;align-self:flex-end;color:#6c757d;width:100%}.quarto-page-breadcrumbs .breadcrumb-item+.breadcrumb-item,.quarto-page-breadcrumbs .breadcrumb-item{padding-right:.33em;padding-left:0}.quarto-page-breadcrumbs .breadcrumb-item::before{padding-right:.33em}.quarto-sidebar-footer{font-size:.875em}.sidebar-section .bi-chevron-right{vertical-align:middle}.sidebar-section .bi-chevron-right::before{font-size:.9em}.notransition{-webkit-transition:none !important;-moz-transition:none !important;-o-transition:none !important;transition:none !important}.btn:focus:not(:focus-visible){box-shadow:none}.page-navigation{display:flex;justify-content:space-between}.nav-page{padding-bottom:.75em}.nav-page .bi{font-size:1.8rem;vertical-align:middle}.nav-page .nav-page-text{padding-left:.25em;padding-right:.25em}.nav-page a{color:#6c757d;text-decoration:none;display:flex;align-items:center}.nav-page a:hover{color:#339}.nav-footer .toc-actions{padding-bottom:.5em;padding-top:.5em}.nav-footer .toc-actions a,.nav-footer .toc-actions a:hover{text-decoration:none}.nav-footer .toc-actions ul{display:flex;list-style:none}.nav-footer .toc-actions ul :first-child{margin-left:auto}.nav-footer .toc-actions ul :last-child{margin-right:auto}.nav-footer .toc-actions ul li{padding-right:1.5em}.nav-footer .toc-actions ul li i.bi{padding-right:.4em}.nav-footer .toc-actions ul li:last-of-type{padding-right:0}.nav-footer{display:flex;flex-direction:row;flex-wrap:wrap;justify-content:space-between;align-items:baseline;text-align:center;padding-top:.5rem;padding-bottom:.5rem;background-color:#fff}body.nav-fixed{padding-top:64px}.nav-footer-contents{color:#6c757d;margin-top:.25rem}.nav-footer{min-height:3.5em;color:#757575}.nav-footer a{color:#757575}.nav-footer .nav-footer-left{font-size:.825em}.nav-footer .nav-footer-center{font-size:.825em}.nav-footer .nav-footer-right{font-size:.825em}.nav-footer-left .footer-items,.nav-footer-center .footer-items,.nav-footer-right .footer-items{display:inline-flex;padding-top:.3em;padding-bottom:.3em;margin-bottom:0em}.nav-footer-left .footer-items .nav-link,.nav-footer-center .footer-items .nav-link,.nav-footer-right .footer-items .nav-link{padding-left:.6em;padding-right:.6em}@media(min-width: 768px){.nav-footer-left{flex:1 1 0px;text-align:left}}@media(max-width: 575.98px){.nav-footer-left{margin-bottom:1em;flex:100%}}@media(min-width: 768px){.nav-footer-right{flex:1 1 0px;text-align:right}}@media(max-width: 575.98px){.nav-footer-right{margin-bottom:1em;flex:100%}}.nav-footer-center{text-align:center;min-height:3em}@media(min-width: 768px){.nav-footer-center{flex:1 1 0px}}.nav-footer-center .footer-items{justify-content:center}@media(max-width: 767.98px){.nav-footer-center{margin-bottom:1em;flex:100%}}@media(max-width: 767.98px){.nav-footer-center{margin-top:3em;order:10}}.navbar .quarto-reader-toggle.reader .quarto-reader-toggle-btn{background-color:#fff;border-radius:3px}@media(max-width: 991.98px){.quarto-reader-toggle{display:none}}.quarto-reader-toggle.reader.quarto-navigation-tool .quarto-reader-toggle-btn{background-color:#595959;border-radius:3px}.quarto-reader-toggle .quarto-reader-toggle-btn{display:inline-flex;padding-left:.2em;padding-right:.2em;margin-left:-0.2em;margin-right:-0.2em;text-align:center}.navbar .quarto-reader-toggle:not(.reader) .bi::before{background-image:url('data:image/svg+xml,')}.navbar .quarto-reader-toggle.reader .bi::before{background-image:url('data:image/svg+xml,')}.sidebar-navigation .quarto-reader-toggle:not(.reader) .bi::before{background-image:url('data:image/svg+xml,')}.sidebar-navigation .quarto-reader-toggle.reader .bi::before{background-image:url('data:image/svg+xml,')}#quarto-back-to-top{display:none;position:fixed;bottom:50px;background-color:#fff;border-radius:.25rem;box-shadow:0 .2rem .5rem #6c757d,0 0 .05rem #6c757d;color:#6c757d;text-decoration:none;font-size:.9em;text-align:center;left:50%;padding:.4rem .8rem;transform:translate(-50%, 0)}#quarto-announcement{padding:.5em;display:flex;justify-content:space-between;margin-bottom:0;font-size:.9em}#quarto-announcement .quarto-announcement-content{margin-right:auto}#quarto-announcement .quarto-announcement-content p{margin-bottom:0}#quarto-announcement .quarto-announcement-icon{margin-right:.5em;font-size:1.2em;margin-top:-0.15em}#quarto-announcement .quarto-announcement-action{cursor:pointer}.aa-DetachedSearchButtonQuery{display:none}.aa-DetachedOverlay ul.aa-List,#quarto-search-results ul.aa-List{list-style:none;padding-left:0}.aa-DetachedOverlay .aa-Panel,#quarto-search-results .aa-Panel{background-color:#fff;position:absolute;z-index:2000}#quarto-search-results .aa-Panel{max-width:400px}#quarto-search input{font-size:.925rem}@media(min-width: 992px){.navbar #quarto-search{margin-left:.25rem;order:999}}.navbar.navbar-expand-sm #quarto-search,.navbar.navbar-expand-md #quarto-search{order:999}@media(min-width: 992px){.navbar .quarto-navbar-tools{order:900}}@media(min-width: 992px){.navbar .quarto-navbar-tools.tools-end{margin-left:auto !important}}@media(max-width: 991.98px){#quarto-sidebar .sidebar-search{display:none}}#quarto-sidebar .sidebar-search .aa-Autocomplete{width:100%}.navbar .aa-Autocomplete .aa-Form{width:180px}.navbar #quarto-search.type-overlay .aa-Autocomplete{width:40px}.navbar #quarto-search.type-overlay .aa-Autocomplete .aa-Form{background-color:inherit;border:none}.navbar #quarto-search.type-overlay .aa-Autocomplete .aa-Form:focus-within{box-shadow:none;outline:none}.navbar #quarto-search.type-overlay .aa-Autocomplete .aa-Form .aa-InputWrapper{display:none}.navbar #quarto-search.type-overlay .aa-Autocomplete .aa-Form .aa-InputWrapper:focus-within{display:inherit}.navbar #quarto-search.type-overlay .aa-Autocomplete .aa-Form .aa-Label svg,.navbar #quarto-search.type-overlay .aa-Autocomplete .aa-Form .aa-LoadingIndicator svg{width:26px;height:26px;color:#fff;opacity:1}.navbar #quarto-search.type-overlay .aa-Autocomplete svg.aa-SubmitIcon{width:26px;height:26px;color:#fff;opacity:1}.aa-Autocomplete .aa-Form,.aa-DetachedFormContainer .aa-Form{align-items:center;background-color:#fff;border:1px solid #dee2e6;border-radius:.25rem;color:#000;display:flex;line-height:1em;margin:0;position:relative;width:100%}.aa-Autocomplete .aa-Form:focus-within,.aa-DetachedFormContainer .aa-Form:focus-within{box-shadow:rgba(232,232,252,.6) 0 0 0 1px;outline:currentColor none medium}.aa-Autocomplete .aa-Form .aa-InputWrapperPrefix,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperPrefix{align-items:center;display:flex;flex-shrink:0;order:1}.aa-Autocomplete .aa-Form .aa-InputWrapperPrefix .aa-Label,.aa-Autocomplete .aa-Form .aa-InputWrapperPrefix .aa-LoadingIndicator,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperPrefix .aa-Label,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperPrefix .aa-LoadingIndicator{cursor:initial;flex-shrink:0;padding:0;text-align:left}.aa-Autocomplete .aa-Form .aa-InputWrapperPrefix .aa-Label svg,.aa-Autocomplete .aa-Form .aa-InputWrapperPrefix .aa-LoadingIndicator svg,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperPrefix .aa-Label svg,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperPrefix .aa-LoadingIndicator svg{color:#000;opacity:.5}.aa-Autocomplete .aa-Form .aa-InputWrapperPrefix .aa-SubmitButton,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperPrefix .aa-SubmitButton{appearance:none;background:none;border:0;margin:0}.aa-Autocomplete .aa-Form .aa-InputWrapperPrefix .aa-LoadingIndicator,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperPrefix .aa-LoadingIndicator{align-items:center;display:flex;justify-content:center}.aa-Autocomplete .aa-Form .aa-InputWrapperPrefix .aa-LoadingIndicator[hidden],.aa-DetachedFormContainer .aa-Form .aa-InputWrapperPrefix .aa-LoadingIndicator[hidden]{display:none}.aa-Autocomplete .aa-Form .aa-InputWrapper,.aa-DetachedFormContainer .aa-Form .aa-InputWrapper{order:3;position:relative;width:100%}.aa-Autocomplete .aa-Form .aa-InputWrapper .aa-Input,.aa-DetachedFormContainer .aa-Form .aa-InputWrapper .aa-Input{appearance:none;background:none;border:0;color:#000;font:inherit;height:calc(1.5em + .1rem + 2px);padding:0;width:100%}.aa-Autocomplete .aa-Form .aa-InputWrapper .aa-Input::placeholder,.aa-DetachedFormContainer .aa-Form .aa-InputWrapper .aa-Input::placeholder{color:#000;opacity:.8}.aa-Autocomplete .aa-Form .aa-InputWrapper .aa-Input:focus,.aa-DetachedFormContainer .aa-Form .aa-InputWrapper .aa-Input:focus{border-color:none;box-shadow:none;outline:none}.aa-Autocomplete .aa-Form .aa-InputWrapper .aa-Input::-webkit-search-decoration,.aa-Autocomplete .aa-Form .aa-InputWrapper .aa-Input::-webkit-search-cancel-button,.aa-Autocomplete .aa-Form .aa-InputWrapper .aa-Input::-webkit-search-results-button,.aa-Autocomplete .aa-Form .aa-InputWrapper .aa-Input::-webkit-search-results-decoration,.aa-DetachedFormContainer .aa-Form .aa-InputWrapper .aa-Input::-webkit-search-decoration,.aa-DetachedFormContainer .aa-Form .aa-InputWrapper .aa-Input::-webkit-search-cancel-button,.aa-DetachedFormContainer .aa-Form .aa-InputWrapper .aa-Input::-webkit-search-results-button,.aa-DetachedFormContainer .aa-Form .aa-InputWrapper .aa-Input::-webkit-search-results-decoration{display:none}.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix{align-items:center;display:flex;order:4}.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-ClearButton,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-ClearButton{align-items:center;background:none;border:0;color:#000;opacity:.8;cursor:pointer;display:flex;margin:0;width:calc(1.5em + .1rem + 2px)}.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-ClearButton:hover,.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-ClearButton:focus,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-ClearButton:hover,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-ClearButton:focus{color:#000;opacity:.8}.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-ClearButton[hidden],.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-ClearButton[hidden]{display:none}.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-ClearButton svg,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-ClearButton svg{width:calc(1.5em + 0.75rem + calc(1px * 2))}.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-CopyButton,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-CopyButton{border:none;align-items:center;background:none;color:#000;opacity:.4;font-size:.7rem;cursor:pointer;display:none;margin:0;width:calc(1em + .1rem + 2px)}.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-CopyButton:hover,.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-CopyButton:focus,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-CopyButton:hover,.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-CopyButton:focus{color:#000;opacity:.8}.aa-Autocomplete .aa-Form .aa-InputWrapperSuffix .aa-CopyButton[hidden],.aa-DetachedFormContainer .aa-Form .aa-InputWrapperSuffix .aa-CopyButton[hidden]{display:none}.aa-PanelLayout:empty{display:none}.quarto-search-no-results.no-query{display:none}.aa-Source:has(.no-query){display:none}#quarto-search-results .aa-Panel{border:solid #dee2e6 1px}#quarto-search-results .aa-SourceNoResults{width:398px}.aa-DetachedOverlay .aa-Panel,#quarto-search-results .aa-Panel{max-height:65vh;overflow-y:auto;font-size:.925rem}.aa-DetachedOverlay .aa-SourceNoResults,#quarto-search-results .aa-SourceNoResults{height:60px;display:flex;justify-content:center;align-items:center}.aa-DetachedOverlay .search-error,#quarto-search-results .search-error{padding-top:10px;padding-left:20px;padding-right:20px;cursor:default}.aa-DetachedOverlay .search-error .search-error-title,#quarto-search-results .search-error .search-error-title{font-size:1.1rem;margin-bottom:.5rem}.aa-DetachedOverlay .search-error .search-error-title .search-error-icon,#quarto-search-results .search-error .search-error-title .search-error-icon{margin-right:8px}.aa-DetachedOverlay .search-error .search-error-text,#quarto-search-results .search-error .search-error-text{font-weight:300}.aa-DetachedOverlay .search-result-text,#quarto-search-results .search-result-text{font-weight:300;overflow:hidden;text-overflow:ellipsis;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;line-height:1.2rem;max-height:2.4rem}.aa-DetachedOverlay .aa-SourceHeader .search-result-header,#quarto-search-results .aa-SourceHeader .search-result-header{font-size:.875rem;background-color:#f2f2f2;padding-left:14px;padding-bottom:4px;padding-top:4px}.aa-DetachedOverlay .aa-SourceHeader .search-result-header-no-results,#quarto-search-results .aa-SourceHeader .search-result-header-no-results{display:none}.aa-DetachedOverlay .aa-SourceFooter .algolia-search-logo,#quarto-search-results .aa-SourceFooter .algolia-search-logo{width:110px;opacity:.85;margin:8px;float:right}.aa-DetachedOverlay .search-result-section,#quarto-search-results .search-result-section{font-size:.925em}.aa-DetachedOverlay a.search-result-link,#quarto-search-results a.search-result-link{color:inherit;text-decoration:none}.aa-DetachedOverlay li.aa-Item[aria-selected=true] .search-item,#quarto-search-results li.aa-Item[aria-selected=true] .search-item{background-color:#e8e8fc}.aa-DetachedOverlay li.aa-Item[aria-selected=true] .search-item.search-result-more,.aa-DetachedOverlay li.aa-Item[aria-selected=true] .search-item .search-result-section,.aa-DetachedOverlay li.aa-Item[aria-selected=true] .search-item .search-result-text,.aa-DetachedOverlay li.aa-Item[aria-selected=true] .search-item .search-result-title-container,.aa-DetachedOverlay li.aa-Item[aria-selected=true] .search-item .search-result-text-container,#quarto-search-results li.aa-Item[aria-selected=true] .search-item.search-result-more,#quarto-search-results li.aa-Item[aria-selected=true] .search-item .search-result-section,#quarto-search-results li.aa-Item[aria-selected=true] .search-item .search-result-text,#quarto-search-results li.aa-Item[aria-selected=true] .search-item .search-result-title-container,#quarto-search-results li.aa-Item[aria-selected=true] .search-item .search-result-text-container{color:#000;background-color:#e8e8fc}.aa-DetachedOverlay li.aa-Item[aria-selected=true] .search-item mark.search-match,.aa-DetachedOverlay li.aa-Item[aria-selected=true] .search-item .search-match.mark,#quarto-search-results li.aa-Item[aria-selected=true] .search-item mark.search-match,#quarto-search-results li.aa-Item[aria-selected=true] .search-item .search-match.mark{color:#000;background-color:#fff}.aa-DetachedOverlay li.aa-Item[aria-selected=false] .search-item,#quarto-search-results li.aa-Item[aria-selected=false] .search-item{background-color:#fff}.aa-DetachedOverlay li.aa-Item[aria-selected=false] .search-item.search-result-more,.aa-DetachedOverlay li.aa-Item[aria-selected=false] .search-item .search-result-section,.aa-DetachedOverlay li.aa-Item[aria-selected=false] .search-item .search-result-text,.aa-DetachedOverlay li.aa-Item[aria-selected=false] .search-item .search-result-title-container,.aa-DetachedOverlay li.aa-Item[aria-selected=false] .search-item .search-result-text-container,#quarto-search-results li.aa-Item[aria-selected=false] .search-item.search-result-more,#quarto-search-results li.aa-Item[aria-selected=false] .search-item .search-result-section,#quarto-search-results li.aa-Item[aria-selected=false] .search-item .search-result-text,#quarto-search-results li.aa-Item[aria-selected=false] .search-item .search-result-title-container,#quarto-search-results li.aa-Item[aria-selected=false] .search-item .search-result-text-container{color:#000}.aa-DetachedOverlay li.aa-Item[aria-selected=false] .search-item mark.search-match,.aa-DetachedOverlay li.aa-Item[aria-selected=false] .search-item .search-match.mark,#quarto-search-results li.aa-Item[aria-selected=false] .search-item mark.search-match,#quarto-search-results li.aa-Item[aria-selected=false] .search-item .search-match.mark{color:inherit;background-color:#fff}.aa-DetachedOverlay .aa-Item .search-result-doc:not(.document-selectable) .search-result-title-container,#quarto-search-results .aa-Item .search-result-doc:not(.document-selectable) .search-result-title-container{background-color:#fff;color:#000}.aa-DetachedOverlay .aa-Item .search-result-doc:not(.document-selectable) .search-result-text-container,#quarto-search-results .aa-Item .search-result-doc:not(.document-selectable) .search-result-text-container{padding-top:0px}.aa-DetachedOverlay li.aa-Item .search-result-doc.document-selectable .search-result-text-container,#quarto-search-results li.aa-Item .search-result-doc.document-selectable .search-result-text-container{margin-top:-4px}.aa-DetachedOverlay .aa-Item,#quarto-search-results .aa-Item{cursor:pointer}.aa-DetachedOverlay .aa-Item .search-item,#quarto-search-results .aa-Item .search-item{border-left:none;border-right:none;border-top:none;background-color:#fff;border-color:#dee2e6;color:#000}.aa-DetachedOverlay .aa-Item .search-item p,#quarto-search-results .aa-Item .search-item p{margin-top:0;margin-bottom:0}.aa-DetachedOverlay .aa-Item .search-item i.bi,#quarto-search-results .aa-Item .search-item i.bi{padding-left:8px;padding-right:8px;font-size:1.3em}.aa-DetachedOverlay .aa-Item .search-item .search-result-title,#quarto-search-results .aa-Item .search-item .search-result-title{margin-top:.3em;margin-bottom:0em}.aa-DetachedOverlay .aa-Item .search-item .search-result-crumbs,#quarto-search-results .aa-Item .search-item .search-result-crumbs{white-space:nowrap;text-overflow:ellipsis;font-size:.8em;font-weight:300;margin-right:1em}.aa-DetachedOverlay .aa-Item .search-item .search-result-crumbs:not(.search-result-crumbs-wrap),#quarto-search-results .aa-Item .search-item .search-result-crumbs:not(.search-result-crumbs-wrap){max-width:30%;margin-left:auto;margin-top:.5em;margin-bottom:.1rem}.aa-DetachedOverlay .aa-Item .search-item .search-result-crumbs.search-result-crumbs-wrap,#quarto-search-results .aa-Item .search-item .search-result-crumbs.search-result-crumbs-wrap{flex-basis:100%;margin-top:0em;margin-bottom:.2em;margin-left:37px}.aa-DetachedOverlay .aa-Item .search-result-title-container,#quarto-search-results .aa-Item .search-result-title-container{font-size:1em;display:flex;flex-wrap:wrap;padding:6px 4px 6px 4px}.aa-DetachedOverlay .aa-Item .search-result-text-container,#quarto-search-results .aa-Item .search-result-text-container{padding-bottom:8px;padding-right:8px;margin-left:42px}.aa-DetachedOverlay .aa-Item .search-result-doc-section,.aa-DetachedOverlay .aa-Item .search-result-more,#quarto-search-results .aa-Item .search-result-doc-section,#quarto-search-results .aa-Item .search-result-more{padding-top:8px;padding-bottom:8px;padding-left:44px}.aa-DetachedOverlay .aa-Item .search-result-more,#quarto-search-results .aa-Item .search-result-more{font-size:.8em;font-weight:400}.aa-DetachedOverlay .aa-Item .search-result-doc,#quarto-search-results .aa-Item .search-result-doc{border-top:1px solid #dee2e6}.aa-DetachedSearchButton{background:none;border:none}.aa-DetachedSearchButton .aa-DetachedSearchButtonPlaceholder{display:none}.navbar .aa-DetachedSearchButton .aa-DetachedSearchButtonIcon{color:#fff}.sidebar-tools-collapse #quarto-search,.sidebar-tools-main #quarto-search{display:inline}.sidebar-tools-collapse #quarto-search .aa-Autocomplete,.sidebar-tools-main #quarto-search .aa-Autocomplete{display:inline}.sidebar-tools-collapse #quarto-search .aa-DetachedSearchButton,.sidebar-tools-main #quarto-search .aa-DetachedSearchButton{padding-left:4px;padding-right:4px}.sidebar-tools-collapse #quarto-search .aa-DetachedSearchButton .aa-DetachedSearchButtonIcon,.sidebar-tools-main #quarto-search .aa-DetachedSearchButton .aa-DetachedSearchButtonIcon{color:#595959}.sidebar-tools-collapse #quarto-search .aa-DetachedSearchButton .aa-DetachedSearchButtonIcon .aa-SubmitIcon,.sidebar-tools-main #quarto-search .aa-DetachedSearchButton .aa-DetachedSearchButtonIcon .aa-SubmitIcon{margin-top:-3px}.aa-DetachedContainer{background:rgba(255,255,255,.65);width:90%;bottom:0;box-shadow:rgba(222,226,230,.6) 0 0 0 1px;outline:currentColor none medium;display:flex;flex-direction:column;left:0;margin:0;overflow:hidden;padding:0;position:fixed;right:0;top:0;z-index:1101}.aa-DetachedContainer::after{height:32px}.aa-DetachedContainer .aa-SourceHeader{margin:var(--aa-spacing-half) 0 var(--aa-spacing-half) 2px}.aa-DetachedContainer .aa-Panel{background-color:#fff;border-radius:0;box-shadow:none;flex-grow:1;margin:0;padding:0;position:relative}.aa-DetachedContainer .aa-PanelLayout{bottom:0;box-shadow:none;left:0;margin:0;max-height:none;overflow-y:auto;position:absolute;right:0;top:0;width:100%}.aa-DetachedFormContainer{background-color:#fff;border-bottom:1px solid #dee2e6;display:flex;flex-direction:row;justify-content:space-between;margin:0;padding:.5em}.aa-DetachedCancelButton{background:none;font-size:.8em;border:0;border-radius:3px;color:#000;cursor:pointer;margin:0 0 0 .5em;padding:0 .5em}.aa-DetachedCancelButton:hover,.aa-DetachedCancelButton:focus{box-shadow:rgba(232,232,252,.6) 0 0 0 1px;outline:currentColor none medium}.aa-DetachedContainer--modal{bottom:inherit;height:auto;margin:0 auto;position:absolute;top:100px;border-radius:6px;max-width:850px}@media(max-width: 575.98px){.aa-DetachedContainer--modal{width:100%;top:0px;border-radius:0px;border:none}}.aa-DetachedContainer--modal .aa-PanelLayout{max-height:var(--aa-detached-modal-max-height);padding-bottom:var(--aa-spacing-half);position:static}.aa-Detached{height:100vh;overflow:hidden}.aa-DetachedOverlay{background-color:rgba(0,0,0,.4);position:fixed;left:0;right:0;top:0;margin:0;padding:0;height:100vh;z-index:1100}.quarto-dashboard.nav-fixed.dashboard-sidebar #quarto-content.quarto-dashboard-content{padding:0em}.quarto-dashboard #quarto-content.quarto-dashboard-content{padding:1em}.quarto-dashboard #quarto-content.quarto-dashboard-content>*{padding-top:0}@media(min-width: 576px){.quarto-dashboard{height:100%}}.quarto-dashboard .card.valuebox.bslib-card.bg-primary{background-color:#5397e9 !important}.quarto-dashboard .card.valuebox.bslib-card.bg-secondary{background-color:#343a40 !important}.quarto-dashboard .card.valuebox.bslib-card.bg-success{background-color:#3aa716 !important}.quarto-dashboard .card.valuebox.bslib-card.bg-info{background-color:rgba(153,84,187,.7019607843) !important}.quarto-dashboard .card.valuebox.bslib-card.bg-warning{background-color:#fa6400 !important}.quarto-dashboard .card.valuebox.bslib-card.bg-danger{background-color:rgba(255,0,57,.7019607843) !important}.quarto-dashboard .card.valuebox.bslib-card.bg-light{background-color:#f8f9fa !important}.quarto-dashboard .card.valuebox.bslib-card.bg-dark{background-color:#343a40 !important}.quarto-dashboard.dashboard-fill{display:flex;flex-direction:column}.quarto-dashboard #quarto-appendix{display:none}.quarto-dashboard #quarto-header #quarto-dashboard-header{border-top:solid 1px #22c472;border-bottom:solid 1px #22c472}.quarto-dashboard #quarto-header #quarto-dashboard-header>nav{padding-left:1em;padding-right:1em}.quarto-dashboard #quarto-header #quarto-dashboard-header>nav .navbar-brand-container{padding-left:0}.quarto-dashboard #quarto-header #quarto-dashboard-header .navbar-toggler{margin-right:0}.quarto-dashboard #quarto-header #quarto-dashboard-header .navbar-toggler-icon{height:1em;width:1em;background-image:url('data:image/svg+xml,')}.quarto-dashboard #quarto-header #quarto-dashboard-header .navbar-brand-container{padding-right:1em}.quarto-dashboard #quarto-header #quarto-dashboard-header .navbar-title{font-size:1.1em}.quarto-dashboard #quarto-header #quarto-dashboard-header .navbar-nav{font-size:.9em}.quarto-dashboard #quarto-dashboard-header .navbar{padding:0}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-container{padding-left:1em}.quarto-dashboard #quarto-dashboard-header .navbar.slim .navbar-brand-container .nav-link,.quarto-dashboard #quarto-dashboard-header .navbar.slim .navbar-nav .nav-link{padding:.7em}.quarto-dashboard #quarto-dashboard-header .navbar .quarto-color-scheme-toggle{order:9}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-toggler{margin-left:.5em;order:10}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-nav .nav-link{padding:.5em;height:100%;display:flex;align-items:center}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-nav .active{background-color:#24cd78}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-brand-container{padding:.5em .5em .5em 0;display:flex;flex-direction:row;margin-right:2em;align-items:center}@media(max-width: 767.98px){.quarto-dashboard #quarto-dashboard-header .navbar .navbar-brand-container{margin-right:auto}}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-collapse{align-self:stretch}@media(min-width: 768px){.quarto-dashboard #quarto-dashboard-header .navbar .navbar-collapse{order:8}}@media(max-width: 767.98px){.quarto-dashboard #quarto-dashboard-header .navbar .navbar-collapse{order:1000;padding-bottom:.5em}}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-collapse .navbar-nav{align-self:stretch}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-title{font-size:1.25em;line-height:1.1em;display:flex;flex-direction:row;flex-wrap:wrap;align-items:baseline}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-title .navbar-title-text{margin-right:.4em}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-title a{text-decoration:none;color:inherit}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-subtitle,.quarto-dashboard #quarto-dashboard-header .navbar .navbar-author{font-size:.9rem;margin-right:.5em}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-author{margin-left:auto}.quarto-dashboard #quarto-dashboard-header .navbar .navbar-logo{max-height:48px;min-height:30px;object-fit:cover;margin-right:1em}.quarto-dashboard #quarto-dashboard-header .navbar .quarto-dashboard-links{order:9;padding-right:1em}.quarto-dashboard #quarto-dashboard-header .navbar .quarto-dashboard-link-text{margin-left:.25em}.quarto-dashboard #quarto-dashboard-header .navbar .quarto-dashboard-link{padding-right:0em;padding-left:.7em;text-decoration:none;color:#fff}.quarto-dashboard .page-layout-custom .tab-content{padding:0;border:none}.quarto-dashboard-img-contain{height:100%;width:100%;object-fit:contain}@media(max-width: 575.98px){.quarto-dashboard .bslib-grid{grid-template-rows:minmax(1em, max-content) !important}.quarto-dashboard .sidebar-content{height:inherit}.quarto-dashboard .page-layout-custom{min-height:100vh}}.quarto-dashboard.dashboard-toolbar>.page-layout-custom,.quarto-dashboard.dashboard-sidebar>.page-layout-custom{padding:0}.quarto-dashboard .quarto-dashboard-content.quarto-dashboard-pages{padding:0}.quarto-dashboard .callout{margin-bottom:0;margin-top:0}.quarto-dashboard .html-fill-container figure{overflow:hidden}.quarto-dashboard bslib-tooltip .rounded-pill{border:solid #6c757d 1px}.quarto-dashboard bslib-tooltip .rounded-pill .svg{fill:#000}.quarto-dashboard .tabset .dashboard-card-no-title .nav-tabs{margin-left:0;margin-right:auto}.quarto-dashboard .tabset .tab-content{border:none}.quarto-dashboard .tabset .card-header .nav-link[role=tab]{margin-top:-6px;padding-top:6px;padding-bottom:6px}.quarto-dashboard .card.valuebox,.quarto-dashboard .card.bslib-value-box{min-height:3rem}.quarto-dashboard .card.valuebox .card-body,.quarto-dashboard .card.bslib-value-box .card-body{padding:0}.quarto-dashboard .bslib-value-box .value-box-value{font-size:clamp(.1em,15cqw,5em)}.quarto-dashboard .bslib-value-box .value-box-showcase .bi{font-size:clamp(.1em,max(18cqw,5.2cqh),5em);text-align:center;height:1em}.quarto-dashboard .bslib-value-box .value-box-showcase .bi::before{vertical-align:1em}.quarto-dashboard .bslib-value-box .value-box-area{margin-top:auto;margin-bottom:auto}.quarto-dashboard .card figure.quarto-float{display:flex;flex-direction:column;align-items:center}.quarto-dashboard .dashboard-scrolling{padding:1em}.quarto-dashboard .full-height{height:100%}.quarto-dashboard .showcase-bottom .value-box-grid{display:grid;grid-template-columns:1fr;grid-template-rows:1fr auto;grid-template-areas:"top" "bottom"}.quarto-dashboard .showcase-bottom .value-box-grid .value-box-showcase{grid-area:bottom;padding:0;margin:0}.quarto-dashboard .showcase-bottom .value-box-grid .value-box-showcase i.bi{font-size:4rem}.quarto-dashboard .showcase-bottom .value-box-grid .value-box-area{grid-area:top}.quarto-dashboard .tab-content{margin-bottom:0}.quarto-dashboard .bslib-card .bslib-navs-card-title{justify-content:stretch;align-items:end}.quarto-dashboard .card-header{display:flex;flex-wrap:wrap;justify-content:space-between}.quarto-dashboard .card-header .card-title{display:flex;flex-direction:column;justify-content:center;margin-bottom:0}.quarto-dashboard .tabset .card-toolbar{margin-bottom:1em}.quarto-dashboard .bslib-grid>.bslib-sidebar-layout{border:none;gap:var(--bslib-spacer, 1rem)}.quarto-dashboard .bslib-grid>.bslib-sidebar-layout>.main{padding:0}.quarto-dashboard .bslib-grid>.bslib-sidebar-layout>.sidebar{border-radius:.25rem;border:1px solid rgba(0,0,0,.175)}.quarto-dashboard .bslib-grid>.bslib-sidebar-layout>.collapse-toggle{display:none}@media(max-width: 767.98px){.quarto-dashboard .bslib-grid>.bslib-sidebar-layout{grid-template-columns:1fr;grid-template-rows:max-content 1fr}.quarto-dashboard .bslib-grid>.bslib-sidebar-layout>.main{grid-column:1;grid-row:2}.quarto-dashboard .bslib-grid>.bslib-sidebar-layout .sidebar{grid-column:1;grid-row:1}}.quarto-dashboard .sidebar-right .sidebar{padding-left:2.5em}.quarto-dashboard .sidebar-right .collapse-toggle{left:2px}.quarto-dashboard .quarto-dashboard .sidebar-right button.collapse-toggle:not(.transitioning){left:unset}.quarto-dashboard aside.sidebar{padding-left:1em;padding-right:1em;background-color:rgba(52,58,64,.25);color:#000}.quarto-dashboard .bslib-sidebar-layout>div.main{padding:.7em}.quarto-dashboard .bslib-sidebar-layout button.collapse-toggle{margin-top:.3em}.quarto-dashboard .bslib-sidebar-layout .collapse-toggle{top:0}.quarto-dashboard .bslib-sidebar-layout.sidebar-collapsed:not(.transitioning):not(.sidebar-right) .collapse-toggle{left:2px}.quarto-dashboard .sidebar>section>.h3:first-of-type{margin-top:0em}.quarto-dashboard .sidebar .h3,.quarto-dashboard .sidebar .h4,.quarto-dashboard .sidebar .h5,.quarto-dashboard .sidebar .h6{margin-top:.5em}.quarto-dashboard .sidebar form{flex-direction:column;align-items:start;margin-bottom:1em}.quarto-dashboard .sidebar form div[class*=oi-][class$=-input]{flex-direction:column}.quarto-dashboard .sidebar form[class*=oi-][class$=-toggle]{flex-direction:row-reverse;align-items:center;justify-content:start}.quarto-dashboard .sidebar form input[type=range]{margin-top:.5em;margin-right:.8em;margin-left:1em}.quarto-dashboard .sidebar label{width:fit-content}.quarto-dashboard .sidebar .card-body{margin-bottom:2em}.quarto-dashboard .sidebar .shiny-input-container{margin-bottom:1em}.quarto-dashboard .sidebar .shiny-options-group{margin-top:0}.quarto-dashboard .sidebar .control-label{margin-bottom:.3em}.quarto-dashboard .card .card-body .quarto-layout-row{align-items:stretch}.quarto-dashboard .toolbar{font-size:.9em;display:flex;flex-direction:row;border-top:solid 1px #bcbfc0;padding:1em;flex-wrap:wrap;background-color:rgba(52,58,64,.25)}.quarto-dashboard .toolbar .cell-output-display{display:flex}.quarto-dashboard .toolbar .shiny-input-container{padding-bottom:.5em;margin-bottom:.5em;width:inherit}.quarto-dashboard .toolbar .shiny-input-container>.checkbox:first-child{margin-top:6px}.quarto-dashboard .toolbar>*:last-child{margin-right:0}.quarto-dashboard .toolbar>*>*{margin-right:1em;align-items:baseline}.quarto-dashboard .toolbar>*>*>a{text-decoration:none;margin-top:auto;margin-bottom:auto}.quarto-dashboard .toolbar .shiny-input-container{padding-bottom:0;margin-bottom:0}.quarto-dashboard .toolbar .shiny-input-container>*{flex-shrink:0;flex-grow:0}.quarto-dashboard .toolbar .form-group.shiny-input-container:not([role=group])>label{margin-bottom:0}.quarto-dashboard .toolbar .shiny-input-container.no-baseline{align-items:start;padding-top:6px}.quarto-dashboard .toolbar .shiny-input-container{display:flex;align-items:baseline}.quarto-dashboard .toolbar .shiny-input-container label{padding-right:.4em}.quarto-dashboard .toolbar .shiny-input-container .bslib-input-switch{margin-top:6px}.quarto-dashboard .toolbar input[type=text]{line-height:1;width:inherit}.quarto-dashboard .toolbar .input-daterange{width:inherit}.quarto-dashboard .toolbar .input-daterange input[type=text]{height:2.4em;width:10em}.quarto-dashboard .toolbar .input-daterange .input-group-addon{height:auto;padding:0;margin-left:-5px !important;margin-right:-5px}.quarto-dashboard .toolbar .input-daterange .input-group-addon .input-group-text{padding-top:0;padding-bottom:0;height:100%}.quarto-dashboard .toolbar span.irs.irs--shiny{width:10em}.quarto-dashboard .toolbar span.irs.irs--shiny .irs-line{top:9px}.quarto-dashboard .toolbar span.irs.irs--shiny .irs-min,.quarto-dashboard .toolbar span.irs.irs--shiny .irs-max,.quarto-dashboard .toolbar span.irs.irs--shiny .irs-from,.quarto-dashboard .toolbar span.irs.irs--shiny .irs-to,.quarto-dashboard .toolbar span.irs.irs--shiny .irs-single{top:20px}.quarto-dashboard .toolbar span.irs.irs--shiny .irs-bar{top:8px}.quarto-dashboard .toolbar span.irs.irs--shiny .irs-handle{top:0px}.quarto-dashboard .toolbar .shiny-input-checkboxgroup>label{margin-top:6px}.quarto-dashboard .toolbar .shiny-input-checkboxgroup>.shiny-options-group{margin-top:0;align-items:baseline}.quarto-dashboard .toolbar .shiny-input-radiogroup>label{margin-top:6px}.quarto-dashboard .toolbar .shiny-input-radiogroup>.shiny-options-group{align-items:baseline;margin-top:0}.quarto-dashboard .toolbar .shiny-input-radiogroup>.shiny-options-group>.radio{margin-right:.3em}.quarto-dashboard .toolbar .form-select{padding-top:.2em;padding-bottom:.2em}.quarto-dashboard .toolbar .shiny-input-select{min-width:6em}.quarto-dashboard .toolbar div.checkbox{margin-bottom:0px}.quarto-dashboard .toolbar>.checkbox:first-child{margin-top:6px}.quarto-dashboard .toolbar form{width:fit-content}.quarto-dashboard .toolbar form label{padding-top:.2em;padding-bottom:.2em;width:fit-content}.quarto-dashboard .toolbar form input[type=date]{width:fit-content}.quarto-dashboard .toolbar form input[type=color]{width:3em}.quarto-dashboard .toolbar form button{padding:.4em}.quarto-dashboard .toolbar form select{width:fit-content}.quarto-dashboard .toolbar>*{font-size:.9em;flex-grow:0}.quarto-dashboard .toolbar .shiny-input-container label{margin-bottom:1px}.quarto-dashboard .toolbar-bottom{margin-top:1em;margin-bottom:0 !important;order:2}.quarto-dashboard .quarto-dashboard-content>.dashboard-toolbar-container>.toolbar-content>.tab-content>.tab-pane>*:not(.bslib-sidebar-layout){padding:1em}.quarto-dashboard .quarto-dashboard-content>.dashboard-toolbar-container>.toolbar-content>*:not(.tab-content){padding:1em}.quarto-dashboard .quarto-dashboard-content>.tab-content>.dashboard-page>.dashboard-toolbar-container>.toolbar-content,.quarto-dashboard .quarto-dashboard-content>.tab-content>.dashboard-page:not(.dashboard-sidebar-container)>*:not(.dashboard-toolbar-container){padding:1em}.quarto-dashboard .toolbar-content{padding:0}.quarto-dashboard .quarto-dashboard-content.quarto-dashboard-pages .tab-pane>.dashboard-toolbar-container .toolbar{border-radius:0;margin-bottom:0}.quarto-dashboard .dashboard-toolbar-container.toolbar-toplevel .toolbar{border-bottom:1px solid rgba(0,0,0,.175)}.quarto-dashboard .dashboard-toolbar-container.toolbar-toplevel .toolbar-bottom{margin-top:0}.quarto-dashboard .dashboard-toolbar-container:not(.toolbar-toplevel) .toolbar{margin-bottom:1em;border-top:none;border-radius:.25rem;border:1px solid rgba(0,0,0,.175)}.quarto-dashboard .vega-embed.has-actions details{width:1.7em;height:2em;position:absolute !important;top:0;right:0}.quarto-dashboard .dashboard-toolbar-container{padding:0}.quarto-dashboard .card .card-header p:last-child,.quarto-dashboard .card .card-footer p:last-child{margin-bottom:0}.quarto-dashboard .card .card-body>.h4:first-child{margin-top:0}.quarto-dashboard .card .card-body{z-index:4}@media(max-width: 767.98px){.quarto-dashboard .card .card-body .itables div.dataTables_wrapper div.dataTables_length,.quarto-dashboard .card .card-body .itables div.dataTables_wrapper div.dataTables_info,.quarto-dashboard .card .card-body .itables div.dataTables_wrapper div.dataTables_paginate{text-align:initial}.quarto-dashboard .card .card-body .itables div.dataTables_wrapper div.dataTables_filter{text-align:right}.quarto-dashboard .card .card-body .itables div.dataTables_wrapper div.dataTables_paginate ul.pagination{justify-content:initial}}.quarto-dashboard .card .card-body .itables .dataTables_wrapper{display:flex;flex-wrap:wrap;justify-content:space-between;align-items:center;padding-top:0}.quarto-dashboard .card .card-body .itables .dataTables_wrapper table{flex-shrink:0}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dt-buttons{margin-bottom:.5em;margin-left:auto;width:fit-content;float:right}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dt-buttons.btn-group{background:#fff;border:none}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dt-buttons .btn-secondary{background-color:#fff;background-image:none;border:solid #dee2e6 1px;padding:.2em .7em}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dt-buttons .btn span{font-size:.8em;color:#000}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_info{margin-left:.5em;margin-bottom:.5em;padding-top:0}@media(min-width: 768px){.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_info{font-size:.875em}}@media(max-width: 767.98px){.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_info{font-size:.8em}}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_filter{margin-bottom:.5em;font-size:.875em}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_filter input[type=search]{padding:1px 5px 1px 5px;font-size:.875em}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_length{flex-basis:1 1 50%;margin-bottom:.5em;font-size:.875em}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_length select{padding:.4em 3em .4em .5em;font-size:.875em;margin-left:.2em;margin-right:.2em}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_paginate{flex-shrink:0}@media(min-width: 768px){.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_paginate{margin-left:auto}}.quarto-dashboard .card .card-body .itables .dataTables_wrapper .dataTables_paginate ul.pagination .paginate_button .page-link{font-size:.8em}.quarto-dashboard .card .card-footer{font-size:.9em}.quarto-dashboard .card .card-toolbar{display:flex;flex-grow:1;flex-direction:row;width:100%;flex-wrap:wrap}.quarto-dashboard .card .card-toolbar>*{font-size:.8em;flex-grow:0}.quarto-dashboard .card .card-toolbar>.card-title{font-size:1em;flex-grow:1;align-self:flex-start;margin-top:.1em}.quarto-dashboard .card .card-toolbar .cell-output-display{display:flex}.quarto-dashboard .card .card-toolbar .shiny-input-container{padding-bottom:.5em;margin-bottom:.5em;width:inherit}.quarto-dashboard .card .card-toolbar .shiny-input-container>.checkbox:first-child{margin-top:6px}.quarto-dashboard .card .card-toolbar>*:last-child{margin-right:0}.quarto-dashboard .card .card-toolbar>*>*{margin-right:1em;align-items:baseline}.quarto-dashboard .card .card-toolbar>*>*>a{text-decoration:none;margin-top:auto;margin-bottom:auto}.quarto-dashboard .card .card-toolbar form{width:fit-content}.quarto-dashboard .card .card-toolbar form label{padding-top:.2em;padding-bottom:.2em;width:fit-content}.quarto-dashboard .card .card-toolbar form input[type=date]{width:fit-content}.quarto-dashboard .card .card-toolbar form input[type=color]{width:3em}.quarto-dashboard .card .card-toolbar form button{padding:.4em}.quarto-dashboard .card .card-toolbar form select{width:fit-content}.quarto-dashboard .card .card-toolbar .cell-output-display{display:flex}.quarto-dashboard .card .card-toolbar .shiny-input-container{padding-bottom:.5em;margin-bottom:.5em;width:inherit}.quarto-dashboard .card .card-toolbar .shiny-input-container>.checkbox:first-child{margin-top:6px}.quarto-dashboard .card .card-toolbar>*:last-child{margin-right:0}.quarto-dashboard .card .card-toolbar>*>*{margin-right:1em;align-items:baseline}.quarto-dashboard .card .card-toolbar>*>*>a{text-decoration:none;margin-top:auto;margin-bottom:auto}.quarto-dashboard .card .card-toolbar .shiny-input-container{padding-bottom:0;margin-bottom:0}.quarto-dashboard .card .card-toolbar .shiny-input-container>*{flex-shrink:0;flex-grow:0}.quarto-dashboard .card .card-toolbar .form-group.shiny-input-container:not([role=group])>label{margin-bottom:0}.quarto-dashboard .card .card-toolbar .shiny-input-container.no-baseline{align-items:start;padding-top:6px}.quarto-dashboard .card .card-toolbar .shiny-input-container{display:flex;align-items:baseline}.quarto-dashboard .card .card-toolbar .shiny-input-container label{padding-right:.4em}.quarto-dashboard .card .card-toolbar .shiny-input-container .bslib-input-switch{margin-top:6px}.quarto-dashboard .card .card-toolbar input[type=text]{line-height:1;width:inherit}.quarto-dashboard .card .card-toolbar .input-daterange{width:inherit}.quarto-dashboard .card .card-toolbar .input-daterange input[type=text]{height:2.4em;width:10em}.quarto-dashboard .card .card-toolbar .input-daterange .input-group-addon{height:auto;padding:0;margin-left:-5px !important;margin-right:-5px}.quarto-dashboard .card .card-toolbar .input-daterange .input-group-addon .input-group-text{padding-top:0;padding-bottom:0;height:100%}.quarto-dashboard .card .card-toolbar span.irs.irs--shiny{width:10em}.quarto-dashboard .card .card-toolbar span.irs.irs--shiny .irs-line{top:9px}.quarto-dashboard .card .card-toolbar span.irs.irs--shiny .irs-min,.quarto-dashboard .card .card-toolbar span.irs.irs--shiny .irs-max,.quarto-dashboard .card .card-toolbar span.irs.irs--shiny .irs-from,.quarto-dashboard .card .card-toolbar span.irs.irs--shiny .irs-to,.quarto-dashboard .card .card-toolbar span.irs.irs--shiny .irs-single{top:20px}.quarto-dashboard .card .card-toolbar span.irs.irs--shiny .irs-bar{top:8px}.quarto-dashboard .card .card-toolbar span.irs.irs--shiny .irs-handle{top:0px}.quarto-dashboard .card .card-toolbar .shiny-input-checkboxgroup>label{margin-top:6px}.quarto-dashboard .card .card-toolbar .shiny-input-checkboxgroup>.shiny-options-group{margin-top:0;align-items:baseline}.quarto-dashboard .card .card-toolbar .shiny-input-radiogroup>label{margin-top:6px}.quarto-dashboard .card .card-toolbar .shiny-input-radiogroup>.shiny-options-group{align-items:baseline;margin-top:0}.quarto-dashboard .card .card-toolbar .shiny-input-radiogroup>.shiny-options-group>.radio{margin-right:.3em}.quarto-dashboard .card .card-toolbar .form-select{padding-top:.2em;padding-bottom:.2em}.quarto-dashboard .card .card-toolbar .shiny-input-select{min-width:6em}.quarto-dashboard .card .card-toolbar div.checkbox{margin-bottom:0px}.quarto-dashboard .card .card-toolbar>.checkbox:first-child{margin-top:6px}.quarto-dashboard .card-body>table>thead{border-top:none}.quarto-dashboard .card-body>.table>:not(caption)>*>*{background-color:#fff}.tableFloatingHeaderOriginal{background-color:#fff;position:sticky !important;top:0 !important}.dashboard-data-table{margin-top:-1px}div.value-box-area span.observablehq--number{font-size:calc(clamp(.1em,15cqw,5em)*1.25);line-height:1.2;color:inherit;font-family:var(--bs-body-font-family)}.quarto-listing{padding-bottom:1em}.listing-pagination{padding-top:.5em}ul.pagination{float:right;padding-left:8px;padding-top:.5em}ul.pagination li{padding-right:.75em}ul.pagination li.disabled a,ul.pagination li.active a{color:#000;text-decoration:none}ul.pagination li:last-of-type{padding-right:0}.listing-actions-group{display:flex}.quarto-listing-filter{margin-bottom:1em;width:200px;margin-left:auto}.quarto-listing-sort{margin-bottom:1em;margin-right:auto;width:auto}.quarto-listing-sort .input-group-text{font-size:.8em}.input-group-text{border-right:none}.quarto-listing-sort select.form-select{font-size:.8em}.listing-no-matching{text-align:center;padding-top:2em;padding-bottom:3em;font-size:1em}#quarto-margin-sidebar .quarto-listing-category{padding-top:0;font-size:1rem}#quarto-margin-sidebar .quarto-listing-category-title{cursor:pointer;font-weight:600;font-size:1rem}.quarto-listing-category .category{cursor:pointer}.quarto-listing-category .category.active{font-weight:600}.quarto-listing-category.category-cloud{display:flex;flex-wrap:wrap;align-items:baseline}.quarto-listing-category.category-cloud .category{padding-right:5px}.quarto-listing-category.category-cloud .category-cloud-1{font-size:.75em}.quarto-listing-category.category-cloud .category-cloud-2{font-size:.95em}.quarto-listing-category.category-cloud .category-cloud-3{font-size:1.15em}.quarto-listing-category.category-cloud .category-cloud-4{font-size:1.35em}.quarto-listing-category.category-cloud .category-cloud-5{font-size:1.55em}.quarto-listing-category.category-cloud .category-cloud-6{font-size:1.75em}.quarto-listing-category.category-cloud .category-cloud-7{font-size:1.95em}.quarto-listing-category.category-cloud .category-cloud-8{font-size:2.15em}.quarto-listing-category.category-cloud .category-cloud-9{font-size:2.35em}.quarto-listing-category.category-cloud .category-cloud-10{font-size:2.55em}.quarto-listing-cols-1{grid-template-columns:repeat(1, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-1{grid-template-columns:repeat(1, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-1{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-2{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-2{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-2{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-3{grid-template-columns:repeat(3, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-3{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-3{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-4{grid-template-columns:repeat(4, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-4{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-4{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-5{grid-template-columns:repeat(5, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-5{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-5{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-6{grid-template-columns:repeat(6, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-6{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-6{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-7{grid-template-columns:repeat(7, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-7{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-7{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-8{grid-template-columns:repeat(8, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-8{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-8{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-9{grid-template-columns:repeat(9, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-9{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-9{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-10{grid-template-columns:repeat(10, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-10{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-10{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-11{grid-template-columns:repeat(11, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-11{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-11{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-cols-12{grid-template-columns:repeat(12, minmax(0, 1fr));gap:1.5em}@media(max-width: 767.98px){.quarto-listing-cols-12{grid-template-columns:repeat(2, minmax(0, 1fr));gap:1.5em}}@media(max-width: 575.98px){.quarto-listing-cols-12{grid-template-columns:minmax(0, 1fr);gap:1.5em}}.quarto-listing-grid{gap:1.5em}.quarto-grid-item.borderless{border:none}.quarto-grid-item.borderless .listing-categories .listing-category:last-of-type,.quarto-grid-item.borderless .listing-categories .listing-category:first-of-type{padding-left:0}.quarto-grid-item.borderless .listing-categories .listing-category{border:0}.quarto-grid-link{text-decoration:none;color:inherit}.quarto-grid-link:hover{text-decoration:none;color:inherit}.quarto-grid-item h5.title,.quarto-grid-item .title.h5{margin-top:0;margin-bottom:0}.quarto-grid-item .card-footer{display:flex;justify-content:space-between;font-size:.8em}.quarto-grid-item .card-footer p{margin-bottom:0}.quarto-grid-item p.card-img-top{margin-bottom:0}.quarto-grid-item p.card-img-top>img{object-fit:cover}.quarto-grid-item .card-other-values{margin-top:.5em;font-size:.8em}.quarto-grid-item .card-other-values tr{margin-bottom:.5em}.quarto-grid-item .card-other-values tr>td:first-of-type{font-weight:600;padding-right:1em;padding-left:1em;vertical-align:top}.quarto-grid-item div.post-contents{display:flex;flex-direction:column;text-decoration:none;height:100%}.quarto-grid-item .listing-item-img-placeholder{background-color:rgba(52,58,64,.25);flex-shrink:0}.quarto-grid-item .card-attribution{padding-top:1em;display:flex;gap:1em;text-transform:uppercase;color:#6c757d;font-weight:500;flex-grow:10;align-items:flex-end}.quarto-grid-item .description{padding-bottom:1em}.quarto-grid-item .card-attribution .date{align-self:flex-end}.quarto-grid-item .card-attribution.justify{justify-content:space-between}.quarto-grid-item .card-attribution.start{justify-content:flex-start}.quarto-grid-item .card-attribution.end{justify-content:flex-end}.quarto-grid-item .card-title{margin-bottom:.1em}.quarto-grid-item .card-subtitle{padding-top:.25em}.quarto-grid-item .card-text{font-size:.9em}.quarto-grid-item .listing-reading-time{padding-bottom:.25em}.quarto-grid-item .card-text-small{font-size:.8em}.quarto-grid-item .card-subtitle.subtitle{font-size:.9em;font-weight:600;padding-bottom:.5em}.quarto-grid-item .listing-categories{display:flex;flex-wrap:wrap;padding-bottom:5px}.quarto-grid-item .listing-categories .listing-category{color:#6c757d;border:solid 1px #dee2e6;border-radius:.25rem;text-transform:uppercase;font-size:.65em;padding-left:.5em;padding-right:.5em;padding-top:.15em;padding-bottom:.15em;cursor:pointer;margin-right:4px;margin-bottom:4px}.quarto-grid-item.card-right{text-align:right}.quarto-grid-item.card-right .listing-categories{justify-content:flex-end}.quarto-grid-item.card-left{text-align:left}.quarto-grid-item.card-center{text-align:center}.quarto-grid-item.card-center .listing-description{text-align:justify}.quarto-grid-item.card-center .listing-categories{justify-content:center}table.quarto-listing-table td.image{padding:0px}table.quarto-listing-table td.image img{width:100%;max-width:50px;object-fit:contain}table.quarto-listing-table a{text-decoration:none;word-break:keep-all}table.quarto-listing-table th a{color:inherit}table.quarto-listing-table th a.asc:after{margin-bottom:-2px;margin-left:5px;display:inline-block;height:1rem;width:1rem;background-repeat:no-repeat;background-size:1rem 1rem;background-image:url('data:image/svg+xml,');content:""}table.quarto-listing-table th a.desc:after{margin-bottom:-2px;margin-left:5px;display:inline-block;height:1rem;width:1rem;background-repeat:no-repeat;background-size:1rem 1rem;background-image:url('data:image/svg+xml,');content:""}table.quarto-listing-table.table-hover td{cursor:pointer}.quarto-post.image-left{flex-direction:row}.quarto-post.image-right{flex-direction:row-reverse}@media(max-width: 767.98px){.quarto-post.image-right,.quarto-post.image-left{gap:0em;flex-direction:column}.quarto-post .metadata{padding-bottom:1em;order:2}.quarto-post .body{order:1}.quarto-post .thumbnail{order:3}}.list.quarto-listing-default div:last-of-type{border-bottom:none}@media(min-width: 992px){.quarto-listing-container-default{margin-right:2em}}div.quarto-post{display:flex;gap:2em;margin-bottom:1.5em;border-bottom:1px solid #dee2e6}@media(max-width: 767.98px){div.quarto-post{padding-bottom:1em}}div.quarto-post .metadata{flex-basis:20%;flex-grow:0;margin-top:.2em;flex-shrink:10}div.quarto-post .thumbnail{flex-basis:30%;flex-grow:0;flex-shrink:0}div.quarto-post .thumbnail img{margin-top:.4em;width:100%;object-fit:cover}div.quarto-post .body{flex-basis:45%;flex-grow:1;flex-shrink:0}div.quarto-post .body h3.listing-title,div.quarto-post .body .listing-title.h3{margin-top:0px;margin-bottom:0px;border-bottom:none}div.quarto-post .body .listing-subtitle{font-size:.875em;margin-bottom:.5em;margin-top:.2em}div.quarto-post .body .description{font-size:.9em}div.quarto-post .body pre code{white-space:pre-wrap}div.quarto-post a{color:#000;text-decoration:none}div.quarto-post .metadata{display:flex;flex-direction:column;font-size:.8em;font-family:"Source Sans Pro",-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";flex-basis:33%}div.quarto-post .listing-categories{display:flex;flex-wrap:wrap;padding-bottom:5px}div.quarto-post .listing-categories .listing-category{color:#6c757d;border:solid 1px #dee2e6;border-radius:.25rem;text-transform:uppercase;font-size:.65em;padding-left:.5em;padding-right:.5em;padding-top:.15em;padding-bottom:.15em;cursor:pointer;margin-right:4px;margin-bottom:4px}div.quarto-post .listing-description{margin-bottom:.5em}div.quarto-about-jolla{display:flex !important;flex-direction:column;align-items:center;margin-top:10%;padding-bottom:1em}div.quarto-about-jolla .about-image{object-fit:cover;margin-left:auto;margin-right:auto;margin-bottom:1.5em}div.quarto-about-jolla img.round{border-radius:50%}div.quarto-about-jolla img.rounded{border-radius:10px}div.quarto-about-jolla .quarto-title h1.title,div.quarto-about-jolla .quarto-title .title.h1{text-align:center}div.quarto-about-jolla .quarto-title .description{text-align:center}div.quarto-about-jolla h2,div.quarto-about-jolla .h2{border-bottom:none}div.quarto-about-jolla .about-sep{width:60%}div.quarto-about-jolla main{text-align:center}div.quarto-about-jolla .about-links{display:flex}@media(min-width: 992px){div.quarto-about-jolla .about-links{flex-direction:row;column-gap:.8em;row-gap:15px;flex-wrap:wrap}}@media(max-width: 991.98px){div.quarto-about-jolla .about-links{flex-direction:column;row-gap:1em;width:100%;padding-bottom:1.5em}}div.quarto-about-jolla .about-link{color:#333;text-decoration:none;border:solid 1px}@media(min-width: 992px){div.quarto-about-jolla .about-link{font-size:.8em;padding:.25em .5em;border-radius:4px}}@media(max-width: 991.98px){div.quarto-about-jolla .about-link{font-size:1.1em;padding:.5em .5em;text-align:center;border-radius:6px}}div.quarto-about-jolla .about-link:hover{color:#4040bf}div.quarto-about-jolla .about-link i.bi{margin-right:.15em}div.quarto-about-solana{display:flex !important;flex-direction:column;padding-top:3em !important;padding-bottom:1em}div.quarto-about-solana .about-entity{display:flex !important;align-items:start;justify-content:space-between}@media(min-width: 992px){div.quarto-about-solana .about-entity{flex-direction:row}}@media(max-width: 991.98px){div.quarto-about-solana .about-entity{flex-direction:column-reverse;align-items:center;text-align:center}}div.quarto-about-solana .about-entity .entity-contents{display:flex;flex-direction:column}@media(max-width: 767.98px){div.quarto-about-solana .about-entity .entity-contents{width:100%}}div.quarto-about-solana .about-entity .about-image{object-fit:cover}@media(max-width: 991.98px){div.quarto-about-solana .about-entity .about-image{margin-bottom:1.5em}}div.quarto-about-solana .about-entity img.round{border-radius:50%}div.quarto-about-solana .about-entity img.rounded{border-radius:10px}div.quarto-about-solana .about-entity .about-links{display:flex;justify-content:left;padding-bottom:1.2em}@media(min-width: 992px){div.quarto-about-solana .about-entity .about-links{flex-direction:row;column-gap:.8em;row-gap:15px;flex-wrap:wrap}}@media(max-width: 991.98px){div.quarto-about-solana .about-entity .about-links{flex-direction:column;row-gap:1em;width:100%;padding-bottom:1.5em}}div.quarto-about-solana .about-entity .about-link{color:#333;text-decoration:none;border:solid 1px}@media(min-width: 992px){div.quarto-about-solana .about-entity .about-link{font-size:.8em;padding:.25em .5em;border-radius:4px}}@media(max-width: 991.98px){div.quarto-about-solana .about-entity .about-link{font-size:1.1em;padding:.5em .5em;text-align:center;border-radius:6px}}div.quarto-about-solana .about-entity .about-link:hover{color:#4040bf}div.quarto-about-solana .about-entity .about-link i.bi{margin-right:.15em}div.quarto-about-solana .about-contents{padding-right:1.5em;flex-basis:0;flex-grow:1}div.quarto-about-solana .about-contents main.content{margin-top:0}div.quarto-about-solana .about-contents h2,div.quarto-about-solana .about-contents .h2{border-bottom:none}div.quarto-about-trestles{display:flex !important;flex-direction:row;padding-top:3em !important;padding-bottom:1em}@media(max-width: 991.98px){div.quarto-about-trestles{flex-direction:column;padding-top:0em !important}}div.quarto-about-trestles .about-entity{display:flex !important;flex-direction:column;align-items:center;text-align:center;padding-right:1em}@media(min-width: 992px){div.quarto-about-trestles .about-entity{flex:0 0 42%}}div.quarto-about-trestles .about-entity .about-image{object-fit:cover;margin-bottom:1.5em}div.quarto-about-trestles .about-entity img.round{border-radius:50%}div.quarto-about-trestles .about-entity img.rounded{border-radius:10px}div.quarto-about-trestles .about-entity .about-links{display:flex;justify-content:center}@media(min-width: 992px){div.quarto-about-trestles .about-entity .about-links{flex-direction:row;column-gap:.8em;row-gap:15px;flex-wrap:wrap}}@media(max-width: 991.98px){div.quarto-about-trestles .about-entity .about-links{flex-direction:column;row-gap:1em;width:100%;padding-bottom:1.5em}}div.quarto-about-trestles .about-entity .about-link{color:#333;text-decoration:none;border:solid 1px}@media(min-width: 992px){div.quarto-about-trestles .about-entity .about-link{font-size:.8em;padding:.25em .5em;border-radius:4px}}@media(max-width: 991.98px){div.quarto-about-trestles .about-entity .about-link{font-size:1.1em;padding:.5em .5em;text-align:center;border-radius:6px}}div.quarto-about-trestles .about-entity .about-link:hover{color:#4040bf}div.quarto-about-trestles .about-entity .about-link i.bi{margin-right:.15em}div.quarto-about-trestles .about-contents{flex-basis:0;flex-grow:1}div.quarto-about-trestles .about-contents h2,div.quarto-about-trestles .about-contents .h2{border-bottom:none}@media(min-width: 992px){div.quarto-about-trestles .about-contents{border-left:solid 1px #dee2e6;padding-left:1.5em}}div.quarto-about-trestles .about-contents main.content{margin-top:0}div.quarto-about-marquee{padding-bottom:1em}div.quarto-about-marquee .about-contents{display:flex;flex-direction:column}div.quarto-about-marquee .about-image{max-height:550px;margin-bottom:1.5em;object-fit:cover}div.quarto-about-marquee img.round{border-radius:50%}div.quarto-about-marquee img.rounded{border-radius:10px}div.quarto-about-marquee h2,div.quarto-about-marquee .h2{border-bottom:none}div.quarto-about-marquee .about-links{display:flex;justify-content:center;padding-top:1.5em}@media(min-width: 992px){div.quarto-about-marquee .about-links{flex-direction:row;column-gap:.8em;row-gap:15px;flex-wrap:wrap}}@media(max-width: 991.98px){div.quarto-about-marquee .about-links{flex-direction:column;row-gap:1em;width:100%;padding-bottom:1.5em}}div.quarto-about-marquee .about-link{color:#333;text-decoration:none;border:solid 1px}@media(min-width: 992px){div.quarto-about-marquee .about-link{font-size:.8em;padding:.25em .5em;border-radius:4px}}@media(max-width: 991.98px){div.quarto-about-marquee .about-link{font-size:1.1em;padding:.5em .5em;text-align:center;border-radius:6px}}div.quarto-about-marquee .about-link:hover{color:#4040bf}div.quarto-about-marquee .about-link i.bi{margin-right:.15em}@media(min-width: 992px){div.quarto-about-marquee .about-link{border:none}}div.quarto-about-broadside{display:flex;flex-direction:column;padding-bottom:1em}div.quarto-about-broadside .about-main{display:flex !important;padding-top:0 !important}@media(min-width: 992px){div.quarto-about-broadside .about-main{flex-direction:row;align-items:flex-start}}@media(max-width: 991.98px){div.quarto-about-broadside .about-main{flex-direction:column}}@media(max-width: 991.98px){div.quarto-about-broadside .about-main .about-entity{flex-shrink:0;width:100%;height:450px;margin-bottom:1.5em;background-size:cover;background-repeat:no-repeat}}@media(min-width: 992px){div.quarto-about-broadside .about-main .about-entity{flex:0 10 50%;margin-right:1.5em;width:100%;height:100%;background-size:100%;background-repeat:no-repeat}}div.quarto-about-broadside .about-main .about-contents{padding-top:14px;flex:0 0 50%}div.quarto-about-broadside h2,div.quarto-about-broadside .h2{border-bottom:none}div.quarto-about-broadside .about-sep{margin-top:1.5em;width:60%;align-self:center}div.quarto-about-broadside .about-links{display:flex;justify-content:center;column-gap:20px;padding-top:1.5em}@media(min-width: 992px){div.quarto-about-broadside .about-links{flex-direction:row;column-gap:.8em;row-gap:15px;flex-wrap:wrap}}@media(max-width: 991.98px){div.quarto-about-broadside .about-links{flex-direction:column;row-gap:1em;width:100%;padding-bottom:1.5em}}div.quarto-about-broadside .about-link{color:#333;text-decoration:none;border:solid 1px}@media(min-width: 992px){div.quarto-about-broadside .about-link{font-size:.8em;padding:.25em .5em;border-radius:4px}}@media(max-width: 991.98px){div.quarto-about-broadside .about-link{font-size:1.1em;padding:.5em .5em;text-align:center;border-radius:6px}}div.quarto-about-broadside .about-link:hover{color:#4040bf}div.quarto-about-broadside .about-link i.bi{margin-right:.15em}@media(min-width: 992px){div.quarto-about-broadside .about-link{border:none}}.tippy-box[data-theme~=quarto]{background-color:#fff;border:solid 1px #dee2e6;border-radius:.25rem;color:#000;font-size:.875rem}.tippy-box[data-theme~=quarto]>.tippy-backdrop{background-color:#fff}.tippy-box[data-theme~=quarto]>.tippy-arrow:after,.tippy-box[data-theme~=quarto]>.tippy-svg-arrow:after{content:"";position:absolute;z-index:-1}.tippy-box[data-theme~=quarto]>.tippy-arrow:after{border-color:rgba(0,0,0,0);border-style:solid}.tippy-box[data-placement^=top]>.tippy-arrow:before{bottom:-6px}.tippy-box[data-placement^=bottom]>.tippy-arrow:before{top:-6px}.tippy-box[data-placement^=right]>.tippy-arrow:before{left:-6px}.tippy-box[data-placement^=left]>.tippy-arrow:before{right:-6px}.tippy-box[data-theme~=quarto][data-placement^=top]>.tippy-arrow:before{border-top-color:#fff}.tippy-box[data-theme~=quarto][data-placement^=top]>.tippy-arrow:after{border-top-color:#dee2e6;border-width:7px 7px 0;top:17px;left:1px}.tippy-box[data-theme~=quarto][data-placement^=top]>.tippy-svg-arrow>svg{top:16px}.tippy-box[data-theme~=quarto][data-placement^=top]>.tippy-svg-arrow:after{top:17px}.tippy-box[data-theme~=quarto][data-placement^=bottom]>.tippy-arrow:before{border-bottom-color:#fff;bottom:16px}.tippy-box[data-theme~=quarto][data-placement^=bottom]>.tippy-arrow:after{border-bottom-color:#dee2e6;border-width:0 7px 7px;bottom:17px;left:1px}.tippy-box[data-theme~=quarto][data-placement^=bottom]>.tippy-svg-arrow>svg{bottom:15px}.tippy-box[data-theme~=quarto][data-placement^=bottom]>.tippy-svg-arrow:after{bottom:17px}.tippy-box[data-theme~=quarto][data-placement^=left]>.tippy-arrow:before{border-left-color:#fff}.tippy-box[data-theme~=quarto][data-placement^=left]>.tippy-arrow:after{border-left-color:#dee2e6;border-width:7px 0 7px 7px;left:17px;top:1px}.tippy-box[data-theme~=quarto][data-placement^=left]>.tippy-svg-arrow>svg{left:11px}.tippy-box[data-theme~=quarto][data-placement^=left]>.tippy-svg-arrow:after{left:12px}.tippy-box[data-theme~=quarto][data-placement^=right]>.tippy-arrow:before{border-right-color:#fff;right:16px}.tippy-box[data-theme~=quarto][data-placement^=right]>.tippy-arrow:after{border-width:7px 7px 7px 0;right:17px;top:1px;border-right-color:#dee2e6}.tippy-box[data-theme~=quarto][data-placement^=right]>.tippy-svg-arrow>svg{right:11px}.tippy-box[data-theme~=quarto][data-placement^=right]>.tippy-svg-arrow:after{right:12px}.tippy-box[data-theme~=quarto]>.tippy-svg-arrow{fill:#000}.tippy-box[data-theme~=quarto]>.tippy-svg-arrow:after{background-image:url();background-size:16px 6px;width:16px;height:6px}.top-right{position:absolute;top:1em;right:1em}.visually-hidden{border:0;clip:rect(0 0 0 0);height:auto;margin:0;overflow:hidden;padding:0;position:absolute;width:1px;white-space:nowrap}.hidden{display:none !important}.zindex-bottom{z-index:-1 !important}figure.figure{display:block}.quarto-layout-panel{margin-bottom:1em}.quarto-layout-panel>figure{width:100%}.quarto-layout-panel>figure>figcaption,.quarto-layout-panel>.panel-caption{margin-top:10pt}.quarto-layout-panel>.table-caption{margin-top:0px}.table-caption p{margin-bottom:.5em}.quarto-layout-row{display:flex;flex-direction:row;align-items:flex-start}.quarto-layout-valign-top{align-items:flex-start}.quarto-layout-valign-bottom{align-items:flex-end}.quarto-layout-valign-center{align-items:center}.quarto-layout-cell{position:relative;margin-right:20px}.quarto-layout-cell:last-child{margin-right:0}.quarto-layout-cell figure,.quarto-layout-cell>p{margin:.2em}.quarto-layout-cell img{max-width:100%}.quarto-layout-cell .html-widget{width:100% !important}.quarto-layout-cell div figure p{margin:0}.quarto-layout-cell figure{display:block;margin-inline-start:0;margin-inline-end:0}.quarto-layout-cell table{display:inline-table}.quarto-layout-cell-subref figcaption,figure .quarto-layout-row figure figcaption{text-align:center;font-style:italic}.quarto-figure{position:relative;margin-bottom:1em}.quarto-figure>figure{width:100%;margin-bottom:0}.quarto-figure-left>figure>p,.quarto-figure-left>figure>div{text-align:left}.quarto-figure-center>figure>p,.quarto-figure-center>figure>div{text-align:center}.quarto-figure-right>figure>p,.quarto-figure-right>figure>div{text-align:right}.quarto-figure>figure>div.cell-annotation,.quarto-figure>figure>div code{text-align:left}figure>p:empty{display:none}figure>p:first-child{margin-top:0;margin-bottom:0}figure>figcaption.quarto-float-caption-bottom{margin-bottom:.5em}figure>figcaption.quarto-float-caption-top{margin-top:.5em}div[id^=tbl-]{position:relative}.quarto-figure>.anchorjs-link{position:absolute;top:.6em;right:.5em}div[id^=tbl-]>.anchorjs-link{position:absolute;top:.7em;right:.3em}.quarto-figure:hover>.anchorjs-link,div[id^=tbl-]:hover>.anchorjs-link,h2:hover>.anchorjs-link,.h2:hover>.anchorjs-link,h3:hover>.anchorjs-link,.h3:hover>.anchorjs-link,h4:hover>.anchorjs-link,.h4:hover>.anchorjs-link,h5:hover>.anchorjs-link,.h5:hover>.anchorjs-link,h6:hover>.anchorjs-link,.h6:hover>.anchorjs-link,.reveal-anchorjs-link>.anchorjs-link{opacity:1}#title-block-header{margin-block-end:1rem;position:relative;margin-top:-1px}#title-block-header .abstract{margin-block-start:1rem}#title-block-header .abstract .abstract-title{font-weight:600}#title-block-header a{text-decoration:none}#title-block-header .author,#title-block-header .date,#title-block-header .doi{margin-block-end:.2rem}#title-block-header .quarto-title-block>div{display:flex}#title-block-header .quarto-title-block>div>h1,#title-block-header .quarto-title-block>div>.h1{flex-grow:1}#title-block-header .quarto-title-block>div>button{flex-shrink:0;height:2.25rem;margin-top:0}@media(min-width: 992px){#title-block-header .quarto-title-block>div>button{margin-top:5px}}tr.header>th>p:last-of-type{margin-bottom:0px}table,table.table{margin-top:.5rem;margin-bottom:.5rem}caption,.table-caption{padding-top:.5rem;padding-bottom:.5rem;text-align:center}figure.quarto-float-tbl figcaption.quarto-float-caption-top{margin-top:.5rem;margin-bottom:.25rem;text-align:center}figure.quarto-float-tbl figcaption.quarto-float-caption-bottom{padding-top:.25rem;margin-bottom:.5rem;text-align:center}.utterances{max-width:none;margin-left:-8px}iframe{margin-bottom:1em}details{margin-bottom:1em}details[show]{margin-bottom:0}details>summary{color:#6c757d}details>summary>p:only-child{display:inline}pre.sourceCode,code.sourceCode{position:relative}dd code:not(.sourceCode),p code:not(.sourceCode){white-space:pre-wrap}code{white-space:pre}@media print{code{white-space:pre-wrap}}pre>code{display:block}pre>code.sourceCode{white-space:pre}pre>code.sourceCode>span>a:first-child::before{text-decoration:none}pre.code-overflow-wrap>code.sourceCode{white-space:pre-wrap}pre.code-overflow-scroll>code.sourceCode{white-space:pre}code a:any-link{color:inherit;text-decoration:none}code a:hover{color:inherit;text-decoration:underline}ul.task-list{padding-left:1em}[data-tippy-root]{display:inline-block}.tippy-content .footnote-back{display:none}.footnote-back{margin-left:.2em}.tippy-content{overflow-x:auto}.quarto-embedded-source-code{display:none}.quarto-unresolved-ref{font-weight:600}.quarto-cover-image{max-width:35%;float:right;margin-left:30px}.cell-output-display .widget-subarea{margin-bottom:1em}.cell-output-display:not(.no-overflow-x),.knitsql-table:not(.no-overflow-x){overflow-x:auto}.panel-input{margin-bottom:1em}.panel-input>div,.panel-input>div>div{display:inline-block;vertical-align:top;padding-right:12px}.panel-input>p:last-child{margin-bottom:0}.layout-sidebar{margin-bottom:1em}.layout-sidebar .tab-content{border:none}.tab-content>.page-columns.active{display:grid}div.sourceCode>iframe{width:100%;height:300px;margin-bottom:-0.5em}a{text-underline-offset:3px}.callout pre.sourceCode{padding-left:0}div.ansi-escaped-output{font-family:monospace;display:block}/*! +* +* ansi colors from IPython notebook's +* +* we also add `bright-[color]-` synonyms for the `-[color]-intense` classes since +* that seems to be what ansi_up emits +* +*/.ansi-black-fg{color:#3e424d}.ansi-black-bg{background-color:#3e424d}.ansi-black-intense-black,.ansi-bright-black-fg{color:#282c36}.ansi-black-intense-black,.ansi-bright-black-bg{background-color:#282c36}.ansi-red-fg{color:#e75c58}.ansi-red-bg{background-color:#e75c58}.ansi-red-intense-red,.ansi-bright-red-fg{color:#b22b31}.ansi-red-intense-red,.ansi-bright-red-bg{background-color:#b22b31}.ansi-green-fg{color:#00a250}.ansi-green-bg{background-color:#00a250}.ansi-green-intense-green,.ansi-bright-green-fg{color:#007427}.ansi-green-intense-green,.ansi-bright-green-bg{background-color:#007427}.ansi-yellow-fg{color:#ddb62b}.ansi-yellow-bg{background-color:#ddb62b}.ansi-yellow-intense-yellow,.ansi-bright-yellow-fg{color:#b27d12}.ansi-yellow-intense-yellow,.ansi-bright-yellow-bg{background-color:#b27d12}.ansi-blue-fg{color:#208ffb}.ansi-blue-bg{background-color:#208ffb}.ansi-blue-intense-blue,.ansi-bright-blue-fg{color:#0065ca}.ansi-blue-intense-blue,.ansi-bright-blue-bg{background-color:#0065ca}.ansi-magenta-fg{color:#d160c4}.ansi-magenta-bg{background-color:#d160c4}.ansi-magenta-intense-magenta,.ansi-bright-magenta-fg{color:#a03196}.ansi-magenta-intense-magenta,.ansi-bright-magenta-bg{background-color:#a03196}.ansi-cyan-fg{color:#60c6c8}.ansi-cyan-bg{background-color:#60c6c8}.ansi-cyan-intense-cyan,.ansi-bright-cyan-fg{color:#258f8f}.ansi-cyan-intense-cyan,.ansi-bright-cyan-bg{background-color:#258f8f}.ansi-white-fg{color:#c5c1b4}.ansi-white-bg{background-color:#c5c1b4}.ansi-white-intense-white,.ansi-bright-white-fg{color:#a1a6b2}.ansi-white-intense-white,.ansi-bright-white-bg{background-color:#a1a6b2}.ansi-default-inverse-fg{color:#fff}.ansi-default-inverse-bg{background-color:#000}.ansi-bold{font-weight:bold}.ansi-underline{text-decoration:underline}:root{--quarto-body-bg: #FFFFFF;--quarto-body-color: #000000;--quarto-text-muted: #6c757d;--quarto-border-color: #dee2e6;--quarto-border-width: 1px}table.gt_table{color:var(--quarto-body-color);font-size:1em;width:100%;background-color:rgba(0,0,0,0);border-top-width:inherit;border-bottom-width:inherit;border-color:var(--quarto-border-color)}table.gt_table th.gt_column_spanner_outer{color:var(--quarto-body-color);background-color:rgba(0,0,0,0);border-top-width:inherit;border-bottom-width:inherit;border-color:var(--quarto-border-color)}table.gt_table th.gt_col_heading{color:var(--quarto-body-color);font-weight:bold;background-color:rgba(0,0,0,0)}table.gt_table thead.gt_col_headings{border-bottom:1px solid currentColor;border-top-width:inherit;border-top-color:var(--quarto-border-color)}table.gt_table thead.gt_col_headings:not(:first-child){border-top-width:1px;border-top-color:var(--quarto-border-color)}table.gt_table td.gt_row{border-bottom-width:1px;border-bottom-color:var(--quarto-border-color);border-top-width:0px}table.gt_table tbody.gt_table_body{border-top-width:1px;border-bottom-width:1px;border-bottom-color:var(--quarto-border-color);border-top-color:currentColor}div.columns{display:initial;gap:initial}div.column{display:inline-block;overflow-x:initial;vertical-align:top;width:50%}.code-annotation-tip-content{word-wrap:break-word}.code-annotation-container-hidden{display:none !important}dl.code-annotation-container-grid{display:grid;grid-template-columns:min-content auto}dl.code-annotation-container-grid dt{grid-column:1}dl.code-annotation-container-grid dd{grid-column:2}pre.sourceCode.code-annotation-code{padding-right:0}code.sourceCode .code-annotation-anchor{z-index:100;position:relative;float:right;background-color:rgba(0,0,0,0)}input[type=checkbox]{margin-right:.5ch}:root{--mermaid-bg-color: #FFFFFF;--mermaid-edge-color: #343a40;--mermaid-node-fg-color: #000000;--mermaid-fg-color: #000000;--mermaid-fg-color--lighter: #1a1a1a;--mermaid-fg-color--lightest: #333333;--mermaid-font-family: Source Sans Pro, -apple-system, BlinkMacSystemFont, Segoe UI, Roboto, Helvetica Neue, Arial, sans-serif, Apple Color Emoji, Segoe UI Emoji, Segoe UI Symbol;--mermaid-label-bg-color: #FFFFFF;--mermaid-label-fg-color: #E8E8FC;--mermaid-node-bg-color: rgba(232, 232, 252, 0.1);--mermaid-node-fg-color: #000000}@media print{:root{font-size:11pt}#quarto-sidebar,#TOC,.nav-page{display:none}.page-columns .content{grid-column-start:page-start}.fixed-top{position:relative}.panel-caption,.figure-caption,figcaption{color:#666}}.code-copy-button{position:absolute;top:0;right:0;border:0;margin-top:5px;margin-right:5px;background-color:rgba(0,0,0,0);z-index:3}.code-copy-button:focus{outline:none}.code-copy-button-tooltip{font-size:.75em}pre.sourceCode:hover>.code-copy-button>.bi::before{display:inline-block;height:1rem;width:1rem;content:"";vertical-align:-0.125em;background-image:url('data:image/svg+xml,');background-repeat:no-repeat;background-size:1rem 1rem}pre.sourceCode:hover>.code-copy-button-checked>.bi::before{background-image:url('data:image/svg+xml,')}pre.sourceCode:hover>.code-copy-button:hover>.bi::before{background-image:url('data:image/svg+xml,')}pre.sourceCode:hover>.code-copy-button-checked:hover>.bi::before{background-image:url('data:image/svg+xml,')}main ol ol,main ul ul,main ol ul,main ul ol{margin-bottom:1em}ul>li:not(:has(>p))>ul,ol>li:not(:has(>p))>ul,ul>li:not(:has(>p))>ol,ol>li:not(:has(>p))>ol{margin-bottom:0}ul>li:not(:has(>p))>ul>li:has(>p),ol>li:not(:has(>p))>ul>li:has(>p),ul>li:not(:has(>p))>ol>li:has(>p),ol>li:not(:has(>p))>ol>li:has(>p){margin-top:1rem}body{margin:0}main.page-columns>header>h1.title,main.page-columns>header>.title.h1{margin-bottom:0}@media(min-width: 992px){body .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start page-start-inset] 35px [body-start-outset] 35px [body-start] 1.5em [body-content-start] minmax(500px, calc(850px - 3em)) [body-content-end] 1.5em [body-end] 35px [body-end-outset] minmax(75px, 145px) [page-end-inset] 35px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.fullcontent:not(.floating):not(.docked) .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start page-start-inset] 35px [body-start-outset] 35px [body-start] 1.5em [body-content-start] minmax(500px, calc(850px - 3em)) [body-content-end] 1.5em [body-end] 35px [body-end-outset] 35px [page-end-inset page-end] 5fr [screen-end-inset] 1.5em}body.slimcontent:not(.floating):not(.docked) .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start page-start-inset] 35px [body-start-outset] 35px [body-start] 1.5em [body-content-start] minmax(500px, calc(850px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(0px, 200px) [page-end-inset] 35px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.listing:not(.floating):not(.docked) .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start] minmax(50px, 100px) [page-start-inset] 50px [body-start-outset] 50px [body-start] 1.5em [body-content-start] minmax(500px, calc(850px - 3em)) [body-content-end] 3em [body-end] 50px [body-end-outset] minmax(0px, 250px) [page-end-inset] minmax(50px, 100px) [page-end] 1fr [screen-end-inset] 1.5em [screen-end]}body:not(.floating):not(.docked) .page-columns.toc-left{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start] 35px [page-start-inset] minmax(0px, 175px) [body-start-outset] 35px [body-start] 1.5em [body-content-start] minmax(450px, calc(800px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(0px, 200px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body:not(.floating):not(.docked) .page-columns.toc-left .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start] 35px [page-start-inset] minmax(0px, 175px) [body-start-outset] 35px [body-start] 1.5em [body-content-start] minmax(450px, calc(800px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(0px, 200px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.floating .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start] minmax(25px, 50px) [page-start-inset] minmax(50px, 150px) [body-start-outset] minmax(25px, 50px) [body-start] 1.5em [body-content-start] minmax(500px, calc(800px - 3em)) [body-content-end] 1.5em [body-end] minmax(25px, 50px) [body-end-outset] minmax(50px, 150px) [page-end-inset] minmax(25px, 50px) [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.docked .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start] minmax(50px, 100px) [page-start-inset] 50px [body-start-outset] 50px [body-start] 1.5em [body-content-start] minmax(500px, calc(1000px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(50px, 100px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.docked.fullcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start] minmax(50px, 100px) [page-start-inset] 50px [body-start-outset] 50px [body-start] 1.5em [body-content-start] minmax(500px, calc(1000px - 3em)) [body-content-end] 1.5em [body-end body-end-outset page-end-inset page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.floating.fullcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start] 50px [page-start-inset] minmax(50px, 150px) [body-start-outset] 50px [body-start] 1.5em [body-content-start] minmax(500px, calc(800px - 3em)) [body-content-end] 1.5em [body-end body-end-outset page-end-inset page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.docked.slimcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start] minmax(50px, 100px) [page-start-inset] 50px [body-start-outset] 50px [body-start] 1.5em [body-content-start] minmax(450px, calc(750px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(0px, 200px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.docked.listing .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start] minmax(50px, 100px) [page-start-inset] 50px [body-start-outset] 50px [body-start] 1.5em [body-content-start] minmax(500px, calc(1000px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(0px, 200px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.floating.slimcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start] 50px [page-start-inset] minmax(50px, 150px) [body-start-outset] 50px [body-start] 1.5em [body-content-start] minmax(450px, calc(750px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(50px, 150px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.floating.listing .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start] minmax(25px, 50px) [page-start-inset] minmax(50px, 150px) [body-start-outset] minmax(25px, 50px) [body-start] 1.5em [body-content-start] minmax(500px, calc(800px - 3em)) [body-content-end] 1.5em [body-end] minmax(25px, 50px) [body-end-outset] minmax(50px, 150px) [page-end-inset] minmax(25px, 50px) [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}}@media(max-width: 991.98px){body .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset] 5fr [body-start] 1.5em [body-content-start] minmax(500px, calc(800px - 3em)) [body-content-end] 1.5em [body-end] 35px [body-end-outset] minmax(75px, 145px) [page-end-inset] 35px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.fullcontent:not(.floating):not(.docked) .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset] 5fr [body-start] 1.5em [body-content-start] minmax(500px, calc(800px - 3em)) [body-content-end] 1.5em [body-end body-end-outset page-end-inset page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.slimcontent:not(.floating):not(.docked) .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset] 5fr [body-start] 1.5em [body-content-start] minmax(500px, calc(800px - 3em)) [body-content-end] 1.5em [body-end] 35px [body-end-outset] minmax(75px, 145px) [page-end-inset] 35px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.listing:not(.floating):not(.docked) .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset] 5fr [body-start] 1.5em [body-content-start] minmax(500px, calc(1250px - 3em)) [body-content-end body-end body-end-outset page-end-inset page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body:not(.floating):not(.docked) .page-columns.toc-left{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start] 35px [page-start-inset] minmax(0px, 145px) [body-start-outset] 35px [body-start] 1.5em [body-content-start] minmax(450px, calc(800px - 3em)) [body-content-end] 1.5em [body-end body-end-outset page-end-inset page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body:not(.floating):not(.docked) .page-columns.toc-left .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start] 35px [page-start-inset] minmax(0px, 145px) [body-start-outset] 35px [body-start] 1.5em [body-content-start] minmax(450px, calc(800px - 3em)) [body-content-end] 1.5em [body-end body-end-outset page-end-inset page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.floating .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start page-start-inset body-start-outset body-start] 1.5em [body-content-start] minmax(500px, calc(750px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(75px, 150px) [page-end-inset] 25px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.docked .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset body-start body-content-start] minmax(500px, calc(750px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(25px, 50px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.docked.fullcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset body-start body-content-start] minmax(500px, calc(1000px - 3em)) [body-content-end] 1.5em [body-end body-end-outset page-end-inset page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.floating.fullcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start page-start-inset body-start-outset body-start] 1em [body-content-start] minmax(500px, calc(800px - 3em)) [body-content-end] 1.5em [body-end body-end-outset page-end-inset page-end] 4fr [screen-end-inset] 1.5em [screen-end]}body.docked.slimcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset body-start body-content-start] minmax(500px, calc(750px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(25px, 50px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.docked.listing .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset body-start body-content-start] minmax(500px, calc(750px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(25px, 50px) [page-end-inset] 50px [page-end] 5fr [screen-end-inset] 1.5em [screen-end]}body.floating.slimcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start page-start-inset body-start-outset body-start] 1em [body-content-start] minmax(500px, calc(750px - 3em)) [body-content-end] 1.5em [body-end] 35px [body-end-outset] minmax(75px, 145px) [page-end-inset] 35px [page-end] 4fr [screen-end-inset] 1.5em [screen-end]}body.floating.listing .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset] 5fr [page-start page-start-inset body-start-outset body-start] 1em [body-content-start] minmax(500px, calc(750px - 3em)) [body-content-end] 1.5em [body-end] 50px [body-end-outset] minmax(75px, 150px) [page-end-inset] 25px [page-end] 4fr [screen-end-inset] 1.5em [screen-end]}}@media(max-width: 767.98px){body .page-columns,body.fullcontent:not(.floating):not(.docked) .page-columns,body.slimcontent:not(.floating):not(.docked) .page-columns,body.docked .page-columns,body.docked.slimcontent .page-columns,body.docked.fullcontent .page-columns,body.floating .page-columns,body.floating.slimcontent .page-columns,body.floating.fullcontent .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset body-start body-content-start] minmax(0px, 1fr) [body-content-end body-end body-end-outset page-end-inset page-end screen-end-inset] 1.5em [screen-end]}body:not(.floating):not(.docked) .page-columns.toc-left{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset body-start body-content-start] minmax(0px, 1fr) [body-content-end body-end body-end-outset page-end-inset page-end screen-end-inset] 1.5em [screen-end]}body:not(.floating):not(.docked) .page-columns.toc-left .page-columns{display:grid;gap:0;grid-template-columns:[screen-start] 1.5em [screen-start-inset page-start page-start-inset body-start-outset body-start body-content-start] minmax(0px, 1fr) [body-content-end body-end body-end-outset page-end-inset page-end screen-end-inset] 1.5em [screen-end]}nav[role=doc-toc]{display:none}}body,.page-row-navigation{grid-template-rows:[page-top] max-content [contents-top] max-content [contents-bottom] max-content [page-bottom]}.page-rows-contents{grid-template-rows:[content-top] minmax(max-content, 1fr) [content-bottom] minmax(60px, max-content) [page-bottom]}.page-full{grid-column:screen-start/screen-end !important}.page-columns>*{grid-column:body-content-start/body-content-end}.page-columns.column-page>*{grid-column:page-start/page-end}.page-columns.column-page-left .page-columns.page-full>*,.page-columns.column-page-left>*{grid-column:page-start/body-content-end}.page-columns.column-page-right .page-columns.page-full>*,.page-columns.column-page-right>*{grid-column:body-content-start/page-end}.page-rows{grid-auto-rows:auto}.header{grid-column:screen-start/screen-end;grid-row:page-top/contents-top}#quarto-content{padding:0;grid-column:screen-start/screen-end;grid-row:contents-top/contents-bottom}body.floating .sidebar.sidebar-navigation{grid-column:page-start/body-start;grid-row:content-top/page-bottom}body.docked .sidebar.sidebar-navigation{grid-column:screen-start/body-start;grid-row:content-top/page-bottom}.sidebar.toc-left{grid-column:page-start/body-start;grid-row:content-top/page-bottom}.sidebar.margin-sidebar{grid-column:body-end/page-end;grid-row:content-top/page-bottom}.page-columns .content{grid-column:body-content-start/body-content-end;grid-row:content-top/content-bottom;align-content:flex-start}.page-columns .page-navigation{grid-column:body-content-start/body-content-end;grid-row:content-bottom/page-bottom}.page-columns .footer{grid-column:screen-start/screen-end;grid-row:contents-bottom/page-bottom}.page-columns .column-body{grid-column:body-content-start/body-content-end}.page-columns .column-body-fullbleed{grid-column:body-start/body-end}.page-columns .column-body-outset{grid-column:body-start-outset/body-end-outset;z-index:998;opacity:.999}.page-columns .column-body-outset table{background:#fff}.page-columns .column-body-outset-left{grid-column:body-start-outset/body-content-end;z-index:998;opacity:.999}.page-columns .column-body-outset-left table{background:#fff}.page-columns .column-body-outset-right{grid-column:body-content-start/body-end-outset;z-index:998;opacity:.999}.page-columns .column-body-outset-right table{background:#fff}.page-columns .column-page{grid-column:page-start/page-end;z-index:998;opacity:.999}.page-columns .column-page table{background:#fff}.page-columns .column-page-inset{grid-column:page-start-inset/page-end-inset;z-index:998;opacity:.999}.page-columns .column-page-inset table{background:#fff}.page-columns .column-page-inset-left{grid-column:page-start-inset/body-content-end;z-index:998;opacity:.999}.page-columns .column-page-inset-left table{background:#fff}.page-columns .column-page-inset-right{grid-column:body-content-start/page-end-inset;z-index:998;opacity:.999}.page-columns .column-page-inset-right figcaption table{background:#fff}.page-columns .column-page-left{grid-column:page-start/body-content-end;z-index:998;opacity:.999}.page-columns .column-page-left table{background:#fff}.page-columns .column-page-right{grid-column:body-content-start/page-end;z-index:998;opacity:.999}.page-columns .column-page-right figcaption table{background:#fff}#quarto-content.page-columns #quarto-margin-sidebar,#quarto-content.page-columns #quarto-sidebar{z-index:1}@media(max-width: 991.98px){#quarto-content.page-columns #quarto-margin-sidebar.collapse,#quarto-content.page-columns #quarto-sidebar.collapse,#quarto-content.page-columns #quarto-margin-sidebar.collapsing,#quarto-content.page-columns #quarto-sidebar.collapsing{z-index:1055}}#quarto-content.page-columns main.column-page,#quarto-content.page-columns main.column-page-right,#quarto-content.page-columns main.column-page-left{z-index:0}.page-columns .column-screen-inset{grid-column:screen-start-inset/screen-end-inset;z-index:998;opacity:.999}.page-columns .column-screen-inset table{background:#fff}.page-columns .column-screen-inset-left{grid-column:screen-start-inset/body-content-end;z-index:998;opacity:.999}.page-columns .column-screen-inset-left table{background:#fff}.page-columns .column-screen-inset-right{grid-column:body-content-start/screen-end-inset;z-index:998;opacity:.999}.page-columns .column-screen-inset-right table{background:#fff}.page-columns .column-screen{grid-column:screen-start/screen-end;z-index:998;opacity:.999}.page-columns .column-screen table{background:#fff}.page-columns .column-screen-left{grid-column:screen-start/body-content-end;z-index:998;opacity:.999}.page-columns .column-screen-left table{background:#fff}.page-columns .column-screen-right{grid-column:body-content-start/screen-end;z-index:998;opacity:.999}.page-columns .column-screen-right table{background:#fff}.page-columns .column-screen-inset-shaded{grid-column:screen-start/screen-end;padding:1em;background:#f8f9fa;z-index:998;opacity:.999;margin-bottom:1em}.zindex-content{z-index:998;opacity:.999}.zindex-modal{z-index:1055;opacity:.999}.zindex-over-content{z-index:999;opacity:.999}img.img-fluid.column-screen,img.img-fluid.column-screen-inset-shaded,img.img-fluid.column-screen-inset,img.img-fluid.column-screen-inset-left,img.img-fluid.column-screen-inset-right,img.img-fluid.column-screen-left,img.img-fluid.column-screen-right{width:100%}@media(min-width: 992px){.margin-caption,div.aside,aside:not(.footnotes):not(.sidebar),.column-margin{grid-column:body-end/page-end !important;z-index:998}.column-sidebar{grid-column:page-start/body-start !important;z-index:998}.column-leftmargin{grid-column:screen-start-inset/body-start !important;z-index:998}.no-row-height{height:1em;overflow:visible}}@media(max-width: 991.98px){.margin-caption,div.aside,aside:not(.footnotes):not(.sidebar),.column-margin{grid-column:body-end/page-end !important;z-index:998}.no-row-height{height:1em;overflow:visible}.page-columns.page-full{overflow:visible}.page-columns.toc-left .margin-caption,.page-columns.toc-left div.aside,.page-columns.toc-left aside:not(.footnotes):not(.sidebar),.page-columns.toc-left .column-margin{grid-column:body-content-start/body-content-end !important;z-index:998;opacity:.999}.page-columns.toc-left .no-row-height{height:initial;overflow:initial}}@media(max-width: 767.98px){.margin-caption,div.aside,aside:not(.footnotes):not(.sidebar),.column-margin{grid-column:body-content-start/body-content-end !important;z-index:998;opacity:.999}.no-row-height{height:initial;overflow:initial}#quarto-margin-sidebar{display:none}#quarto-sidebar-toc-left{display:none}.hidden-sm{display:none}}.panel-grid{display:grid;grid-template-rows:repeat(1, 1fr);grid-template-columns:repeat(24, 1fr);gap:1em}.panel-grid .g-col-1{grid-column:auto/span 1}.panel-grid .g-col-2{grid-column:auto/span 2}.panel-grid .g-col-3{grid-column:auto/span 3}.panel-grid .g-col-4{grid-column:auto/span 4}.panel-grid .g-col-5{grid-column:auto/span 5}.panel-grid .g-col-6{grid-column:auto/span 6}.panel-grid .g-col-7{grid-column:auto/span 7}.panel-grid .g-col-8{grid-column:auto/span 8}.panel-grid .g-col-9{grid-column:auto/span 9}.panel-grid .g-col-10{grid-column:auto/span 10}.panel-grid .g-col-11{grid-column:auto/span 11}.panel-grid .g-col-12{grid-column:auto/span 12}.panel-grid .g-col-13{grid-column:auto/span 13}.panel-grid .g-col-14{grid-column:auto/span 14}.panel-grid .g-col-15{grid-column:auto/span 15}.panel-grid .g-col-16{grid-column:auto/span 16}.panel-grid .g-col-17{grid-column:auto/span 17}.panel-grid .g-col-18{grid-column:auto/span 18}.panel-grid .g-col-19{grid-column:auto/span 19}.panel-grid .g-col-20{grid-column:auto/span 20}.panel-grid .g-col-21{grid-column:auto/span 21}.panel-grid .g-col-22{grid-column:auto/span 22}.panel-grid .g-col-23{grid-column:auto/span 23}.panel-grid .g-col-24{grid-column:auto/span 24}.panel-grid .g-start-1{grid-column-start:1}.panel-grid .g-start-2{grid-column-start:2}.panel-grid .g-start-3{grid-column-start:3}.panel-grid .g-start-4{grid-column-start:4}.panel-grid .g-start-5{grid-column-start:5}.panel-grid .g-start-6{grid-column-start:6}.panel-grid .g-start-7{grid-column-start:7}.panel-grid .g-start-8{grid-column-start:8}.panel-grid .g-start-9{grid-column-start:9}.panel-grid .g-start-10{grid-column-start:10}.panel-grid .g-start-11{grid-column-start:11}.panel-grid .g-start-12{grid-column-start:12}.panel-grid .g-start-13{grid-column-start:13}.panel-grid .g-start-14{grid-column-start:14}.panel-grid .g-start-15{grid-column-start:15}.panel-grid .g-start-16{grid-column-start:16}.panel-grid .g-start-17{grid-column-start:17}.panel-grid .g-start-18{grid-column-start:18}.panel-grid .g-start-19{grid-column-start:19}.panel-grid .g-start-20{grid-column-start:20}.panel-grid .g-start-21{grid-column-start:21}.panel-grid .g-start-22{grid-column-start:22}.panel-grid .g-start-23{grid-column-start:23}@media(min-width: 576px){.panel-grid .g-col-sm-1{grid-column:auto/span 1}.panel-grid .g-col-sm-2{grid-column:auto/span 2}.panel-grid .g-col-sm-3{grid-column:auto/span 3}.panel-grid .g-col-sm-4{grid-column:auto/span 4}.panel-grid .g-col-sm-5{grid-column:auto/span 5}.panel-grid .g-col-sm-6{grid-column:auto/span 6}.panel-grid .g-col-sm-7{grid-column:auto/span 7}.panel-grid .g-col-sm-8{grid-column:auto/span 8}.panel-grid .g-col-sm-9{grid-column:auto/span 9}.panel-grid .g-col-sm-10{grid-column:auto/span 10}.panel-grid .g-col-sm-11{grid-column:auto/span 11}.panel-grid .g-col-sm-12{grid-column:auto/span 12}.panel-grid .g-col-sm-13{grid-column:auto/span 13}.panel-grid .g-col-sm-14{grid-column:auto/span 14}.panel-grid .g-col-sm-15{grid-column:auto/span 15}.panel-grid .g-col-sm-16{grid-column:auto/span 16}.panel-grid .g-col-sm-17{grid-column:auto/span 17}.panel-grid .g-col-sm-18{grid-column:auto/span 18}.panel-grid .g-col-sm-19{grid-column:auto/span 19}.panel-grid .g-col-sm-20{grid-column:auto/span 20}.panel-grid .g-col-sm-21{grid-column:auto/span 21}.panel-grid .g-col-sm-22{grid-column:auto/span 22}.panel-grid .g-col-sm-23{grid-column:auto/span 23}.panel-grid .g-col-sm-24{grid-column:auto/span 24}.panel-grid .g-start-sm-1{grid-column-start:1}.panel-grid .g-start-sm-2{grid-column-start:2}.panel-grid .g-start-sm-3{grid-column-start:3}.panel-grid .g-start-sm-4{grid-column-start:4}.panel-grid .g-start-sm-5{grid-column-start:5}.panel-grid .g-start-sm-6{grid-column-start:6}.panel-grid .g-start-sm-7{grid-column-start:7}.panel-grid .g-start-sm-8{grid-column-start:8}.panel-grid .g-start-sm-9{grid-column-start:9}.panel-grid .g-start-sm-10{grid-column-start:10}.panel-grid .g-start-sm-11{grid-column-start:11}.panel-grid .g-start-sm-12{grid-column-start:12}.panel-grid .g-start-sm-13{grid-column-start:13}.panel-grid .g-start-sm-14{grid-column-start:14}.panel-grid .g-start-sm-15{grid-column-start:15}.panel-grid .g-start-sm-16{grid-column-start:16}.panel-grid .g-start-sm-17{grid-column-start:17}.panel-grid .g-start-sm-18{grid-column-start:18}.panel-grid .g-start-sm-19{grid-column-start:19}.panel-grid .g-start-sm-20{grid-column-start:20}.panel-grid .g-start-sm-21{grid-column-start:21}.panel-grid .g-start-sm-22{grid-column-start:22}.panel-grid .g-start-sm-23{grid-column-start:23}}@media(min-width: 768px){.panel-grid .g-col-md-1{grid-column:auto/span 1}.panel-grid .g-col-md-2{grid-column:auto/span 2}.panel-grid .g-col-md-3{grid-column:auto/span 3}.panel-grid .g-col-md-4{grid-column:auto/span 4}.panel-grid .g-col-md-5{grid-column:auto/span 5}.panel-grid .g-col-md-6{grid-column:auto/span 6}.panel-grid .g-col-md-7{grid-column:auto/span 7}.panel-grid .g-col-md-8{grid-column:auto/span 8}.panel-grid .g-col-md-9{grid-column:auto/span 9}.panel-grid .g-col-md-10{grid-column:auto/span 10}.panel-grid .g-col-md-11{grid-column:auto/span 11}.panel-grid .g-col-md-12{grid-column:auto/span 12}.panel-grid .g-col-md-13{grid-column:auto/span 13}.panel-grid .g-col-md-14{grid-column:auto/span 14}.panel-grid .g-col-md-15{grid-column:auto/span 15}.panel-grid .g-col-md-16{grid-column:auto/span 16}.panel-grid .g-col-md-17{grid-column:auto/span 17}.panel-grid .g-col-md-18{grid-column:auto/span 18}.panel-grid .g-col-md-19{grid-column:auto/span 19}.panel-grid .g-col-md-20{grid-column:auto/span 20}.panel-grid .g-col-md-21{grid-column:auto/span 21}.panel-grid .g-col-md-22{grid-column:auto/span 22}.panel-grid .g-col-md-23{grid-column:auto/span 23}.panel-grid .g-col-md-24{grid-column:auto/span 24}.panel-grid .g-start-md-1{grid-column-start:1}.panel-grid .g-start-md-2{grid-column-start:2}.panel-grid .g-start-md-3{grid-column-start:3}.panel-grid .g-start-md-4{grid-column-start:4}.panel-grid .g-start-md-5{grid-column-start:5}.panel-grid .g-start-md-6{grid-column-start:6}.panel-grid .g-start-md-7{grid-column-start:7}.panel-grid .g-start-md-8{grid-column-start:8}.panel-grid .g-start-md-9{grid-column-start:9}.panel-grid .g-start-md-10{grid-column-start:10}.panel-grid .g-start-md-11{grid-column-start:11}.panel-grid .g-start-md-12{grid-column-start:12}.panel-grid .g-start-md-13{grid-column-start:13}.panel-grid .g-start-md-14{grid-column-start:14}.panel-grid .g-start-md-15{grid-column-start:15}.panel-grid .g-start-md-16{grid-column-start:16}.panel-grid .g-start-md-17{grid-column-start:17}.panel-grid .g-start-md-18{grid-column-start:18}.panel-grid .g-start-md-19{grid-column-start:19}.panel-grid .g-start-md-20{grid-column-start:20}.panel-grid .g-start-md-21{grid-column-start:21}.panel-grid .g-start-md-22{grid-column-start:22}.panel-grid .g-start-md-23{grid-column-start:23}}@media(min-width: 992px){.panel-grid .g-col-lg-1{grid-column:auto/span 1}.panel-grid .g-col-lg-2{grid-column:auto/span 2}.panel-grid .g-col-lg-3{grid-column:auto/span 3}.panel-grid .g-col-lg-4{grid-column:auto/span 4}.panel-grid .g-col-lg-5{grid-column:auto/span 5}.panel-grid .g-col-lg-6{grid-column:auto/span 6}.panel-grid .g-col-lg-7{grid-column:auto/span 7}.panel-grid .g-col-lg-8{grid-column:auto/span 8}.panel-grid .g-col-lg-9{grid-column:auto/span 9}.panel-grid .g-col-lg-10{grid-column:auto/span 10}.panel-grid .g-col-lg-11{grid-column:auto/span 11}.panel-grid .g-col-lg-12{grid-column:auto/span 12}.panel-grid .g-col-lg-13{grid-column:auto/span 13}.panel-grid .g-col-lg-14{grid-column:auto/span 14}.panel-grid .g-col-lg-15{grid-column:auto/span 15}.panel-grid .g-col-lg-16{grid-column:auto/span 16}.panel-grid .g-col-lg-17{grid-column:auto/span 17}.panel-grid .g-col-lg-18{grid-column:auto/span 18}.panel-grid .g-col-lg-19{grid-column:auto/span 19}.panel-grid .g-col-lg-20{grid-column:auto/span 20}.panel-grid .g-col-lg-21{grid-column:auto/span 21}.panel-grid .g-col-lg-22{grid-column:auto/span 22}.panel-grid .g-col-lg-23{grid-column:auto/span 23}.panel-grid .g-col-lg-24{grid-column:auto/span 24}.panel-grid .g-start-lg-1{grid-column-start:1}.panel-grid .g-start-lg-2{grid-column-start:2}.panel-grid .g-start-lg-3{grid-column-start:3}.panel-grid .g-start-lg-4{grid-column-start:4}.panel-grid .g-start-lg-5{grid-column-start:5}.panel-grid .g-start-lg-6{grid-column-start:6}.panel-grid .g-start-lg-7{grid-column-start:7}.panel-grid .g-start-lg-8{grid-column-start:8}.panel-grid .g-start-lg-9{grid-column-start:9}.panel-grid .g-start-lg-10{grid-column-start:10}.panel-grid .g-start-lg-11{grid-column-start:11}.panel-grid .g-start-lg-12{grid-column-start:12}.panel-grid .g-start-lg-13{grid-column-start:13}.panel-grid .g-start-lg-14{grid-column-start:14}.panel-grid .g-start-lg-15{grid-column-start:15}.panel-grid .g-start-lg-16{grid-column-start:16}.panel-grid .g-start-lg-17{grid-column-start:17}.panel-grid .g-start-lg-18{grid-column-start:18}.panel-grid .g-start-lg-19{grid-column-start:19}.panel-grid .g-start-lg-20{grid-column-start:20}.panel-grid .g-start-lg-21{grid-column-start:21}.panel-grid .g-start-lg-22{grid-column-start:22}.panel-grid .g-start-lg-23{grid-column-start:23}}@media(min-width: 1200px){.panel-grid .g-col-xl-1{grid-column:auto/span 1}.panel-grid .g-col-xl-2{grid-column:auto/span 2}.panel-grid .g-col-xl-3{grid-column:auto/span 3}.panel-grid .g-col-xl-4{grid-column:auto/span 4}.panel-grid .g-col-xl-5{grid-column:auto/span 5}.panel-grid .g-col-xl-6{grid-column:auto/span 6}.panel-grid .g-col-xl-7{grid-column:auto/span 7}.panel-grid .g-col-xl-8{grid-column:auto/span 8}.panel-grid .g-col-xl-9{grid-column:auto/span 9}.panel-grid .g-col-xl-10{grid-column:auto/span 10}.panel-grid .g-col-xl-11{grid-column:auto/span 11}.panel-grid .g-col-xl-12{grid-column:auto/span 12}.panel-grid .g-col-xl-13{grid-column:auto/span 13}.panel-grid .g-col-xl-14{grid-column:auto/span 14}.panel-grid .g-col-xl-15{grid-column:auto/span 15}.panel-grid .g-col-xl-16{grid-column:auto/span 16}.panel-grid .g-col-xl-17{grid-column:auto/span 17}.panel-grid .g-col-xl-18{grid-column:auto/span 18}.panel-grid .g-col-xl-19{grid-column:auto/span 19}.panel-grid .g-col-xl-20{grid-column:auto/span 20}.panel-grid .g-col-xl-21{grid-column:auto/span 21}.panel-grid .g-col-xl-22{grid-column:auto/span 22}.panel-grid .g-col-xl-23{grid-column:auto/span 23}.panel-grid .g-col-xl-24{grid-column:auto/span 24}.panel-grid .g-start-xl-1{grid-column-start:1}.panel-grid .g-start-xl-2{grid-column-start:2}.panel-grid .g-start-xl-3{grid-column-start:3}.panel-grid .g-start-xl-4{grid-column-start:4}.panel-grid .g-start-xl-5{grid-column-start:5}.panel-grid .g-start-xl-6{grid-column-start:6}.panel-grid .g-start-xl-7{grid-column-start:7}.panel-grid .g-start-xl-8{grid-column-start:8}.panel-grid .g-start-xl-9{grid-column-start:9}.panel-grid .g-start-xl-10{grid-column-start:10}.panel-grid .g-start-xl-11{grid-column-start:11}.panel-grid .g-start-xl-12{grid-column-start:12}.panel-grid .g-start-xl-13{grid-column-start:13}.panel-grid .g-start-xl-14{grid-column-start:14}.panel-grid .g-start-xl-15{grid-column-start:15}.panel-grid .g-start-xl-16{grid-column-start:16}.panel-grid .g-start-xl-17{grid-column-start:17}.panel-grid .g-start-xl-18{grid-column-start:18}.panel-grid .g-start-xl-19{grid-column-start:19}.panel-grid .g-start-xl-20{grid-column-start:20}.panel-grid .g-start-xl-21{grid-column-start:21}.panel-grid .g-start-xl-22{grid-column-start:22}.panel-grid .g-start-xl-23{grid-column-start:23}}@media(min-width: 1400px){.panel-grid .g-col-xxl-1{grid-column:auto/span 1}.panel-grid .g-col-xxl-2{grid-column:auto/span 2}.panel-grid .g-col-xxl-3{grid-column:auto/span 3}.panel-grid .g-col-xxl-4{grid-column:auto/span 4}.panel-grid .g-col-xxl-5{grid-column:auto/span 5}.panel-grid .g-col-xxl-6{grid-column:auto/span 6}.panel-grid .g-col-xxl-7{grid-column:auto/span 7}.panel-grid .g-col-xxl-8{grid-column:auto/span 8}.panel-grid .g-col-xxl-9{grid-column:auto/span 9}.panel-grid .g-col-xxl-10{grid-column:auto/span 10}.panel-grid .g-col-xxl-11{grid-column:auto/span 11}.panel-grid .g-col-xxl-12{grid-column:auto/span 12}.panel-grid .g-col-xxl-13{grid-column:auto/span 13}.panel-grid .g-col-xxl-14{grid-column:auto/span 14}.panel-grid .g-col-xxl-15{grid-column:auto/span 15}.panel-grid .g-col-xxl-16{grid-column:auto/span 16}.panel-grid .g-col-xxl-17{grid-column:auto/span 17}.panel-grid .g-col-xxl-18{grid-column:auto/span 18}.panel-grid .g-col-xxl-19{grid-column:auto/span 19}.panel-grid .g-col-xxl-20{grid-column:auto/span 20}.panel-grid .g-col-xxl-21{grid-column:auto/span 21}.panel-grid .g-col-xxl-22{grid-column:auto/span 22}.panel-grid .g-col-xxl-23{grid-column:auto/span 23}.panel-grid .g-col-xxl-24{grid-column:auto/span 24}.panel-grid .g-start-xxl-1{grid-column-start:1}.panel-grid .g-start-xxl-2{grid-column-start:2}.panel-grid .g-start-xxl-3{grid-column-start:3}.panel-grid .g-start-xxl-4{grid-column-start:4}.panel-grid .g-start-xxl-5{grid-column-start:5}.panel-grid .g-start-xxl-6{grid-column-start:6}.panel-grid .g-start-xxl-7{grid-column-start:7}.panel-grid .g-start-xxl-8{grid-column-start:8}.panel-grid .g-start-xxl-9{grid-column-start:9}.panel-grid .g-start-xxl-10{grid-column-start:10}.panel-grid .g-start-xxl-11{grid-column-start:11}.panel-grid .g-start-xxl-12{grid-column-start:12}.panel-grid .g-start-xxl-13{grid-column-start:13}.panel-grid .g-start-xxl-14{grid-column-start:14}.panel-grid .g-start-xxl-15{grid-column-start:15}.panel-grid .g-start-xxl-16{grid-column-start:16}.panel-grid .g-start-xxl-17{grid-column-start:17}.panel-grid .g-start-xxl-18{grid-column-start:18}.panel-grid .g-start-xxl-19{grid-column-start:19}.panel-grid .g-start-xxl-20{grid-column-start:20}.panel-grid .g-start-xxl-21{grid-column-start:21}.panel-grid .g-start-xxl-22{grid-column-start:22}.panel-grid .g-start-xxl-23{grid-column-start:23}}main{margin-top:1em;margin-bottom:1em}h1,.h1,h2,.h2{color:inherit;margin-top:2rem;margin-bottom:1rem;font-weight:600}h1.title,.title.h1{margin-top:0}main.content>section:first-of-type>h2:first-child,main.content>section:first-of-type>.h2:first-child{margin-top:0}h2,.h2{border-bottom:1px solid #dee2e6;padding-bottom:.5rem}h3,.h3{font-weight:600}h3,.h3,h4,.h4{opacity:.9;margin-top:1.5rem}h5,.h5,h6,.h6{opacity:.9}.header-section-number{color:#404040}.nav-link.active .header-section-number{color:inherit}mark,.mark{padding:0em}.panel-caption,.figure-caption,.subfigure-caption,.table-caption,figcaption,caption{font-size:.9rem;color:#404040}.quarto-layout-cell[data-ref-parent] caption{color:#404040}.column-margin figcaption,.margin-caption,div.aside,aside,.column-margin{color:#404040;font-size:.825rem}.panel-caption.margin-caption{text-align:inherit}.column-margin.column-container p{margin-bottom:0}.column-margin.column-container>*:not(.collapse):first-child{padding-bottom:.5em;display:block}.column-margin.column-container>*:not(.collapse):not(:first-child){padding-top:.5em;padding-bottom:.5em;display:block}.column-margin.column-container>*.collapse:not(.show){display:none}@media(min-width: 768px){.column-margin.column-container .callout-margin-content:first-child{margin-top:4.5em}.column-margin.column-container .callout-margin-content-simple:first-child{margin-top:3.5em}}.margin-caption>*{padding-top:.5em;padding-bottom:.5em}@media(max-width: 767.98px){.quarto-layout-row{flex-direction:column}}.nav-tabs .nav-item{margin-top:1px;cursor:pointer}.tab-content{margin-top:0px;border-left:#dee2e6 1px solid;border-right:#dee2e6 1px solid;border-bottom:#dee2e6 1px solid;margin-left:0;padding:1em;margin-bottom:1em}@media(max-width: 767.98px){.layout-sidebar{margin-left:0;margin-right:0}}.panel-sidebar,.panel-sidebar .form-control,.panel-input,.panel-input .form-control,.selectize-dropdown{font-size:.9rem}.panel-sidebar .form-control,.panel-input .form-control{padding-top:.1rem}.tab-pane div.sourceCode{margin-top:0px}.tab-pane>p{padding-top:0}.tab-pane>p:nth-child(1){padding-top:0}.tab-pane>p:last-child{margin-bottom:0}.tab-pane>pre:last-child{margin-bottom:0}.tab-content>.tab-pane:not(.active){display:none !important}div.sourceCode{background-color:rgba(233,236,239,.65);border:1px solid rgba(233,236,239,.65)}pre.sourceCode{background-color:rgba(0,0,0,0)}pre.sourceCode{border:none;font-size:.875em;overflow:visible !important;padding:.4em}div.sourceCode{overflow-y:hidden}.callout div.sourceCode{margin-left:initial}.blockquote{font-size:inherit;padding-left:1rem;padding-right:1.5rem;color:#404040}.blockquote h1:first-child,.blockquote .h1:first-child,.blockquote h2:first-child,.blockquote .h2:first-child,.blockquote h3:first-child,.blockquote .h3:first-child,.blockquote h4:first-child,.blockquote .h4:first-child,.blockquote h5:first-child,.blockquote .h5:first-child{margin-top:0}pre{background-color:initial;padding:initial;border:initial}p pre code:not(.sourceCode),li pre code:not(.sourceCode),pre code:not(.sourceCode){background-color:initial}p code:not(.sourceCode),li code:not(.sourceCode),td code:not(.sourceCode){background-color:#f8f9fa;padding:.2em}nav p code:not(.sourceCode),nav li code:not(.sourceCode),nav td code:not(.sourceCode){background-color:rgba(0,0,0,0);padding:0}td code:not(.sourceCode){white-space:pre-wrap}#quarto-embedded-source-code-modal>.modal-dialog{max-width:1000px;padding-left:1.75rem;padding-right:1.75rem}#quarto-embedded-source-code-modal>.modal-dialog>.modal-content>.modal-body{padding:0}#quarto-embedded-source-code-modal>.modal-dialog>.modal-content>.modal-body div.sourceCode{margin:0;padding:.2rem .2rem;border-radius:0px;border:none}#quarto-embedded-source-code-modal>.modal-dialog>.modal-content>.modal-header{padding:.7rem}.code-tools-button{font-size:1rem;padding:.15rem .15rem;margin-left:5px;color:#6c757d;background-color:rgba(0,0,0,0);transition:initial;cursor:pointer}.code-tools-button>.bi::before{display:inline-block;height:1rem;width:1rem;content:"";vertical-align:-0.125em;background-image:url('data:image/svg+xml,');background-repeat:no-repeat;background-size:1rem 1rem}.code-tools-button:hover>.bi::before{background-image:url('data:image/svg+xml,')}#quarto-embedded-source-code-modal .code-copy-button>.bi::before{background-image:url('data:image/svg+xml,')}#quarto-embedded-source-code-modal .code-copy-button-checked>.bi::before{background-image:url('data:image/svg+xml,')}.sidebar{will-change:top;transition:top 200ms linear;position:sticky;overflow-y:auto;padding-top:1.2em;max-height:100vh}.sidebar.toc-left,.sidebar.margin-sidebar{top:0px;padding-top:1em}.sidebar.quarto-banner-title-block-sidebar>*{padding-top:1.65em}figure .quarto-notebook-link{margin-top:.5em}.quarto-notebook-link{font-size:.75em;color:#6c757d;margin-bottom:1em;text-decoration:none;display:block}.quarto-notebook-link:hover{text-decoration:underline;color:#4040bf}.quarto-notebook-link::before{display:inline-block;height:.75rem;width:.75rem;margin-bottom:0em;margin-right:.25em;content:"";vertical-align:-0.125em;background-image:url('data:image/svg+xml,');background-repeat:no-repeat;background-size:.75rem .75rem}.toc-actions i.bi,.quarto-code-links i.bi,.quarto-other-links i.bi,.quarto-alternate-notebooks i.bi,.quarto-alternate-formats i.bi{margin-right:.4em;font-size:.8rem}.quarto-other-links-text-target .quarto-code-links i.bi,.quarto-other-links-text-target .quarto-other-links i.bi{margin-right:.2em}.quarto-other-formats-text-target .quarto-alternate-formats i.bi{margin-right:.1em}.toc-actions i.bi.empty,.quarto-code-links i.bi.empty,.quarto-other-links i.bi.empty,.quarto-alternate-notebooks i.bi.empty,.quarto-alternate-formats i.bi.empty{padding-left:1em}.quarto-notebook h2,.quarto-notebook .h2{border-bottom:none}.quarto-notebook .cell-container{display:flex}.quarto-notebook .cell-container .cell{flex-grow:4}.quarto-notebook .cell-container .cell-decorator{padding-top:1.5em;padding-right:1em;text-align:right}.quarto-notebook .cell-container.code-fold .cell-decorator{padding-top:3em}.quarto-notebook .cell-code code{white-space:pre-wrap}.quarto-notebook .cell .cell-output-stderr pre code,.quarto-notebook .cell .cell-output-stdout pre code{white-space:pre-wrap;overflow-wrap:anywhere}.toc-actions,.quarto-alternate-formats,.quarto-other-links,.quarto-code-links,.quarto-alternate-notebooks{padding-left:0em}.sidebar .toc-actions a,.sidebar .quarto-alternate-formats a,.sidebar .quarto-other-links a,.sidebar .quarto-code-links a,.sidebar .quarto-alternate-notebooks a,.sidebar nav[role=doc-toc] a{text-decoration:none}.sidebar .toc-actions a:hover,.sidebar .quarto-other-links a:hover,.sidebar .quarto-code-links a:hover,.sidebar .quarto-alternate-formats a:hover,.sidebar .quarto-alternate-notebooks a:hover{color:#4040bf}.sidebar .toc-actions h2,.sidebar .toc-actions .h2,.sidebar .quarto-code-links h2,.sidebar .quarto-code-links .h2,.sidebar .quarto-other-links h2,.sidebar .quarto-other-links .h2,.sidebar .quarto-alternate-notebooks h2,.sidebar .quarto-alternate-notebooks .h2,.sidebar .quarto-alternate-formats h2,.sidebar .quarto-alternate-formats .h2,.sidebar nav[role=doc-toc]>h2,.sidebar nav[role=doc-toc]>.h2{font-weight:500;margin-bottom:.2rem;margin-top:.3rem;font-family:inherit;border-bottom:0;padding-bottom:0;padding-top:0px}.sidebar .toc-actions>h2,.sidebar .toc-actions>.h2,.sidebar .quarto-code-links>h2,.sidebar .quarto-code-links>.h2,.sidebar .quarto-other-links>h2,.sidebar .quarto-other-links>.h2,.sidebar .quarto-alternate-notebooks>h2,.sidebar .quarto-alternate-notebooks>.h2,.sidebar .quarto-alternate-formats>h2,.sidebar .quarto-alternate-formats>.h2{font-size:.8rem}.sidebar nav[role=doc-toc]>h2,.sidebar nav[role=doc-toc]>.h2{font-size:.875rem}.sidebar nav[role=doc-toc]>ul a{border-left:1px solid #e9ecef;padding-left:.6rem}.sidebar .toc-actions h2>ul a,.sidebar .toc-actions .h2>ul a,.sidebar .quarto-code-links h2>ul a,.sidebar .quarto-code-links .h2>ul a,.sidebar .quarto-other-links h2>ul a,.sidebar .quarto-other-links .h2>ul a,.sidebar .quarto-alternate-notebooks h2>ul a,.sidebar .quarto-alternate-notebooks .h2>ul a,.sidebar .quarto-alternate-formats h2>ul a,.sidebar .quarto-alternate-formats .h2>ul a{border-left:none;padding-left:.6rem}.sidebar .toc-actions ul a:empty,.sidebar .quarto-code-links ul a:empty,.sidebar .quarto-other-links ul a:empty,.sidebar .quarto-alternate-notebooks ul a:empty,.sidebar .quarto-alternate-formats ul a:empty,.sidebar nav[role=doc-toc]>ul a:empty{display:none}.sidebar .toc-actions ul,.sidebar .quarto-code-links ul,.sidebar .quarto-other-links ul,.sidebar .quarto-alternate-notebooks ul,.sidebar .quarto-alternate-formats ul{padding-left:0;list-style:none}.sidebar nav[role=doc-toc] ul{list-style:none;padding-left:0;list-style:none}.sidebar nav[role=doc-toc]>ul{margin-left:.45em}.quarto-margin-sidebar nav[role=doc-toc]{padding-left:.5em}.sidebar .toc-actions>ul,.sidebar .quarto-code-links>ul,.sidebar .quarto-other-links>ul,.sidebar .quarto-alternate-notebooks>ul,.sidebar .quarto-alternate-formats>ul{font-size:.8rem}.sidebar nav[role=doc-toc]>ul{font-size:.875rem}.sidebar .toc-actions ul li a,.sidebar .quarto-code-links ul li a,.sidebar .quarto-other-links ul li a,.sidebar .quarto-alternate-notebooks ul li a,.sidebar .quarto-alternate-formats ul li a,.sidebar nav[role=doc-toc]>ul li a{line-height:1.1rem;padding-bottom:.2rem;padding-top:.2rem;color:inherit}.sidebar nav[role=doc-toc] ul>li>ul>li>a{padding-left:1.2em}.sidebar nav[role=doc-toc] ul>li>ul>li>ul>li>a{padding-left:2.4em}.sidebar nav[role=doc-toc] ul>li>ul>li>ul>li>ul>li>a{padding-left:3.6em}.sidebar nav[role=doc-toc] ul>li>ul>li>ul>li>ul>li>ul>li>a{padding-left:4.8em}.sidebar nav[role=doc-toc] ul>li>ul>li>ul>li>ul>li>ul>li>ul>li>a{padding-left:6em}.sidebar nav[role=doc-toc] ul>li>a.active,.sidebar nav[role=doc-toc] ul>li>ul>li>a.active{border-left:1px solid #4040bf;color:#4040bf !important}.sidebar nav[role=doc-toc] ul>li>a:hover,.sidebar nav[role=doc-toc] ul>li>ul>li>a:hover{color:#4040bf !important}kbd,.kbd{color:#000;background-color:#f8f9fa;border:1px solid;border-radius:5px;border-color:#dee2e6}.quarto-appendix-contents div.hanging-indent{margin-left:0em}.quarto-appendix-contents div.hanging-indent div.csl-entry{margin-left:1em;text-indent:-1em}.citation a,.footnote-ref{text-decoration:none}.footnotes ol{padding-left:1em}.tippy-content>*{margin-bottom:.7em}.tippy-content>*:last-child{margin-bottom:0}.callout{margin-top:1.25rem;margin-bottom:1.25rem;border-radius:.25rem;overflow-wrap:break-word}.callout .callout-title-container{overflow-wrap:anywhere}.callout.callout-style-simple{padding:.4em .7em;border-left:5px solid;border-right:1px solid #dee2e6;border-top:1px solid #dee2e6;border-bottom:1px solid #dee2e6}.callout.callout-style-default{border-left:5px solid;border-right:1px solid #dee2e6;border-top:1px solid #dee2e6;border-bottom:1px solid #dee2e6}.callout .callout-body-container{flex-grow:1}.callout.callout-style-simple .callout-body{font-size:.9rem;font-weight:400}.callout.callout-style-default .callout-body{font-size:.9rem;font-weight:400}.callout:not(.no-icon).callout-titled.callout-style-simple .callout-body{padding-left:1.6em}.callout.callout-titled>.callout-header{padding-top:.2em;margin-bottom:-0.2em}.callout.callout-style-simple>div.callout-header{border-bottom:none;font-size:.9rem;font-weight:600;opacity:75%}.callout.callout-style-default>div.callout-header{border-bottom:none;font-weight:600;opacity:85%;font-size:.9rem;padding-left:.5em;padding-right:.5em}.callout.callout-style-default .callout-body{padding-left:.5em;padding-right:.5em}.callout.callout-style-default .callout-body>:first-child{padding-top:.5rem;margin-top:0}.callout>div.callout-header[data-bs-toggle=collapse]{cursor:pointer}.callout.callout-style-default .callout-header[aria-expanded=false],.callout.callout-style-default .callout-header[aria-expanded=true]{padding-top:0px;margin-bottom:0px;align-items:center}.callout.callout-titled .callout-body>:last-child:not(.sourceCode),.callout.callout-titled .callout-body>div>:last-child:not(.sourceCode){padding-bottom:.5rem;margin-bottom:0}.callout:not(.callout-titled) .callout-body>:first-child,.callout:not(.callout-titled) .callout-body>div>:first-child{margin-top:.25rem}.callout:not(.callout-titled) .callout-body>:last-child,.callout:not(.callout-titled) .callout-body>div>:last-child{margin-bottom:.2rem}.callout.callout-style-simple .callout-icon::before,.callout.callout-style-simple .callout-toggle::before{height:1rem;width:1rem;display:inline-block;content:"";background-repeat:no-repeat;background-size:1rem 1rem}.callout.callout-style-default .callout-icon::before,.callout.callout-style-default .callout-toggle::before{height:.9rem;width:.9rem;display:inline-block;content:"";background-repeat:no-repeat;background-size:.9rem .9rem}.callout.callout-style-default .callout-toggle::before{margin-top:5px}.callout .callout-btn-toggle .callout-toggle::before{transition:transform .2s linear}.callout .callout-header[aria-expanded=false] .callout-toggle::before{transform:rotate(-90deg)}.callout .callout-header[aria-expanded=true] .callout-toggle::before{transform:none}.callout.callout-style-simple:not(.no-icon) div.callout-icon-container{padding-top:.2em;padding-right:.55em}.callout.callout-style-default:not(.no-icon) div.callout-icon-container{padding-top:.1em;padding-right:.35em}.callout.callout-style-default:not(.no-icon) div.callout-title-container{margin-top:-1px}.callout.callout-style-default.callout-caution:not(.no-icon) div.callout-icon-container{padding-top:.3em;padding-right:.35em}.callout>.callout-body>.callout-icon-container>.no-icon,.callout>.callout-header>.callout-icon-container>.no-icon{display:none}div.callout.callout{border-left-color:#6c757d}div.callout.callout-style-default>.callout-header{background-color:#6c757d}div.callout-note.callout{border-left-color:#2780e3}div.callout-note.callout-style-default>.callout-header{background-color:#e9f2fc}div.callout-note:not(.callout-titled) .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-note.callout-titled .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-note .callout-toggle::before{background-image:url('data:image/svg+xml,')}div.callout-tip.callout{border-left-color:#3fb618}div.callout-tip.callout-style-default>.callout-header{background-color:#ecf8e8}div.callout-tip:not(.callout-titled) .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-tip.callout-titled .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-tip .callout-toggle::before{background-image:url('data:image/svg+xml,')}div.callout-warning.callout{border-left-color:#ff7518}div.callout-warning.callout-style-default>.callout-header{background-color:#fff1e8}div.callout-warning:not(.callout-titled) .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-warning.callout-titled .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-warning .callout-toggle::before{background-image:url('data:image/svg+xml,')}div.callout-caution.callout{border-left-color:#f0ad4e}div.callout-caution.callout-style-default>.callout-header{background-color:#fef7ed}div.callout-caution:not(.callout-titled) .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-caution.callout-titled .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-caution .callout-toggle::before{background-image:url('data:image/svg+xml,')}div.callout-important.callout{border-left-color:#ff0039}div.callout-important.callout-style-default>.callout-header{background-color:#ffe6eb}div.callout-important:not(.callout-titled) .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-important.callout-titled .callout-icon::before{background-image:url('data:image/svg+xml,');}div.callout-important .callout-toggle::before{background-image:url('data:image/svg+xml,')}.quarto-toggle-container{display:flex;align-items:center}.quarto-reader-toggle .bi::before,.quarto-color-scheme-toggle .bi::before{display:inline-block;height:1rem;width:1rem;content:"";background-repeat:no-repeat;background-size:1rem 1rem}.sidebar-navigation{padding-left:20px}.navbar{background-color:#3cdd8c;color:#fff}.navbar .quarto-color-scheme-toggle:not(.alternate) .bi::before{background-image:url('data:image/svg+xml,')}.navbar .quarto-color-scheme-toggle.alternate .bi::before{background-image:url('data:image/svg+xml,')}.sidebar-navigation .quarto-color-scheme-toggle:not(.alternate) .bi::before{background-image:url('data:image/svg+xml,')}.sidebar-navigation .quarto-color-scheme-toggle.alternate .bi::before{background-image:url('data:image/svg+xml,')}.quarto-sidebar-toggle{border-color:#dee2e6;border-bottom-left-radius:.25rem;border-bottom-right-radius:.25rem;border-style:solid;border-width:1px;overflow:hidden;border-top-width:0px;padding-top:0px !important}.quarto-sidebar-toggle-title{cursor:pointer;padding-bottom:2px;margin-left:.25em;text-align:center;font-weight:400;font-size:.775em}#quarto-content .quarto-sidebar-toggle{background:#fafafa}#quarto-content .quarto-sidebar-toggle-title{color:#000}.quarto-sidebar-toggle-icon{color:#dee2e6;margin-right:.5em;float:right;transition:transform .2s ease}.quarto-sidebar-toggle-icon::before{padding-top:5px}.quarto-sidebar-toggle.expanded .quarto-sidebar-toggle-icon{transform:rotate(-180deg)}.quarto-sidebar-toggle.expanded .quarto-sidebar-toggle-title{border-bottom:solid #dee2e6 1px}.quarto-sidebar-toggle-contents{background-color:#fff;padding-right:10px;padding-left:10px;margin-top:0px !important;transition:max-height .5s ease}.quarto-sidebar-toggle.expanded .quarto-sidebar-toggle-contents{padding-top:1em;padding-bottom:10px}@media(max-width: 767.98px){.sidebar-menu-container{padding-bottom:5em}}.quarto-sidebar-toggle:not(.expanded) .quarto-sidebar-toggle-contents{padding-top:0px !important;padding-bottom:0px}nav[role=doc-toc]{z-index:1020}#quarto-sidebar>*,nav[role=doc-toc]>*{transition:opacity .1s ease,border .1s ease}#quarto-sidebar.slow>*,nav[role=doc-toc].slow>*{transition:opacity .4s ease,border .4s ease}.quarto-color-scheme-toggle:not(.alternate).top-right .bi::before{background-image:url('data:image/svg+xml,')}.quarto-color-scheme-toggle.alternate.top-right .bi::before{background-image:url('data:image/svg+xml,')}#quarto-appendix.default{border-top:1px solid #dee2e6}#quarto-appendix.default{background-color:#fff;padding-top:1.5em;margin-top:2em;z-index:998}#quarto-appendix.default .quarto-appendix-heading{margin-top:0;line-height:1.4em;font-weight:600;opacity:.9;border-bottom:none;margin-bottom:0}#quarto-appendix.default .footnotes ol,#quarto-appendix.default .footnotes ol li>p:last-of-type,#quarto-appendix.default .quarto-appendix-contents>p:last-of-type{margin-bottom:0}#quarto-appendix.default .footnotes ol{margin-left:.5em}#quarto-appendix.default .quarto-appendix-secondary-label{margin-bottom:.4em}#quarto-appendix.default .quarto-appendix-bibtex{font-size:.7em;padding:1em;border:solid 1px #dee2e6;margin-bottom:1em}#quarto-appendix.default .quarto-appendix-bibtex code.sourceCode{white-space:pre-wrap}#quarto-appendix.default .quarto-appendix-citeas{font-size:.9em;padding:1em;border:solid 1px #dee2e6;margin-bottom:1em}#quarto-appendix.default .quarto-appendix-heading{font-size:1em !important}#quarto-appendix.default *[role=doc-endnotes]>ol,#quarto-appendix.default .quarto-appendix-contents>*:not(h2):not(.h2){font-size:.9em}#quarto-appendix.default section{padding-bottom:1.5em}#quarto-appendix.default section *[role=doc-endnotes],#quarto-appendix.default section>*:not(a){opacity:.9;word-wrap:break-word}.btn.btn-quarto,div.cell-output-display .btn-quarto{--bs-btn-color: #cacccd;--bs-btn-bg: #343a40;--bs-btn-border-color: #343a40;--bs-btn-hover-color: #cacccd;--bs-btn-hover-bg: #52585d;--bs-btn-hover-border-color: #484e53;--bs-btn-focus-shadow-rgb: 75, 80, 85;--bs-btn-active-color: #fff;--bs-btn-active-bg: #5d6166;--bs-btn-active-border-color: #484e53;--bs-btn-active-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color: #fff;--bs-btn-disabled-bg: #343a40;--bs-btn-disabled-border-color: #343a40}nav.quarto-secondary-nav.color-navbar{background-color:#3cdd8c;color:#fff}nav.quarto-secondary-nav.color-navbar h1,nav.quarto-secondary-nav.color-navbar .h1,nav.quarto-secondary-nav.color-navbar .quarto-btn-toggle{color:#fff}@media(max-width: 991.98px){body.nav-sidebar .quarto-title-banner{margin-bottom:0;padding-bottom:1em}body.nav-sidebar #title-block-header{margin-block-end:0}}p.subtitle{margin-top:.25em;margin-bottom:.5em}code a:any-link{color:inherit;text-decoration-color:#6c757d}/*! light */div.observablehq table thead tr th{background-color:var(--bs-body-bg)}input,button,select,optgroup,textarea{background-color:var(--bs-body-bg)}.code-annotated .code-copy-button{margin-right:1.25em;margin-top:0;padding-bottom:0;padding-top:3px}.code-annotation-gutter-bg{background-color:#fff}.code-annotation-gutter{background-color:rgba(233,236,239,.65)}.code-annotation-gutter,.code-annotation-gutter-bg{height:100%;width:calc(20px + .5em);position:absolute;top:0;right:0}dl.code-annotation-container-grid dt{margin-right:1em;margin-top:.25rem}dl.code-annotation-container-grid dt{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;color:#1a1a1a;border:solid #1a1a1a 1px;border-radius:50%;height:22px;width:22px;line-height:22px;font-size:11px;text-align:center;vertical-align:middle;text-decoration:none}dl.code-annotation-container-grid dt[data-target-cell]{cursor:pointer}dl.code-annotation-container-grid dt[data-target-cell].code-annotation-active{color:#fff;border:solid #aaa 1px;background-color:#aaa}pre.code-annotation-code{padding-top:0;padding-bottom:0}pre.code-annotation-code code{z-index:3}#code-annotation-line-highlight-gutter{width:100%;border-top:solid rgba(170,170,170,.2666666667) 1px;border-bottom:solid rgba(170,170,170,.2666666667) 1px;z-index:2;background-color:rgba(170,170,170,.1333333333)}#code-annotation-line-highlight{margin-left:-4em;width:calc(100% + 4em);border-top:solid rgba(170,170,170,.2666666667) 1px;border-bottom:solid rgba(170,170,170,.2666666667) 1px;z-index:2;background-color:rgba(170,170,170,.1333333333)}code.sourceCode .code-annotation-anchor.code-annotation-active{background-color:var(--quarto-hl-normal-color, #aaaaaa);border:solid var(--quarto-hl-normal-color, #aaaaaa) 1px;color:#e9ecef;font-weight:bolder}code.sourceCode .code-annotation-anchor{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;color:var(--quarto-hl-co-color);border:solid var(--quarto-hl-co-color) 1px;border-radius:50%;height:18px;width:18px;font-size:9px;margin-top:2px}code.sourceCode button.code-annotation-anchor{padding:2px;user-select:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;-o-user-select:none}code.sourceCode a.code-annotation-anchor{line-height:18px;text-align:center;vertical-align:middle;cursor:default;text-decoration:none}@media print{.page-columns .column-screen-inset{grid-column:page-start-inset/page-end-inset;z-index:998;opacity:.999}.page-columns .column-screen-inset table{background:#fff}.page-columns .column-screen-inset-left{grid-column:page-start-inset/body-content-end;z-index:998;opacity:.999}.page-columns .column-screen-inset-left table{background:#fff}.page-columns .column-screen-inset-right{grid-column:body-content-start/page-end-inset;z-index:998;opacity:.999}.page-columns .column-screen-inset-right table{background:#fff}.page-columns .column-screen{grid-column:page-start/page-end;z-index:998;opacity:.999}.page-columns .column-screen table{background:#fff}.page-columns .column-screen-left{grid-column:page-start/body-content-end;z-index:998;opacity:.999}.page-columns .column-screen-left table{background:#fff}.page-columns .column-screen-right{grid-column:body-content-start/page-end;z-index:998;opacity:.999}.page-columns .column-screen-right table{background:#fff}.page-columns .column-screen-inset-shaded{grid-column:page-start-inset/page-end-inset;padding:1em;background:#f8f9fa;z-index:998;opacity:.999;margin-bottom:1em}}.quarto-video{margin-bottom:1em}.table{border-top:1px solid #ccc;border-bottom:1px solid #ccc}.table>thead{border-top-width:0;border-bottom:1px solid gray}.table a{word-break:break-word}.table>:not(caption)>*>*{background-color:unset;color:unset}#quarto-document-content .crosstalk-input .checkbox input[type=checkbox],#quarto-document-content .crosstalk-input .checkbox-inline input[type=checkbox]{position:unset;margin-top:unset;margin-left:unset}#quarto-document-content .row{margin-left:unset;margin-right:unset}.quarto-xref{white-space:nowrap}#quarto-draft-alert{margin-top:0px;margin-bottom:0px;padding:.3em;text-align:center;font-size:.9em}#quarto-draft-alert i{margin-right:.3em}#quarto-back-to-top{z-index:1000}pre{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:0.875em;font-weight:400}pre code{font-family:inherit;font-size:inherit;font-weight:inherit}code{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:0.875em;font-weight:400}a{background-color:rgba(0,0,0,0);font-weight:400;text-decoration:underline}a.external:after{content:"";background-image:url('data:image/svg+xml,');background-size:contain;background-repeat:no-repeat;background-position:center center;margin-left:.2em;padding-right:.75em}div.sourceCode code a.external:after{content:none}a.external:after:hover{cursor:pointer}.quarto-ext-icon{display:inline-block;font-size:.75em;padding-left:.3em}.code-with-filename .code-with-filename-file{margin-bottom:0;padding-bottom:2px;padding-top:2px;padding-left:.7em;border:var(--quarto-border-width) solid var(--quarto-border-color);border-radius:var(--quarto-border-radius);border-bottom:0;border-bottom-left-radius:0%;border-bottom-right-radius:0%}.code-with-filename div.sourceCode,.reveal .code-with-filename div.sourceCode{margin-top:0;border-top-left-radius:0%;border-top-right-radius:0%}.code-with-filename .code-with-filename-file pre{margin-bottom:0}.code-with-filename .code-with-filename-file{background-color:rgba(219,219,219,.8)}.quarto-dark .code-with-filename .code-with-filename-file{background-color:#555}.code-with-filename .code-with-filename-file strong{font-weight:400}.quarto-title-banner{margin-bottom:1em;color:#fff;background:#3cdd8c}.quarto-title-banner a{color:#fff}.quarto-title-banner h1,.quarto-title-banner .h1,.quarto-title-banner h2,.quarto-title-banner .h2{color:#fff}.quarto-title-banner .code-tools-button{color:#fff}.quarto-title-banner .code-tools-button:hover{color:#fff}.quarto-title-banner .code-tools-button>.bi::before{background-image:url('data:image/svg+xml,')}.quarto-title-banner .code-tools-button:hover>.bi::before{background-image:url('data:image/svg+xml,')}.quarto-title-banner .quarto-title .title{font-weight:600}.quarto-title-banner .quarto-categories{margin-top:.75em}@media(min-width: 992px){.quarto-title-banner{padding-top:2.5em;padding-bottom:2.5em}}@media(max-width: 991.98px){.quarto-title-banner{padding-top:1em;padding-bottom:1em}}@media(max-width: 767.98px){body.hypothesis-enabled #title-block-header>*{padding-right:20px}}main.quarto-banner-title-block>section:first-child>h2,main.quarto-banner-title-block>section:first-child>.h2,main.quarto-banner-title-block>section:first-child>h3,main.quarto-banner-title-block>section:first-child>.h3,main.quarto-banner-title-block>section:first-child>h4,main.quarto-banner-title-block>section:first-child>.h4{margin-top:0}.quarto-title .quarto-categories{display:flex;flex-wrap:wrap;row-gap:.5em;column-gap:.4em;padding-bottom:.5em;margin-top:.75em}.quarto-title .quarto-categories .quarto-category{padding:.25em .75em;font-size:.65em;text-transform:uppercase;border:solid 1px;border-radius:.25rem;opacity:.6}.quarto-title .quarto-categories .quarto-category a{color:inherit}.quarto-title-meta-container{display:grid;grid-template-columns:1fr auto}.quarto-title-meta-column-end{display:flex;flex-direction:column;padding-left:1em}.quarto-title-meta-column-end a .bi{margin-right:.3em}#title-block-header.quarto-title-block.default .quarto-title-meta{display:grid;grid-template-columns:repeat(2, 1fr);grid-column-gap:1em}#title-block-header.quarto-title-block.default .quarto-title .title{margin-bottom:0}#title-block-header.quarto-title-block.default .quarto-title-author-orcid img{margin-top:-0.2em;height:.8em;width:.8em}#title-block-header.quarto-title-block.default .quarto-title-author-email{opacity:.7}#title-block-header.quarto-title-block.default .quarto-description p:last-of-type{margin-bottom:0}#title-block-header.quarto-title-block.default .quarto-title-meta-contents p,#title-block-header.quarto-title-block.default .quarto-title-authors p,#title-block-header.quarto-title-block.default .quarto-title-affiliations p{margin-bottom:.1em}#title-block-header.quarto-title-block.default .quarto-title-meta-heading{text-transform:uppercase;margin-top:1em;font-size:.8em;opacity:.8;font-weight:400}#title-block-header.quarto-title-block.default .quarto-title-meta-contents{font-size:.9em}#title-block-header.quarto-title-block.default .quarto-title-meta-contents p.affiliation:last-of-type{margin-bottom:.1em}#title-block-header.quarto-title-block.default p.affiliation{margin-bottom:.1em}#title-block-header.quarto-title-block.default .keywords,#title-block-header.quarto-title-block.default .description,#title-block-header.quarto-title-block.default .abstract{margin-top:0}#title-block-header.quarto-title-block.default .keywords>p,#title-block-header.quarto-title-block.default .description>p,#title-block-header.quarto-title-block.default .abstract>p{font-size:.9em}#title-block-header.quarto-title-block.default .keywords>p:last-of-type,#title-block-header.quarto-title-block.default .description>p:last-of-type,#title-block-header.quarto-title-block.default .abstract>p:last-of-type{margin-bottom:0}#title-block-header.quarto-title-block.default .keywords .block-title,#title-block-header.quarto-title-block.default .description .block-title,#title-block-header.quarto-title-block.default .abstract .block-title{margin-top:1em;text-transform:uppercase;font-size:.8em;opacity:.8;font-weight:400}#title-block-header.quarto-title-block.default .quarto-title-meta-author{display:grid;grid-template-columns:minmax(max-content, 1fr) 1fr;grid-column-gap:1em}.quarto-title-tools-only{display:flex;justify-content:right}body{-webkit-font-smoothing:antialiased}.badge.bg-light{color:#343a40}.progress .progress-bar{font-size:8px;line-height:8px}:root{--quarto-scss-export-gray-300: #dee2e6;--quarto-scss-export-gray-500: #adb5bd;--quarto-scss-export-gray-600: #6c757d;--quarto-scss-export-gray-800: #343a40;--quarto-scss-export-card-cap-bg: rgba(52, 58, 64, 0.25);--quarto-scss-export-border-color: #dee2e6;--quarto-scss-export-text-muted: #6c757d;--quarto-scss-export-body-bg: #FFFFFF;--quarto-scss-export-body-color: #000000;--quarto-scss-export-link-color: #4040BF;--quarto-scss-export-primary: #E8E8FC;--quarto-scss-export-code-bg: #f8f9fa;--quarto-scss-export-code-color: #7d12ba;--quarto-scss-export-navbar-bg: #3CDD8C;--quarto-scss-export-navbar-fg: #FFFFFF;--quarto-scss-export-navbar-hl: #000000;--quarto-scss-export-white: #fff;--quarto-scss-export-gray-100: #f8f9fa;--quarto-scss-export-gray-200: #e9ecef;--quarto-scss-export-gray-400: #ced4da;--quarto-scss-export-gray-700: #495057;--quarto-scss-export-gray-900: #212529;--quarto-scss-export-black: #000;--quarto-scss-export-blue: #2780e3;--quarto-scss-export-indigo: #6610f2;--quarto-scss-export-purple: #613d7c;--quarto-scss-export-pink: #e83e8c;--quarto-scss-export-red: #ff0039;--quarto-scss-export-orange: #f0ad4e;--quarto-scss-export-yellow: #ff7518;--quarto-scss-export-green: #3fb618;--quarto-scss-export-teal: #20c997;--quarto-scss-export-cyan: #9954bb;--quarto-scss-export-secondary: #343a40;--quarto-scss-export-success: #3fb618;--quarto-scss-export-info: #9954bb;--quarto-scss-export-warning: #ff7518;--quarto-scss-export-danger: #ff0039;--quarto-scss-export-light: #f8f9fa;--quarto-scss-export-dark: #343a40;--quarto-scss-export-title-banner-color: ;--quarto-scss-export-title-banner-bg: ;--quarto-scss-export-btn-code-copy-color: #5E5E5E;--quarto-scss-export-btn-code-copy-color-active: #4758AB;--quarto-scss-export-sidebar-bg: #FFFFFF;--quarto-scss-export-link-color-bg: transparent;--quarto-scss-export-toc-color: #4040BF;--quarto-scss-export-toc-active-border: #4040BF;--quarto-scss-export-toc-inactive-border: #e9ecef;--quarto-scss-export-navbar-default: #E8E8FC;--quarto-scss-export-navbar-hl-override: #29297a;--quarto-scss-export-btn-bg: #343a40;--quarto-scss-export-btn-fg: #cacccd;--quarto-scss-export-body-contrast-bg: #FFFFFF;--quarto-scss-export-body-contrast-color: #000000;--quarto-scss-export-navbar-brand: #FFFFFF;--quarto-scss-export-navbar-brand-hl: #000000;--quarto-scss-export-navbar-toggler-border-color: rgba(255, 255, 255, 0);--quarto-scss-export-navbar-hover-color: rgba(0, 0, 0, 0.8);--quarto-scss-export-navbar-disabled-color: rgba(255, 255, 255, 0.75);--quarto-scss-export-sidebar-fg: #595959;--quarto-scss-export-title-block-color: #000000;--quarto-scss-export-title-block-contast-color: #FFFFFF;--quarto-scss-export-footer-bg: #FFFFFF;--quarto-scss-export-footer-fg: #757575;--quarto-scss-export-popover-bg: #FFFFFF;--quarto-scss-export-input-bg: #FFFFFF;--quarto-scss-export-input-border-color: #dee2e6;--quarto-scss-export-code-annotation-higlight-color: rgba(170, 170, 170, 0.2666666667);--quarto-scss-export-code-annotation-higlight-bg: rgba(170, 170, 170, 0.1333333333);--quarto-scss-export-table-group-separator-color: gray;--quarto-scss-export-table-group-separator-color-lighter: #cccccc;--quarto-scss-export-link-decoration: underline;--quarto-scss-export-table-border-color: #dee2e6;--quarto-scss-export-sidebar-glass-bg: rgba(102, 102, 102, 0.4);--quarto-scss-export-color-contrast-dark: #000;--quarto-scss-export-color-contrast-light: #fff;--quarto-scss-export-blue-100: #d4e6f9;--quarto-scss-export-blue-200: #a9ccf4;--quarto-scss-export-blue-300: #7db3ee;--quarto-scss-export-blue-400: #5299e9;--quarto-scss-export-blue-500: #2780e3;--quarto-scss-export-blue-600: #1f66b6;--quarto-scss-export-blue-700: #174d88;--quarto-scss-export-blue-800: #10335b;--quarto-scss-export-blue-900: #081a2d;--quarto-scss-export-indigo-100: #e0cffc;--quarto-scss-export-indigo-200: #c29ffa;--quarto-scss-export-indigo-300: #a370f7;--quarto-scss-export-indigo-400: #8540f5;--quarto-scss-export-indigo-500: #6610f2;--quarto-scss-export-indigo-600: #520dc2;--quarto-scss-export-indigo-700: #3d0a91;--quarto-scss-export-indigo-800: #290661;--quarto-scss-export-indigo-900: #140330;--quarto-scss-export-purple-100: #dfd8e5;--quarto-scss-export-purple-200: #c0b1cb;--quarto-scss-export-purple-300: #a08bb0;--quarto-scss-export-purple-400: #816496;--quarto-scss-export-purple-500: #613d7c;--quarto-scss-export-purple-600: #4e3163;--quarto-scss-export-purple-700: #3a254a;--quarto-scss-export-purple-800: #271832;--quarto-scss-export-purple-900: #130c19;--quarto-scss-export-pink-100: #fad8e8;--quarto-scss-export-pink-200: #f6b2d1;--quarto-scss-export-pink-300: #f18bba;--quarto-scss-export-pink-400: #ed65a3;--quarto-scss-export-pink-500: #e83e8c;--quarto-scss-export-pink-600: #ba3270;--quarto-scss-export-pink-700: #8b2554;--quarto-scss-export-pink-800: #5d1938;--quarto-scss-export-pink-900: #2e0c1c;--quarto-scss-export-red-100: #ffccd7;--quarto-scss-export-red-200: #ff99b0;--quarto-scss-export-red-300: #ff6688;--quarto-scss-export-red-400: #ff3361;--quarto-scss-export-red-500: #ff0039;--quarto-scss-export-red-600: #cc002e;--quarto-scss-export-red-700: #990022;--quarto-scss-export-red-800: #660017;--quarto-scss-export-red-900: #33000b;--quarto-scss-export-orange-100: #fcefdc;--quarto-scss-export-orange-200: #f9deb8;--quarto-scss-export-orange-300: #f6ce95;--quarto-scss-export-orange-400: #f3bd71;--quarto-scss-export-orange-500: #f0ad4e;--quarto-scss-export-orange-600: #c08a3e;--quarto-scss-export-orange-700: #90682f;--quarto-scss-export-orange-800: #60451f;--quarto-scss-export-orange-900: #302310;--quarto-scss-export-yellow-100: #ffe3d1;--quarto-scss-export-yellow-200: #ffc8a3;--quarto-scss-export-yellow-300: #ffac74;--quarto-scss-export-yellow-400: #ff9146;--quarto-scss-export-yellow-500: #ff7518;--quarto-scss-export-yellow-600: #cc5e13;--quarto-scss-export-yellow-700: #99460e;--quarto-scss-export-yellow-800: #662f0a;--quarto-scss-export-yellow-900: #331705;--quarto-scss-export-green-100: #d9f0d1;--quarto-scss-export-green-200: #b2e2a3;--quarto-scss-export-green-300: #8cd374;--quarto-scss-export-green-400: #65c546;--quarto-scss-export-green-500: #3fb618;--quarto-scss-export-green-600: #329213;--quarto-scss-export-green-700: #266d0e;--quarto-scss-export-green-800: #19490a;--quarto-scss-export-green-900: #0d2405;--quarto-scss-export-teal-100: #d2f4ea;--quarto-scss-export-teal-200: #a6e9d5;--quarto-scss-export-teal-300: #79dfc1;--quarto-scss-export-teal-400: #4dd4ac;--quarto-scss-export-teal-500: #20c997;--quarto-scss-export-teal-600: #1aa179;--quarto-scss-export-teal-700: #13795b;--quarto-scss-export-teal-800: #0d503c;--quarto-scss-export-teal-900: #06281e;--quarto-scss-export-cyan-100: #ebddf1;--quarto-scss-export-cyan-200: #d6bbe4;--quarto-scss-export-cyan-300: #c298d6;--quarto-scss-export-cyan-400: #ad76c9;--quarto-scss-export-cyan-500: #9954bb;--quarto-scss-export-cyan-600: #7a4396;--quarto-scss-export-cyan-700: #5c3270;--quarto-scss-export-cyan-800: #3d224b;--quarto-scss-export-cyan-900: #1f1125;--quarto-scss-export-default: #343a40;--quarto-scss-export-primary-text-emphasis: #5d5d65;--quarto-scss-export-secondary-text-emphasis: #15171a;--quarto-scss-export-success-text-emphasis: #19490a;--quarto-scss-export-info-text-emphasis: #3d224b;--quarto-scss-export-warning-text-emphasis: #662f0a;--quarto-scss-export-danger-text-emphasis: #660017;--quarto-scss-export-light-text-emphasis: #495057;--quarto-scss-export-dark-text-emphasis: #495057;--quarto-scss-export-primary-bg-subtle: #fafafe;--quarto-scss-export-secondary-bg-subtle: #d6d8d9;--quarto-scss-export-success-bg-subtle: #d9f0d1;--quarto-scss-export-info-bg-subtle: #ebddf1;--quarto-scss-export-warning-bg-subtle: #ffe3d1;--quarto-scss-export-danger-bg-subtle: #ffccd7;--quarto-scss-export-light-bg-subtle: #fcfcfd;--quarto-scss-export-dark-bg-subtle: #ced4da;--quarto-scss-export-primary-border-subtle: #f6f6fe;--quarto-scss-export-secondary-border-subtle: #aeb0b3;--quarto-scss-export-success-border-subtle: #b2e2a3;--quarto-scss-export-info-border-subtle: #d6bbe4;--quarto-scss-export-warning-border-subtle: #ffc8a3;--quarto-scss-export-danger-border-subtle: #ff99b0;--quarto-scss-export-light-border-subtle: #e9ecef;--quarto-scss-export-dark-border-subtle: #adb5bd;--quarto-scss-export-body-text-align: ;--quarto-scss-export-body-secondary-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-body-secondary-bg: #e9ecef;--quarto-scss-export-body-tertiary-color: rgba(0, 0, 0, 0.5);--quarto-scss-export-body-tertiary-bg: #f8f9fa;--quarto-scss-export-body-emphasis-color: #000;--quarto-scss-export-link-hover-color: #333399;--quarto-scss-export-link-hover-decoration: ;--quarto-scss-export-border-color-translucent: rgba(0, 0, 0, 0.175);--quarto-scss-export-component-active-bg: #E8E8FC;--quarto-scss-export-component-active-color: #000;--quarto-scss-export-focus-ring-color: rgba(232, 232, 252, 0.25);--quarto-scss-export-headings-font-family: ;--quarto-scss-export-headings-font-style: ;--quarto-scss-export-display-font-family: ;--quarto-scss-export-display-font-style: ;--quarto-scss-export-blockquote-footer-color: #6c757d;--quarto-scss-export-blockquote-border-color: #e9ecef;--quarto-scss-export-hr-bg-color: ;--quarto-scss-export-hr-height: ;--quarto-scss-export-hr-border-color: ;--quarto-scss-export-legend-font-weight: ;--quarto-scss-export-mark-bg: #ffe3d1;--quarto-scss-export-table-color: #000000;--quarto-scss-export-table-bg: #FFFFFF;--quarto-scss-export-table-accent-bg: transparent;--quarto-scss-export-table-th-font-weight: ;--quarto-scss-export-table-striped-color: #000000;--quarto-scss-export-table-striped-bg: rgba(0, 0, 0, 0.05);--quarto-scss-export-table-active-color: #000000;--quarto-scss-export-table-active-bg: rgba(0, 0, 0, 0.1);--quarto-scss-export-table-hover-color: #000000;--quarto-scss-export-table-hover-bg: rgba(0, 0, 0, 0.075);--quarto-scss-export-table-caption-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-input-btn-font-family: ;--quarto-scss-export-input-btn-focus-color: rgba(232, 232, 252, 0.25);--quarto-scss-export-btn-color: #000000;--quarto-scss-export-btn-font-family: ;--quarto-scss-export-btn-white-space: ;--quarto-scss-export-btn-link-color: #4040BF;--quarto-scss-export-btn-link-hover-color: #333399;--quarto-scss-export-btn-link-disabled-color: #6c757d;--quarto-scss-export-form-text-font-style: ;--quarto-scss-export-form-text-font-weight: ;--quarto-scss-export-form-text-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-form-label-font-size: ;--quarto-scss-export-form-label-font-style: ;--quarto-scss-export-form-label-font-weight: ;--quarto-scss-export-form-label-color: ;--quarto-scss-export-input-font-family: ;--quarto-scss-export-input-disabled-color: ;--quarto-scss-export-input-disabled-bg: #e9ecef;--quarto-scss-export-input-disabled-border-color: ;--quarto-scss-export-input-color: #000000;--quarto-scss-export-input-focus-bg: #FFFFFF;--quarto-scss-export-input-focus-border-color: #f4f4fe;--quarto-scss-export-input-focus-color: #000000;--quarto-scss-export-input-placeholder-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-input-plaintext-color: #000000;--quarto-scss-export-form-check-label-color: ;--quarto-scss-export-form-check-transition: ;--quarto-scss-export-form-check-input-bg: #FFFFFF;--quarto-scss-export-form-check-input-focus-border: #f4f4fe;--quarto-scss-export-form-check-input-checked-color: #000;--quarto-scss-export-form-check-input-checked-bg-color: #E8E8FC;--quarto-scss-export-form-check-input-checked-border-color: #E8E8FC;--quarto-scss-export-form-check-input-indeterminate-color: #000;--quarto-scss-export-form-check-input-indeterminate-bg-color: #E8E8FC;--quarto-scss-export-form-check-input-indeterminate-border-color: #E8E8FC;--quarto-scss-export-form-switch-color: rgba(0, 0, 0, 0.25);--quarto-scss-export-form-switch-focus-color: #f4f4fe;--quarto-scss-export-form-switch-checked-color: #000;--quarto-scss-export-input-group-addon-color: #000000;--quarto-scss-export-input-group-addon-bg: #f8f9fa;--quarto-scss-export-input-group-addon-border-color: #dee2e6;--quarto-scss-export-form-select-font-family: ;--quarto-scss-export-form-select-color: #000000;--quarto-scss-export-form-select-bg: #FFFFFF;--quarto-scss-export-form-select-disabled-color: ;--quarto-scss-export-form-select-disabled-bg: #e9ecef;--quarto-scss-export-form-select-disabled-border-color: ;--quarto-scss-export-form-select-indicator-color: #343a40;--quarto-scss-export-form-select-border-color: #dee2e6;--quarto-scss-export-form-select-focus-border-color: #f4f4fe;--quarto-scss-export-form-range-track-bg: #f8f9fa;--quarto-scss-export-form-range-thumb-bg: #E8E8FC;--quarto-scss-export-form-range-thumb-active-bg: #f8f8fe;--quarto-scss-export-form-range-thumb-disabled-bg: rgba(0, 0, 0, 0.75);--quarto-scss-export-form-file-button-color: #000000;--quarto-scss-export-form-file-button-bg: #f8f9fa;--quarto-scss-export-form-file-button-hover-bg: #e9ecef;--quarto-scss-export-form-floating-label-disabled-color: #6c757d;--quarto-scss-export-form-feedback-font-style: ;--quarto-scss-export-form-feedback-valid-color: #3fb618;--quarto-scss-export-form-feedback-invalid-color: #ff0039;--quarto-scss-export-form-feedback-icon-valid-color: #3fb618;--quarto-scss-export-form-feedback-icon-invalid-color: #ff0039;--quarto-scss-export-form-valid-color: #3fb618;--quarto-scss-export-form-valid-border-color: #3fb618;--quarto-scss-export-form-invalid-color: #ff0039;--quarto-scss-export-form-invalid-border-color: #ff0039;--quarto-scss-export-nav-link-font-size: ;--quarto-scss-export-nav-link-font-weight: ;--quarto-scss-export-nav-link-color: #4040BF;--quarto-scss-export-nav-link-hover-color: #333399;--quarto-scss-export-nav-link-disabled-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-nav-tabs-border-color: #dee2e6;--quarto-scss-export-nav-tabs-link-hover-border-color: #e9ecef #e9ecef #dee2e6;--quarto-scss-export-nav-tabs-link-active-color: #000;--quarto-scss-export-nav-tabs-link-active-bg: #FFFFFF;--quarto-scss-export-nav-pills-link-active-bg: #E8E8FC;--quarto-scss-export-nav-pills-link-active-color: #000;--quarto-scss-export-nav-underline-link-active-color: #000;--quarto-scss-export-navbar-padding-x: ;--quarto-scss-export-navbar-light-contrast: #000;--quarto-scss-export-navbar-dark-contrast: #000;--quarto-scss-export-navbar-light-icon-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-navbar-dark-icon-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-dropdown-color: #000000;--quarto-scss-export-dropdown-bg: #FFFFFF;--quarto-scss-export-dropdown-border-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-dropdown-divider-bg: rgba(0, 0, 0, 0.175);--quarto-scss-export-dropdown-link-color: #000000;--quarto-scss-export-dropdown-link-hover-color: #000000;--quarto-scss-export-dropdown-link-hover-bg: #f8f9fa;--quarto-scss-export-dropdown-link-active-bg: #E8E8FC;--quarto-scss-export-dropdown-link-active-color: #000;--quarto-scss-export-dropdown-link-disabled-color: rgba(0, 0, 0, 0.5);--quarto-scss-export-dropdown-header-color: #6c757d;--quarto-scss-export-dropdown-dark-color: #dee2e6;--quarto-scss-export-dropdown-dark-bg: #343a40;--quarto-scss-export-dropdown-dark-border-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-dropdown-dark-divider-bg: rgba(0, 0, 0, 0.175);--quarto-scss-export-dropdown-dark-box-shadow: ;--quarto-scss-export-dropdown-dark-link-color: #dee2e6;--quarto-scss-export-dropdown-dark-link-hover-color: #fff;--quarto-scss-export-dropdown-dark-link-hover-bg: rgba(255, 255, 255, 0.15);--quarto-scss-export-dropdown-dark-link-active-color: #000;--quarto-scss-export-dropdown-dark-link-active-bg: #E8E8FC;--quarto-scss-export-dropdown-dark-link-disabled-color: #adb5bd;--quarto-scss-export-dropdown-dark-header-color: #adb5bd;--quarto-scss-export-pagination-color: #4040BF;--quarto-scss-export-pagination-bg: #FFFFFF;--quarto-scss-export-pagination-border-color: #dee2e6;--quarto-scss-export-pagination-focus-color: #333399;--quarto-scss-export-pagination-focus-bg: #e9ecef;--quarto-scss-export-pagination-hover-color: #333399;--quarto-scss-export-pagination-hover-bg: #f8f9fa;--quarto-scss-export-pagination-hover-border-color: #dee2e6;--quarto-scss-export-pagination-active-color: #000;--quarto-scss-export-pagination-active-bg: #E8E8FC;--quarto-scss-export-pagination-active-border-color: #E8E8FC;--quarto-scss-export-pagination-disabled-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-pagination-disabled-bg: #e9ecef;--quarto-scss-export-pagination-disabled-border-color: #dee2e6;--quarto-scss-export-card-title-color: ;--quarto-scss-export-card-subtitle-color: ;--quarto-scss-export-card-border-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-card-box-shadow: ;--quarto-scss-export-card-cap-color: ;--quarto-scss-export-card-height: ;--quarto-scss-export-card-color: ;--quarto-scss-export-card-bg: #FFFFFF;--quarto-scss-export-accordion-color: #000000;--quarto-scss-export-accordion-bg: #FFFFFF;--quarto-scss-export-accordion-border-color: #dee2e6;--quarto-scss-export-accordion-button-color: #000000;--quarto-scss-export-accordion-button-bg: #FFFFFF;--quarto-scss-export-accordion-button-active-bg: #fafafe;--quarto-scss-export-accordion-button-active-color: #5d5d65;--quarto-scss-export-accordion-button-focus-border-color: #f4f4fe;--quarto-scss-export-accordion-icon-color: #000000;--quarto-scss-export-accordion-icon-active-color: #5d5d65;--quarto-scss-export-tooltip-color: #FFFFFF;--quarto-scss-export-tooltip-bg: #000;--quarto-scss-export-tooltip-margin: ;--quarto-scss-export-tooltip-arrow-color: ;--quarto-scss-export-form-feedback-tooltip-line-height: ;--quarto-scss-export-popover-border-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-popover-header-bg: #e9ecef;--quarto-scss-export-popover-body-color: #000000;--quarto-scss-export-popover-arrow-color: #FFFFFF;--quarto-scss-export-popover-arrow-outer-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-toast-color: ;--quarto-scss-export-toast-background-color: rgba(255, 255, 255, 0.85);--quarto-scss-export-toast-border-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-toast-header-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-toast-header-background-color: rgba(255, 255, 255, 0.85);--quarto-scss-export-toast-header-border-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-badge-color: #fff;--quarto-scss-export-modal-content-color: ;--quarto-scss-export-modal-content-bg: #FFFFFF;--quarto-scss-export-modal-content-border-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-modal-backdrop-bg: #000;--quarto-scss-export-modal-header-border-color: #dee2e6;--quarto-scss-export-modal-footer-bg: ;--quarto-scss-export-modal-footer-border-color: #dee2e6;--quarto-scss-export-progress-bg: #e9ecef;--quarto-scss-export-progress-bar-color: #fff;--quarto-scss-export-progress-bar-bg: #E8E8FC;--quarto-scss-export-list-group-color: #000000;--quarto-scss-export-list-group-bg: #FFFFFF;--quarto-scss-export-list-group-border-color: #dee2e6;--quarto-scss-export-list-group-hover-bg: #f8f9fa;--quarto-scss-export-list-group-active-bg: #E8E8FC;--quarto-scss-export-list-group-active-color: #000;--quarto-scss-export-list-group-active-border-color: #E8E8FC;--quarto-scss-export-list-group-disabled-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-list-group-disabled-bg: #FFFFFF;--quarto-scss-export-list-group-action-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-list-group-action-hover-color: #000;--quarto-scss-export-list-group-action-active-color: #000000;--quarto-scss-export-list-group-action-active-bg: #e9ecef;--quarto-scss-export-thumbnail-bg: #FFFFFF;--quarto-scss-export-thumbnail-border-color: #dee2e6;--quarto-scss-export-figure-caption-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-breadcrumb-font-size: ;--quarto-scss-export-breadcrumb-bg: ;--quarto-scss-export-breadcrumb-divider-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-breadcrumb-active-color: rgba(0, 0, 0, 0.75);--quarto-scss-export-breadcrumb-border-radius: ;--quarto-scss-export-carousel-control-color: #fff;--quarto-scss-export-carousel-indicator-active-bg: #fff;--quarto-scss-export-carousel-caption-color: #fff;--quarto-scss-export-carousel-dark-indicator-active-bg: #000;--quarto-scss-export-carousel-dark-caption-color: #000;--quarto-scss-export-btn-close-color: #000;--quarto-scss-export-offcanvas-border-color: rgba(0, 0, 0, 0.175);--quarto-scss-export-offcanvas-bg-color: #FFFFFF;--quarto-scss-export-offcanvas-color: #000000;--quarto-scss-export-offcanvas-backdrop-bg: #000;--quarto-scss-export-code-color-dark: white;--quarto-scss-export-kbd-color: #FFFFFF;--quarto-scss-export-kbd-bg: #000000;--quarto-scss-export-nested-kbd-font-weight: ;--quarto-scss-export-pre-bg: #f8f9fa;--quarto-scss-export-pre-color: #000;--quarto-scss-export-bslib-sidebar-bg: rgba(var(--bs-emphasis-color-rgb, 0, 0, 0), 0.05);--quarto-scss-export-bslib-sidebar-toggle-bg: rgba(var(--bs-emphasis-color-rgb, 0, 0, 0), 0.1);--quarto-scss-export-bslib-page-sidebar-title-bg: #3CDD8C;--quarto-scss-export-bslib-page-sidebar-title-color: #000;--quarto-scss-export-sidebar-color: #595959;--quarto-scss-export-sidebar-hover-color: rgba(64, 64, 191, 0.8);--quarto-scss-export-sidebar-disabled-color: rgba(89, 89, 89, 0.75);--quarto-scss-export-valuebox-bg-primary: #5397e9;--quarto-scss-export-valuebox-bg-secondary: #343a40;--quarto-scss-export-valuebox-bg-success: #3aa716;--quarto-scss-export-valuebox-bg-info: rgba(153, 84, 187, 0.7019607843);--quarto-scss-export-valuebox-bg-warning: #fa6400;--quarto-scss-export-valuebox-bg-danger: rgba(255, 0, 57, 0.7019607843);--quarto-scss-export-valuebox-bg-light: #f8f9fa;--quarto-scss-export-valuebox-bg-dark: #343a40;--quarto-scss-export-mermaid-bg-color: #FFFFFF;--quarto-scss-export-mermaid-edge-color: #343a40;--quarto-scss-export-mermaid-node-fg-color: #000000;--quarto-scss-export-mermaid-fg-color: #000000;--quarto-scss-export-mermaid-fg-color--lighter: #1a1a1a;--quarto-scss-export-mermaid-fg-color--lightest: #333333;--quarto-scss-export-mermaid-label-bg-color: #FFFFFF;--quarto-scss-export-mermaid-label-fg-color: #E8E8FC;--quarto-scss-export-mermaid-node-bg-color: rgba(232, 232, 252, 0.1);--quarto-scss-export-code-block-border-left-color: #dee2e6;--quarto-scss-export-callout-color-note: #2780e3;--quarto-scss-export-callout-color-tip: #3fb618;--quarto-scss-export-callout-color-important: #ff0039;--quarto-scss-export-callout-color-caution: #f0ad4e;--quarto-scss-export-callout-color-warning: #ff7518} \ No newline at end of file diff --git a/site_libs/bootstrap/bootstrap-icons.css b/site_libs/bootstrap/bootstrap-icons.css new file mode 100644 index 00000000..285e4448 --- /dev/null +++ b/site_libs/bootstrap/bootstrap-icons.css @@ -0,0 +1,2078 @@ +/*! + * Bootstrap Icons v1.11.1 (https://icons.getbootstrap.com/) + * Copyright 2019-2023 The Bootstrap Authors + * Licensed under MIT (https://github.com/twbs/icons/blob/main/LICENSE) + */ + +@font-face { + font-display: block; + font-family: "bootstrap-icons"; + src: +url("./bootstrap-icons.woff?2820a3852bdb9a5832199cc61cec4e65") format("woff"); +} + +.bi::before, +[class^="bi-"]::before, +[class*=" bi-"]::before { + display: inline-block; + font-family: bootstrap-icons !important; + font-style: normal; + font-weight: normal !important; + font-variant: normal; + text-transform: none; + line-height: 1; + vertical-align: -.125em; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} + +.bi-123::before { content: "\f67f"; } +.bi-alarm-fill::before { content: "\f101"; } +.bi-alarm::before { content: "\f102"; } +.bi-align-bottom::before { content: "\f103"; } +.bi-align-center::before { content: "\f104"; } +.bi-align-end::before { content: "\f105"; } +.bi-align-middle::before { content: "\f106"; } +.bi-align-start::before { content: "\f107"; } +.bi-align-top::before { content: "\f108"; } +.bi-alt::before { content: "\f109"; } +.bi-app-indicator::before { content: "\f10a"; } +.bi-app::before { content: "\f10b"; } +.bi-archive-fill::before { content: "\f10c"; } +.bi-archive::before { content: "\f10d"; } +.bi-arrow-90deg-down::before { content: "\f10e"; } +.bi-arrow-90deg-left::before { content: "\f10f"; } +.bi-arrow-90deg-right::before { content: "\f110"; } +.bi-arrow-90deg-up::before { content: "\f111"; } +.bi-arrow-bar-down::before { content: "\f112"; } +.bi-arrow-bar-left::before { content: "\f113"; } +.bi-arrow-bar-right::before { content: "\f114"; } +.bi-arrow-bar-up::before { content: "\f115"; } +.bi-arrow-clockwise::before { content: "\f116"; } +.bi-arrow-counterclockwise::before { content: "\f117"; } +.bi-arrow-down-circle-fill::before { content: "\f118"; } +.bi-arrow-down-circle::before { content: "\f119"; } +.bi-arrow-down-left-circle-fill::before { content: "\f11a"; } +.bi-arrow-down-left-circle::before { content: "\f11b"; } +.bi-arrow-down-left-square-fill::before { content: "\f11c"; } +.bi-arrow-down-left-square::before { content: "\f11d"; } +.bi-arrow-down-left::before { content: "\f11e"; } +.bi-arrow-down-right-circle-fill::before { content: "\f11f"; } +.bi-arrow-down-right-circle::before { content: "\f120"; } +.bi-arrow-down-right-square-fill::before { content: "\f121"; } +.bi-arrow-down-right-square::before { content: "\f122"; } +.bi-arrow-down-right::before { content: "\f123"; } +.bi-arrow-down-short::before { content: "\f124"; } +.bi-arrow-down-square-fill::before { content: "\f125"; } +.bi-arrow-down-square::before { content: "\f126"; } +.bi-arrow-down-up::before { content: "\f127"; } +.bi-arrow-down::before { content: "\f128"; } +.bi-arrow-left-circle-fill::before { content: "\f129"; } +.bi-arrow-left-circle::before { content: "\f12a"; } +.bi-arrow-left-right::before { content: "\f12b"; } +.bi-arrow-left-short::before { content: "\f12c"; } +.bi-arrow-left-square-fill::before { content: "\f12d"; } +.bi-arrow-left-square::before { content: "\f12e"; } +.bi-arrow-left::before { content: "\f12f"; } +.bi-arrow-repeat::before { content: "\f130"; } +.bi-arrow-return-left::before { content: "\f131"; } +.bi-arrow-return-right::before { content: "\f132"; } +.bi-arrow-right-circle-fill::before { content: "\f133"; } +.bi-arrow-right-circle::before { content: "\f134"; } +.bi-arrow-right-short::before { content: "\f135"; } +.bi-arrow-right-square-fill::before { content: "\f136"; } +.bi-arrow-right-square::before { content: "\f137"; } +.bi-arrow-right::before { content: "\f138"; } +.bi-arrow-up-circle-fill::before { content: "\f139"; } +.bi-arrow-up-circle::before { content: "\f13a"; } +.bi-arrow-up-left-circle-fill::before { content: "\f13b"; } +.bi-arrow-up-left-circle::before { content: "\f13c"; } +.bi-arrow-up-left-square-fill::before { content: "\f13d"; } +.bi-arrow-up-left-square::before { content: "\f13e"; } +.bi-arrow-up-left::before { content: "\f13f"; } +.bi-arrow-up-right-circle-fill::before { content: "\f140"; } +.bi-arrow-up-right-circle::before { content: "\f141"; } +.bi-arrow-up-right-square-fill::before { content: "\f142"; } +.bi-arrow-up-right-square::before { content: "\f143"; } +.bi-arrow-up-right::before { content: "\f144"; } +.bi-arrow-up-short::before { content: "\f145"; } +.bi-arrow-up-square-fill::before { content: "\f146"; } +.bi-arrow-up-square::before { content: "\f147"; } +.bi-arrow-up::before { content: "\f148"; } +.bi-arrows-angle-contract::before { content: "\f149"; } +.bi-arrows-angle-expand::before { content: "\f14a"; } +.bi-arrows-collapse::before { content: "\f14b"; } +.bi-arrows-expand::before { content: "\f14c"; } +.bi-arrows-fullscreen::before { content: "\f14d"; } +.bi-arrows-move::before { content: "\f14e"; } +.bi-aspect-ratio-fill::before { content: "\f14f"; } +.bi-aspect-ratio::before { content: "\f150"; } +.bi-asterisk::before { content: "\f151"; } +.bi-at::before { content: "\f152"; } +.bi-award-fill::before { content: "\f153"; } +.bi-award::before { content: "\f154"; } +.bi-back::before { content: "\f155"; } +.bi-backspace-fill::before { content: "\f156"; } +.bi-backspace-reverse-fill::before { content: "\f157"; } +.bi-backspace-reverse::before { content: "\f158"; } +.bi-backspace::before { content: "\f159"; } +.bi-badge-3d-fill::before { content: "\f15a"; } +.bi-badge-3d::before { content: "\f15b"; } +.bi-badge-4k-fill::before { content: "\f15c"; } +.bi-badge-4k::before { content: "\f15d"; } +.bi-badge-8k-fill::before { content: "\f15e"; } +.bi-badge-8k::before { content: "\f15f"; } +.bi-badge-ad-fill::before { content: "\f160"; } +.bi-badge-ad::before { content: "\f161"; } +.bi-badge-ar-fill::before { content: "\f162"; } +.bi-badge-ar::before { content: "\f163"; } +.bi-badge-cc-fill::before { content: "\f164"; } +.bi-badge-cc::before { content: "\f165"; } +.bi-badge-hd-fill::before { content: "\f166"; } +.bi-badge-hd::before { content: "\f167"; } +.bi-badge-tm-fill::before { content: "\f168"; } +.bi-badge-tm::before { content: "\f169"; } +.bi-badge-vo-fill::before { content: "\f16a"; } +.bi-badge-vo::before { content: "\f16b"; } +.bi-badge-vr-fill::before { content: "\f16c"; } +.bi-badge-vr::before { content: "\f16d"; } +.bi-badge-wc-fill::before { content: "\f16e"; } +.bi-badge-wc::before { content: "\f16f"; } +.bi-bag-check-fill::before { content: "\f170"; } +.bi-bag-check::before { content: "\f171"; } +.bi-bag-dash-fill::before { content: "\f172"; } +.bi-bag-dash::before { content: "\f173"; } +.bi-bag-fill::before { content: "\f174"; } +.bi-bag-plus-fill::before { content: "\f175"; } +.bi-bag-plus::before { content: "\f176"; } +.bi-bag-x-fill::before { content: "\f177"; } +.bi-bag-x::before { content: "\f178"; } +.bi-bag::before { content: "\f179"; } +.bi-bar-chart-fill::before { content: "\f17a"; } +.bi-bar-chart-line-fill::before { content: "\f17b"; } +.bi-bar-chart-line::before { content: "\f17c"; } +.bi-bar-chart-steps::before { content: "\f17d"; } +.bi-bar-chart::before { content: "\f17e"; } +.bi-basket-fill::before { content: "\f17f"; } +.bi-basket::before { content: "\f180"; } +.bi-basket2-fill::before { content: "\f181"; } +.bi-basket2::before { content: "\f182"; } +.bi-basket3-fill::before { content: "\f183"; } +.bi-basket3::before { content: "\f184"; } +.bi-battery-charging::before { content: "\f185"; } +.bi-battery-full::before { content: "\f186"; } +.bi-battery-half::before { content: "\f187"; } +.bi-battery::before { content: "\f188"; } +.bi-bell-fill::before { content: "\f189"; } +.bi-bell::before { content: "\f18a"; } +.bi-bezier::before { content: "\f18b"; } +.bi-bezier2::before { content: "\f18c"; } +.bi-bicycle::before { content: "\f18d"; } +.bi-binoculars-fill::before { content: "\f18e"; } +.bi-binoculars::before { content: "\f18f"; } +.bi-blockquote-left::before { content: "\f190"; } +.bi-blockquote-right::before { content: "\f191"; } +.bi-book-fill::before { content: "\f192"; } +.bi-book-half::before { content: "\f193"; } +.bi-book::before { content: "\f194"; } +.bi-bookmark-check-fill::before { content: "\f195"; } +.bi-bookmark-check::before { content: "\f196"; } +.bi-bookmark-dash-fill::before { content: "\f197"; } +.bi-bookmark-dash::before { content: "\f198"; } +.bi-bookmark-fill::before { content: "\f199"; } +.bi-bookmark-heart-fill::before { content: "\f19a"; } +.bi-bookmark-heart::before { content: "\f19b"; } +.bi-bookmark-plus-fill::before { content: "\f19c"; } +.bi-bookmark-plus::before { content: "\f19d"; } +.bi-bookmark-star-fill::before { content: "\f19e"; } +.bi-bookmark-star::before { content: "\f19f"; } +.bi-bookmark-x-fill::before { content: "\f1a0"; } +.bi-bookmark-x::before { content: "\f1a1"; } +.bi-bookmark::before { content: "\f1a2"; } +.bi-bookmarks-fill::before { content: "\f1a3"; } +.bi-bookmarks::before { content: "\f1a4"; } +.bi-bookshelf::before { content: "\f1a5"; } +.bi-bootstrap-fill::before { content: "\f1a6"; } +.bi-bootstrap-reboot::before { content: "\f1a7"; } +.bi-bootstrap::before { content: "\f1a8"; } +.bi-border-all::before { content: "\f1a9"; } +.bi-border-bottom::before { content: "\f1aa"; } +.bi-border-center::before { content: "\f1ab"; } +.bi-border-inner::before { content: "\f1ac"; } +.bi-border-left::before { content: "\f1ad"; } +.bi-border-middle::before { content: "\f1ae"; } +.bi-border-outer::before { content: "\f1af"; } +.bi-border-right::before { content: "\f1b0"; } +.bi-border-style::before { content: "\f1b1"; } +.bi-border-top::before { content: "\f1b2"; } +.bi-border-width::before { content: "\f1b3"; } +.bi-border::before { content: "\f1b4"; } +.bi-bounding-box-circles::before { content: "\f1b5"; } +.bi-bounding-box::before { content: "\f1b6"; } +.bi-box-arrow-down-left::before { content: "\f1b7"; } +.bi-box-arrow-down-right::before { content: "\f1b8"; } +.bi-box-arrow-down::before { content: "\f1b9"; } +.bi-box-arrow-in-down-left::before { content: "\f1ba"; } +.bi-box-arrow-in-down-right::before { content: "\f1bb"; } +.bi-box-arrow-in-down::before { content: "\f1bc"; } +.bi-box-arrow-in-left::before { content: "\f1bd"; } +.bi-box-arrow-in-right::before { content: "\f1be"; } +.bi-box-arrow-in-up-left::before { content: "\f1bf"; } +.bi-box-arrow-in-up-right::before { content: "\f1c0"; } +.bi-box-arrow-in-up::before { content: "\f1c1"; } +.bi-box-arrow-left::before { content: "\f1c2"; } +.bi-box-arrow-right::before { content: "\f1c3"; } +.bi-box-arrow-up-left::before { content: "\f1c4"; } +.bi-box-arrow-up-right::before { content: "\f1c5"; } +.bi-box-arrow-up::before { content: "\f1c6"; } +.bi-box-seam::before { content: "\f1c7"; } +.bi-box::before { content: "\f1c8"; } +.bi-braces::before { content: "\f1c9"; } +.bi-bricks::before { content: "\f1ca"; } +.bi-briefcase-fill::before { content: "\f1cb"; } +.bi-briefcase::before { content: "\f1cc"; } +.bi-brightness-alt-high-fill::before { content: "\f1cd"; } +.bi-brightness-alt-high::before { content: "\f1ce"; } +.bi-brightness-alt-low-fill::before { content: "\f1cf"; } +.bi-brightness-alt-low::before { content: "\f1d0"; } +.bi-brightness-high-fill::before { content: "\f1d1"; } +.bi-brightness-high::before { content: "\f1d2"; } +.bi-brightness-low-fill::before { content: "\f1d3"; } +.bi-brightness-low::before { content: "\f1d4"; } +.bi-broadcast-pin::before { content: "\f1d5"; } +.bi-broadcast::before { content: "\f1d6"; } +.bi-brush-fill::before { content: "\f1d7"; } +.bi-brush::before { content: "\f1d8"; } +.bi-bucket-fill::before { content: "\f1d9"; } +.bi-bucket::before { content: "\f1da"; } +.bi-bug-fill::before { content: "\f1db"; } +.bi-bug::before { content: "\f1dc"; } +.bi-building::before { content: "\f1dd"; } +.bi-bullseye::before { content: "\f1de"; } +.bi-calculator-fill::before { content: "\f1df"; } +.bi-calculator::before { content: "\f1e0"; } +.bi-calendar-check-fill::before { content: "\f1e1"; } +.bi-calendar-check::before { content: "\f1e2"; } +.bi-calendar-date-fill::before { content: "\f1e3"; } +.bi-calendar-date::before { content: "\f1e4"; } +.bi-calendar-day-fill::before { content: "\f1e5"; } +.bi-calendar-day::before { content: "\f1e6"; } +.bi-calendar-event-fill::before { content: "\f1e7"; } +.bi-calendar-event::before { content: "\f1e8"; } +.bi-calendar-fill::before { content: "\f1e9"; } +.bi-calendar-minus-fill::before { content: "\f1ea"; } +.bi-calendar-minus::before { content: "\f1eb"; } +.bi-calendar-month-fill::before { content: "\f1ec"; } +.bi-calendar-month::before { content: "\f1ed"; } +.bi-calendar-plus-fill::before { content: "\f1ee"; } +.bi-calendar-plus::before { content: "\f1ef"; } +.bi-calendar-range-fill::before { content: "\f1f0"; } +.bi-calendar-range::before { content: "\f1f1"; } +.bi-calendar-week-fill::before { content: "\f1f2"; } +.bi-calendar-week::before { content: "\f1f3"; } +.bi-calendar-x-fill::before { content: "\f1f4"; } +.bi-calendar-x::before { content: "\f1f5"; } +.bi-calendar::before { content: "\f1f6"; } +.bi-calendar2-check-fill::before { content: "\f1f7"; } +.bi-calendar2-check::before { content: "\f1f8"; } +.bi-calendar2-date-fill::before { content: "\f1f9"; } +.bi-calendar2-date::before { content: "\f1fa"; } +.bi-calendar2-day-fill::before { content: "\f1fb"; } +.bi-calendar2-day::before { content: "\f1fc"; } +.bi-calendar2-event-fill::before { content: "\f1fd"; } +.bi-calendar2-event::before { content: "\f1fe"; } +.bi-calendar2-fill::before { content: "\f1ff"; } +.bi-calendar2-minus-fill::before { content: "\f200"; } +.bi-calendar2-minus::before { content: "\f201"; } +.bi-calendar2-month-fill::before { content: "\f202"; } +.bi-calendar2-month::before { content: "\f203"; } +.bi-calendar2-plus-fill::before { content: "\f204"; } +.bi-calendar2-plus::before { content: "\f205"; } +.bi-calendar2-range-fill::before { content: "\f206"; } +.bi-calendar2-range::before { content: "\f207"; } +.bi-calendar2-week-fill::before { content: "\f208"; } +.bi-calendar2-week::before { content: "\f209"; } +.bi-calendar2-x-fill::before { content: "\f20a"; } +.bi-calendar2-x::before { content: "\f20b"; } +.bi-calendar2::before { content: "\f20c"; } +.bi-calendar3-event-fill::before { content: "\f20d"; } +.bi-calendar3-event::before { content: "\f20e"; } +.bi-calendar3-fill::before { content: "\f20f"; } +.bi-calendar3-range-fill::before { content: "\f210"; } +.bi-calendar3-range::before { content: "\f211"; } +.bi-calendar3-week-fill::before { content: "\f212"; } +.bi-calendar3-week::before { content: "\f213"; } +.bi-calendar3::before { content: "\f214"; } +.bi-calendar4-event::before { content: "\f215"; } +.bi-calendar4-range::before { content: "\f216"; } +.bi-calendar4-week::before { content: "\f217"; } +.bi-calendar4::before { content: "\f218"; } +.bi-camera-fill::before { content: "\f219"; } +.bi-camera-reels-fill::before { content: "\f21a"; } +.bi-camera-reels::before { content: "\f21b"; } +.bi-camera-video-fill::before { content: "\f21c"; } +.bi-camera-video-off-fill::before { content: "\f21d"; } +.bi-camera-video-off::before { content: "\f21e"; } +.bi-camera-video::before { content: "\f21f"; } +.bi-camera::before { content: "\f220"; } +.bi-camera2::before { content: "\f221"; } +.bi-capslock-fill::before { content: "\f222"; } +.bi-capslock::before { content: "\f223"; } +.bi-card-checklist::before { content: "\f224"; } +.bi-card-heading::before { content: "\f225"; } +.bi-card-image::before { content: "\f226"; } +.bi-card-list::before { content: "\f227"; } +.bi-card-text::before { content: "\f228"; } +.bi-caret-down-fill::before { content: "\f229"; } +.bi-caret-down-square-fill::before { content: "\f22a"; } +.bi-caret-down-square::before { content: "\f22b"; } +.bi-caret-down::before { content: "\f22c"; } +.bi-caret-left-fill::before { content: "\f22d"; } +.bi-caret-left-square-fill::before { content: "\f22e"; } +.bi-caret-left-square::before { content: "\f22f"; } +.bi-caret-left::before { content: "\f230"; } +.bi-caret-right-fill::before { content: "\f231"; } +.bi-caret-right-square-fill::before { content: "\f232"; } +.bi-caret-right-square::before { content: "\f233"; } +.bi-caret-right::before { content: "\f234"; } +.bi-caret-up-fill::before { content: "\f235"; } +.bi-caret-up-square-fill::before { content: "\f236"; } +.bi-caret-up-square::before { content: "\f237"; } +.bi-caret-up::before { content: "\f238"; } +.bi-cart-check-fill::before { content: "\f239"; } +.bi-cart-check::before { content: "\f23a"; } +.bi-cart-dash-fill::before { content: "\f23b"; } +.bi-cart-dash::before { content: "\f23c"; } +.bi-cart-fill::before { content: "\f23d"; } +.bi-cart-plus-fill::before { content: "\f23e"; } +.bi-cart-plus::before { content: "\f23f"; } +.bi-cart-x-fill::before { content: "\f240"; } +.bi-cart-x::before { content: "\f241"; } +.bi-cart::before { content: "\f242"; } +.bi-cart2::before { content: "\f243"; } +.bi-cart3::before { content: "\f244"; } +.bi-cart4::before { content: "\f245"; } +.bi-cash-stack::before { content: "\f246"; } +.bi-cash::before { content: "\f247"; } +.bi-cast::before { content: "\f248"; } +.bi-chat-dots-fill::before { content: "\f249"; } +.bi-chat-dots::before { content: "\f24a"; } +.bi-chat-fill::before { content: "\f24b"; } +.bi-chat-left-dots-fill::before { content: "\f24c"; } +.bi-chat-left-dots::before { content: "\f24d"; } +.bi-chat-left-fill::before { content: "\f24e"; } +.bi-chat-left-quote-fill::before { content: "\f24f"; } +.bi-chat-left-quote::before { content: "\f250"; } +.bi-chat-left-text-fill::before { content: "\f251"; } +.bi-chat-left-text::before { content: "\f252"; } +.bi-chat-left::before { content: "\f253"; } +.bi-chat-quote-fill::before { content: "\f254"; } +.bi-chat-quote::before { content: "\f255"; } +.bi-chat-right-dots-fill::before { content: "\f256"; } +.bi-chat-right-dots::before { content: "\f257"; } +.bi-chat-right-fill::before { content: "\f258"; } +.bi-chat-right-quote-fill::before { content: "\f259"; } +.bi-chat-right-quote::before { content: "\f25a"; } +.bi-chat-right-text-fill::before { content: "\f25b"; } +.bi-chat-right-text::before { content: "\f25c"; } +.bi-chat-right::before { content: "\f25d"; } +.bi-chat-square-dots-fill::before { content: "\f25e"; } +.bi-chat-square-dots::before { content: "\f25f"; } +.bi-chat-square-fill::before { content: "\f260"; } +.bi-chat-square-quote-fill::before { content: "\f261"; } +.bi-chat-square-quote::before { content: "\f262"; } +.bi-chat-square-text-fill::before { content: "\f263"; } +.bi-chat-square-text::before { content: "\f264"; } +.bi-chat-square::before { content: "\f265"; } +.bi-chat-text-fill::before { content: "\f266"; } +.bi-chat-text::before { content: "\f267"; } +.bi-chat::before { content: "\f268"; } +.bi-check-all::before { content: "\f269"; } +.bi-check-circle-fill::before { content: "\f26a"; } +.bi-check-circle::before { content: "\f26b"; } +.bi-check-square-fill::before { content: "\f26c"; } +.bi-check-square::before { content: "\f26d"; } +.bi-check::before { content: "\f26e"; } +.bi-check2-all::before { content: "\f26f"; } +.bi-check2-circle::before { content: "\f270"; } +.bi-check2-square::before { content: "\f271"; } +.bi-check2::before { content: "\f272"; } +.bi-chevron-bar-contract::before { content: "\f273"; } +.bi-chevron-bar-down::before { content: "\f274"; } +.bi-chevron-bar-expand::before { content: "\f275"; } +.bi-chevron-bar-left::before { content: "\f276"; } +.bi-chevron-bar-right::before { content: "\f277"; } +.bi-chevron-bar-up::before { content: "\f278"; } +.bi-chevron-compact-down::before { content: "\f279"; } +.bi-chevron-compact-left::before { content: "\f27a"; } +.bi-chevron-compact-right::before { content: "\f27b"; } +.bi-chevron-compact-up::before { content: "\f27c"; } +.bi-chevron-contract::before { content: "\f27d"; } +.bi-chevron-double-down::before { content: "\f27e"; } +.bi-chevron-double-left::before { content: "\f27f"; } +.bi-chevron-double-right::before { content: "\f280"; } +.bi-chevron-double-up::before { content: "\f281"; } +.bi-chevron-down::before { content: "\f282"; } +.bi-chevron-expand::before { content: "\f283"; } +.bi-chevron-left::before { content: "\f284"; } +.bi-chevron-right::before { content: "\f285"; } +.bi-chevron-up::before { content: "\f286"; } +.bi-circle-fill::before { content: "\f287"; } +.bi-circle-half::before { content: "\f288"; } +.bi-circle-square::before { content: "\f289"; } +.bi-circle::before { content: "\f28a"; } +.bi-clipboard-check::before { content: "\f28b"; } +.bi-clipboard-data::before { content: "\f28c"; } +.bi-clipboard-minus::before { content: "\f28d"; } +.bi-clipboard-plus::before { content: "\f28e"; } +.bi-clipboard-x::before { content: "\f28f"; } +.bi-clipboard::before { content: "\f290"; } +.bi-clock-fill::before { content: "\f291"; } +.bi-clock-history::before { content: "\f292"; } +.bi-clock::before { content: "\f293"; } +.bi-cloud-arrow-down-fill::before { content: "\f294"; } +.bi-cloud-arrow-down::before { content: "\f295"; } +.bi-cloud-arrow-up-fill::before { content: "\f296"; } +.bi-cloud-arrow-up::before { content: "\f297"; } +.bi-cloud-check-fill::before { content: "\f298"; } +.bi-cloud-check::before { content: "\f299"; } +.bi-cloud-download-fill::before { content: "\f29a"; } +.bi-cloud-download::before { content: "\f29b"; } +.bi-cloud-drizzle-fill::before { content: "\f29c"; } +.bi-cloud-drizzle::before { content: "\f29d"; } +.bi-cloud-fill::before { content: "\f29e"; } +.bi-cloud-fog-fill::before { content: "\f29f"; } +.bi-cloud-fog::before { content: "\f2a0"; } +.bi-cloud-fog2-fill::before { content: "\f2a1"; } +.bi-cloud-fog2::before { content: "\f2a2"; } +.bi-cloud-hail-fill::before { content: "\f2a3"; } +.bi-cloud-hail::before { content: "\f2a4"; } +.bi-cloud-haze-fill::before { content: "\f2a6"; } +.bi-cloud-haze::before { content: "\f2a7"; } +.bi-cloud-haze2-fill::before { content: "\f2a8"; } +.bi-cloud-lightning-fill::before { content: "\f2a9"; } +.bi-cloud-lightning-rain-fill::before { content: "\f2aa"; } +.bi-cloud-lightning-rain::before { content: "\f2ab"; } +.bi-cloud-lightning::before { content: "\f2ac"; } +.bi-cloud-minus-fill::before { content: "\f2ad"; } +.bi-cloud-minus::before { content: "\f2ae"; } +.bi-cloud-moon-fill::before { content: "\f2af"; } +.bi-cloud-moon::before { content: "\f2b0"; } +.bi-cloud-plus-fill::before { content: "\f2b1"; } +.bi-cloud-plus::before { content: "\f2b2"; } +.bi-cloud-rain-fill::before { content: "\f2b3"; } +.bi-cloud-rain-heavy-fill::before { content: "\f2b4"; } +.bi-cloud-rain-heavy::before { content: "\f2b5"; } +.bi-cloud-rain::before { content: "\f2b6"; } +.bi-cloud-slash-fill::before { content: "\f2b7"; } +.bi-cloud-slash::before { content: "\f2b8"; } +.bi-cloud-sleet-fill::before { content: "\f2b9"; } +.bi-cloud-sleet::before { content: "\f2ba"; } +.bi-cloud-snow-fill::before { content: "\f2bb"; } +.bi-cloud-snow::before { content: "\f2bc"; } +.bi-cloud-sun-fill::before { content: "\f2bd"; } +.bi-cloud-sun::before { content: "\f2be"; } +.bi-cloud-upload-fill::before { content: "\f2bf"; } +.bi-cloud-upload::before { content: "\f2c0"; } +.bi-cloud::before { content: "\f2c1"; } +.bi-clouds-fill::before { content: "\f2c2"; } +.bi-clouds::before { content: "\f2c3"; } +.bi-cloudy-fill::before { content: "\f2c4"; } +.bi-cloudy::before { content: "\f2c5"; } +.bi-code-slash::before { content: "\f2c6"; } +.bi-code-square::before { content: "\f2c7"; } +.bi-code::before { content: "\f2c8"; } +.bi-collection-fill::before { content: "\f2c9"; } +.bi-collection-play-fill::before { content: "\f2ca"; } +.bi-collection-play::before { content: "\f2cb"; } +.bi-collection::before { content: "\f2cc"; } +.bi-columns-gap::before { content: "\f2cd"; } +.bi-columns::before { content: "\f2ce"; } +.bi-command::before { content: "\f2cf"; } +.bi-compass-fill::before { content: "\f2d0"; } +.bi-compass::before { content: "\f2d1"; } +.bi-cone-striped::before { content: "\f2d2"; } +.bi-cone::before { content: "\f2d3"; } +.bi-controller::before { content: "\f2d4"; } +.bi-cpu-fill::before { content: "\f2d5"; } +.bi-cpu::before { content: "\f2d6"; } +.bi-credit-card-2-back-fill::before { content: "\f2d7"; } +.bi-credit-card-2-back::before { content: "\f2d8"; } +.bi-credit-card-2-front-fill::before { content: "\f2d9"; } +.bi-credit-card-2-front::before { content: "\f2da"; } +.bi-credit-card-fill::before { content: "\f2db"; } +.bi-credit-card::before { content: "\f2dc"; } +.bi-crop::before { content: "\f2dd"; } +.bi-cup-fill::before { content: "\f2de"; } +.bi-cup-straw::before { content: "\f2df"; } +.bi-cup::before { content: "\f2e0"; } +.bi-cursor-fill::before { content: "\f2e1"; } +.bi-cursor-text::before { content: "\f2e2"; } +.bi-cursor::before { content: "\f2e3"; } +.bi-dash-circle-dotted::before { content: "\f2e4"; } +.bi-dash-circle-fill::before { content: "\f2e5"; } +.bi-dash-circle::before { content: "\f2e6"; } +.bi-dash-square-dotted::before { content: "\f2e7"; } +.bi-dash-square-fill::before { content: "\f2e8"; } +.bi-dash-square::before { content: "\f2e9"; } +.bi-dash::before { content: "\f2ea"; } +.bi-diagram-2-fill::before { content: "\f2eb"; } +.bi-diagram-2::before { content: "\f2ec"; } +.bi-diagram-3-fill::before { content: "\f2ed"; } +.bi-diagram-3::before { content: "\f2ee"; } +.bi-diamond-fill::before { content: "\f2ef"; } +.bi-diamond-half::before { content: "\f2f0"; } +.bi-diamond::before { content: "\f2f1"; } +.bi-dice-1-fill::before { content: "\f2f2"; } +.bi-dice-1::before { content: "\f2f3"; } +.bi-dice-2-fill::before { content: "\f2f4"; } +.bi-dice-2::before { content: "\f2f5"; } +.bi-dice-3-fill::before { content: "\f2f6"; } +.bi-dice-3::before { content: "\f2f7"; } +.bi-dice-4-fill::before { content: "\f2f8"; } +.bi-dice-4::before { content: "\f2f9"; } +.bi-dice-5-fill::before { content: "\f2fa"; } +.bi-dice-5::before { content: "\f2fb"; } +.bi-dice-6-fill::before { content: "\f2fc"; } +.bi-dice-6::before { content: "\f2fd"; } +.bi-disc-fill::before { content: "\f2fe"; } +.bi-disc::before { content: "\f2ff"; } +.bi-discord::before { content: "\f300"; } +.bi-display-fill::before { content: "\f301"; } +.bi-display::before { content: "\f302"; } +.bi-distribute-horizontal::before { content: "\f303"; } +.bi-distribute-vertical::before { content: "\f304"; } +.bi-door-closed-fill::before { content: "\f305"; } +.bi-door-closed::before { content: "\f306"; } +.bi-door-open-fill::before { content: "\f307"; } +.bi-door-open::before { content: "\f308"; } +.bi-dot::before { content: "\f309"; } +.bi-download::before { content: "\f30a"; } +.bi-droplet-fill::before { content: "\f30b"; } +.bi-droplet-half::before { content: "\f30c"; } +.bi-droplet::before { content: "\f30d"; } +.bi-earbuds::before { content: "\f30e"; } +.bi-easel-fill::before { content: "\f30f"; } +.bi-easel::before { content: "\f310"; } +.bi-egg-fill::before { content: "\f311"; } +.bi-egg-fried::before { content: "\f312"; } +.bi-egg::before { content: "\f313"; } +.bi-eject-fill::before { content: "\f314"; } +.bi-eject::before { content: "\f315"; } +.bi-emoji-angry-fill::before { content: "\f316"; } +.bi-emoji-angry::before { content: "\f317"; } +.bi-emoji-dizzy-fill::before { content: "\f318"; } +.bi-emoji-dizzy::before { content: "\f319"; } +.bi-emoji-expressionless-fill::before { content: "\f31a"; } +.bi-emoji-expressionless::before { content: "\f31b"; } +.bi-emoji-frown-fill::before { content: "\f31c"; } +.bi-emoji-frown::before { content: "\f31d"; } +.bi-emoji-heart-eyes-fill::before { content: "\f31e"; } +.bi-emoji-heart-eyes::before { content: "\f31f"; } +.bi-emoji-laughing-fill::before { content: "\f320"; } +.bi-emoji-laughing::before { content: "\f321"; } +.bi-emoji-neutral-fill::before { content: "\f322"; } +.bi-emoji-neutral::before { content: "\f323"; } +.bi-emoji-smile-fill::before { content: "\f324"; } +.bi-emoji-smile-upside-down-fill::before { content: "\f325"; } +.bi-emoji-smile-upside-down::before { content: "\f326"; } +.bi-emoji-smile::before { content: "\f327"; } +.bi-emoji-sunglasses-fill::before { content: "\f328"; } +.bi-emoji-sunglasses::before { content: "\f329"; } +.bi-emoji-wink-fill::before { content: "\f32a"; } +.bi-emoji-wink::before { content: "\f32b"; } +.bi-envelope-fill::before { content: "\f32c"; } +.bi-envelope-open-fill::before { content: "\f32d"; } +.bi-envelope-open::before { content: "\f32e"; } +.bi-envelope::before { content: "\f32f"; } +.bi-eraser-fill::before { content: "\f330"; } +.bi-eraser::before { content: "\f331"; } +.bi-exclamation-circle-fill::before { content: "\f332"; } +.bi-exclamation-circle::before { content: "\f333"; } +.bi-exclamation-diamond-fill::before { content: "\f334"; } +.bi-exclamation-diamond::before { content: "\f335"; } +.bi-exclamation-octagon-fill::before { content: "\f336"; } +.bi-exclamation-octagon::before { content: "\f337"; } +.bi-exclamation-square-fill::before { content: "\f338"; } +.bi-exclamation-square::before { content: "\f339"; } +.bi-exclamation-triangle-fill::before { content: "\f33a"; } +.bi-exclamation-triangle::before { content: "\f33b"; } +.bi-exclamation::before { content: "\f33c"; } +.bi-exclude::before { content: "\f33d"; } +.bi-eye-fill::before { content: "\f33e"; } +.bi-eye-slash-fill::before { content: "\f33f"; } +.bi-eye-slash::before { content: "\f340"; } +.bi-eye::before { content: "\f341"; } +.bi-eyedropper::before { content: "\f342"; } +.bi-eyeglasses::before { content: "\f343"; } +.bi-facebook::before { content: "\f344"; } +.bi-file-arrow-down-fill::before { content: "\f345"; } +.bi-file-arrow-down::before { content: "\f346"; } +.bi-file-arrow-up-fill::before { content: "\f347"; } +.bi-file-arrow-up::before { content: "\f348"; } +.bi-file-bar-graph-fill::before { content: "\f349"; } +.bi-file-bar-graph::before { content: "\f34a"; } +.bi-file-binary-fill::before { content: "\f34b"; } +.bi-file-binary::before { content: "\f34c"; } +.bi-file-break-fill::before { content: "\f34d"; } +.bi-file-break::before { content: "\f34e"; } +.bi-file-check-fill::before { content: "\f34f"; } +.bi-file-check::before { content: "\f350"; } +.bi-file-code-fill::before { content: "\f351"; } +.bi-file-code::before { content: "\f352"; } +.bi-file-diff-fill::before { content: "\f353"; } +.bi-file-diff::before { content: "\f354"; } +.bi-file-earmark-arrow-down-fill::before { content: "\f355"; } +.bi-file-earmark-arrow-down::before { content: "\f356"; } +.bi-file-earmark-arrow-up-fill::before { content: "\f357"; } +.bi-file-earmark-arrow-up::before { content: "\f358"; } +.bi-file-earmark-bar-graph-fill::before { content: "\f359"; } +.bi-file-earmark-bar-graph::before { content: "\f35a"; } +.bi-file-earmark-binary-fill::before { content: "\f35b"; } +.bi-file-earmark-binary::before { content: "\f35c"; } +.bi-file-earmark-break-fill::before { content: "\f35d"; } +.bi-file-earmark-break::before { content: "\f35e"; } +.bi-file-earmark-check-fill::before { content: "\f35f"; } +.bi-file-earmark-check::before { content: "\f360"; } +.bi-file-earmark-code-fill::before { content: "\f361"; } +.bi-file-earmark-code::before { content: "\f362"; } +.bi-file-earmark-diff-fill::before { content: "\f363"; } +.bi-file-earmark-diff::before { content: "\f364"; } +.bi-file-earmark-easel-fill::before { content: "\f365"; } +.bi-file-earmark-easel::before { content: "\f366"; } +.bi-file-earmark-excel-fill::before { content: "\f367"; } +.bi-file-earmark-excel::before { content: "\f368"; } +.bi-file-earmark-fill::before { content: "\f369"; } +.bi-file-earmark-font-fill::before { content: "\f36a"; } +.bi-file-earmark-font::before { content: "\f36b"; } +.bi-file-earmark-image-fill::before { content: "\f36c"; } +.bi-file-earmark-image::before { content: "\f36d"; } +.bi-file-earmark-lock-fill::before { content: "\f36e"; } +.bi-file-earmark-lock::before { content: "\f36f"; } +.bi-file-earmark-lock2-fill::before { content: "\f370"; } +.bi-file-earmark-lock2::before { content: "\f371"; } +.bi-file-earmark-medical-fill::before { content: "\f372"; } +.bi-file-earmark-medical::before { content: "\f373"; } +.bi-file-earmark-minus-fill::before { content: "\f374"; } +.bi-file-earmark-minus::before { content: "\f375"; } +.bi-file-earmark-music-fill::before { content: "\f376"; } +.bi-file-earmark-music::before { content: "\f377"; } +.bi-file-earmark-person-fill::before { content: "\f378"; } +.bi-file-earmark-person::before { content: "\f379"; } +.bi-file-earmark-play-fill::before { content: "\f37a"; } +.bi-file-earmark-play::before { content: "\f37b"; } +.bi-file-earmark-plus-fill::before { content: "\f37c"; } +.bi-file-earmark-plus::before { content: "\f37d"; } +.bi-file-earmark-post-fill::before { content: "\f37e"; } +.bi-file-earmark-post::before { content: "\f37f"; } +.bi-file-earmark-ppt-fill::before { content: "\f380"; } +.bi-file-earmark-ppt::before { content: "\f381"; } +.bi-file-earmark-richtext-fill::before { content: "\f382"; } +.bi-file-earmark-richtext::before { content: "\f383"; } +.bi-file-earmark-ruled-fill::before { content: "\f384"; } +.bi-file-earmark-ruled::before { content: "\f385"; } +.bi-file-earmark-slides-fill::before { content: "\f386"; } +.bi-file-earmark-slides::before { content: "\f387"; } +.bi-file-earmark-spreadsheet-fill::before { content: "\f388"; } +.bi-file-earmark-spreadsheet::before { content: "\f389"; } +.bi-file-earmark-text-fill::before { content: "\f38a"; } +.bi-file-earmark-text::before { content: "\f38b"; } +.bi-file-earmark-word-fill::before { content: "\f38c"; } +.bi-file-earmark-word::before { content: "\f38d"; } +.bi-file-earmark-x-fill::before { content: "\f38e"; } +.bi-file-earmark-x::before { content: "\f38f"; } +.bi-file-earmark-zip-fill::before { content: "\f390"; } +.bi-file-earmark-zip::before { content: "\f391"; } +.bi-file-earmark::before { content: "\f392"; } +.bi-file-easel-fill::before { content: "\f393"; } +.bi-file-easel::before { content: "\f394"; } +.bi-file-excel-fill::before { content: "\f395"; } +.bi-file-excel::before { content: "\f396"; } +.bi-file-fill::before { content: "\f397"; } +.bi-file-font-fill::before { content: "\f398"; } +.bi-file-font::before { content: "\f399"; } +.bi-file-image-fill::before { content: "\f39a"; } +.bi-file-image::before { content: "\f39b"; } +.bi-file-lock-fill::before { content: "\f39c"; } +.bi-file-lock::before { content: "\f39d"; } +.bi-file-lock2-fill::before { content: "\f39e"; } +.bi-file-lock2::before { content: "\f39f"; } +.bi-file-medical-fill::before { content: "\f3a0"; } +.bi-file-medical::before { content: "\f3a1"; } +.bi-file-minus-fill::before { content: "\f3a2"; } +.bi-file-minus::before { content: "\f3a3"; } +.bi-file-music-fill::before { content: "\f3a4"; } +.bi-file-music::before { content: "\f3a5"; } +.bi-file-person-fill::before { content: "\f3a6"; } +.bi-file-person::before { content: "\f3a7"; } +.bi-file-play-fill::before { content: "\f3a8"; } +.bi-file-play::before { content: "\f3a9"; } +.bi-file-plus-fill::before { content: "\f3aa"; } +.bi-file-plus::before { content: "\f3ab"; } +.bi-file-post-fill::before { content: "\f3ac"; } +.bi-file-post::before { content: "\f3ad"; } +.bi-file-ppt-fill::before { content: "\f3ae"; } +.bi-file-ppt::before { content: "\f3af"; } +.bi-file-richtext-fill::before { content: "\f3b0"; } +.bi-file-richtext::before { content: "\f3b1"; } +.bi-file-ruled-fill::before { content: "\f3b2"; } +.bi-file-ruled::before { content: "\f3b3"; } +.bi-file-slides-fill::before { content: "\f3b4"; } +.bi-file-slides::before { content: "\f3b5"; } +.bi-file-spreadsheet-fill::before { content: "\f3b6"; } +.bi-file-spreadsheet::before { content: "\f3b7"; } +.bi-file-text-fill::before { content: "\f3b8"; } +.bi-file-text::before { content: "\f3b9"; } +.bi-file-word-fill::before { content: "\f3ba"; } +.bi-file-word::before { content: "\f3bb"; } +.bi-file-x-fill::before { content: "\f3bc"; } +.bi-file-x::before { content: "\f3bd"; } +.bi-file-zip-fill::before { content: "\f3be"; } +.bi-file-zip::before { content: "\f3bf"; } +.bi-file::before { content: "\f3c0"; } +.bi-files-alt::before { content: "\f3c1"; } +.bi-files::before { content: "\f3c2"; } +.bi-film::before { content: "\f3c3"; } +.bi-filter-circle-fill::before { content: "\f3c4"; } +.bi-filter-circle::before { content: "\f3c5"; } +.bi-filter-left::before { content: "\f3c6"; } +.bi-filter-right::before { content: "\f3c7"; } +.bi-filter-square-fill::before { content: "\f3c8"; } +.bi-filter-square::before { content: "\f3c9"; } +.bi-filter::before { content: "\f3ca"; } +.bi-flag-fill::before { content: "\f3cb"; } +.bi-flag::before { content: "\f3cc"; } +.bi-flower1::before { content: "\f3cd"; } +.bi-flower2::before { content: "\f3ce"; } +.bi-flower3::before { content: "\f3cf"; } +.bi-folder-check::before { content: "\f3d0"; } +.bi-folder-fill::before { content: "\f3d1"; } +.bi-folder-minus::before { content: "\f3d2"; } +.bi-folder-plus::before { content: "\f3d3"; } +.bi-folder-symlink-fill::before { content: "\f3d4"; } +.bi-folder-symlink::before { content: "\f3d5"; } +.bi-folder-x::before { content: "\f3d6"; } +.bi-folder::before { content: "\f3d7"; } +.bi-folder2-open::before { content: "\f3d8"; } +.bi-folder2::before { content: "\f3d9"; } +.bi-fonts::before { content: "\f3da"; } +.bi-forward-fill::before { content: "\f3db"; } +.bi-forward::before { content: "\f3dc"; } +.bi-front::before { content: "\f3dd"; } +.bi-fullscreen-exit::before { content: "\f3de"; } +.bi-fullscreen::before { content: "\f3df"; } +.bi-funnel-fill::before { content: "\f3e0"; } +.bi-funnel::before { content: "\f3e1"; } +.bi-gear-fill::before { content: "\f3e2"; } +.bi-gear-wide-connected::before { content: "\f3e3"; } +.bi-gear-wide::before { content: "\f3e4"; } +.bi-gear::before { content: "\f3e5"; } +.bi-gem::before { content: "\f3e6"; } +.bi-geo-alt-fill::before { content: "\f3e7"; } +.bi-geo-alt::before { content: "\f3e8"; } +.bi-geo-fill::before { content: "\f3e9"; } +.bi-geo::before { content: "\f3ea"; } +.bi-gift-fill::before { content: "\f3eb"; } +.bi-gift::before { content: "\f3ec"; } +.bi-github::before { content: "\f3ed"; } +.bi-globe::before { content: "\f3ee"; } +.bi-globe2::before { content: "\f3ef"; } +.bi-google::before { content: "\f3f0"; } +.bi-graph-down::before { content: "\f3f1"; } +.bi-graph-up::before { content: "\f3f2"; } +.bi-grid-1x2-fill::before { content: "\f3f3"; } +.bi-grid-1x2::before { content: "\f3f4"; } +.bi-grid-3x2-gap-fill::before { content: "\f3f5"; } +.bi-grid-3x2-gap::before { content: "\f3f6"; } +.bi-grid-3x2::before { content: "\f3f7"; } +.bi-grid-3x3-gap-fill::before { content: "\f3f8"; } +.bi-grid-3x3-gap::before { content: "\f3f9"; } +.bi-grid-3x3::before { content: "\f3fa"; } +.bi-grid-fill::before { content: "\f3fb"; } +.bi-grid::before { content: "\f3fc"; } +.bi-grip-horizontal::before { content: "\f3fd"; } +.bi-grip-vertical::before { content: "\f3fe"; } +.bi-hammer::before { content: "\f3ff"; } +.bi-hand-index-fill::before { content: "\f400"; } +.bi-hand-index-thumb-fill::before { content: "\f401"; } +.bi-hand-index-thumb::before { content: "\f402"; } +.bi-hand-index::before { content: "\f403"; } +.bi-hand-thumbs-down-fill::before { content: "\f404"; } +.bi-hand-thumbs-down::before { content: "\f405"; } +.bi-hand-thumbs-up-fill::before { content: "\f406"; } +.bi-hand-thumbs-up::before { content: "\f407"; } +.bi-handbag-fill::before { content: "\f408"; } +.bi-handbag::before { content: "\f409"; } +.bi-hash::before { content: "\f40a"; } +.bi-hdd-fill::before { content: "\f40b"; } +.bi-hdd-network-fill::before { content: "\f40c"; } +.bi-hdd-network::before { content: "\f40d"; } +.bi-hdd-rack-fill::before { content: "\f40e"; } +.bi-hdd-rack::before { content: "\f40f"; } +.bi-hdd-stack-fill::before { content: "\f410"; } +.bi-hdd-stack::before { content: "\f411"; } +.bi-hdd::before { content: "\f412"; } +.bi-headphones::before { content: "\f413"; } +.bi-headset::before { content: "\f414"; } +.bi-heart-fill::before { content: "\f415"; } +.bi-heart-half::before { content: "\f416"; } +.bi-heart::before { content: "\f417"; } +.bi-heptagon-fill::before { content: "\f418"; } +.bi-heptagon-half::before { content: "\f419"; } +.bi-heptagon::before { content: "\f41a"; } +.bi-hexagon-fill::before { content: "\f41b"; } +.bi-hexagon-half::before { content: "\f41c"; } +.bi-hexagon::before { content: "\f41d"; } +.bi-hourglass-bottom::before { content: "\f41e"; } +.bi-hourglass-split::before { content: "\f41f"; } +.bi-hourglass-top::before { content: "\f420"; } +.bi-hourglass::before { content: "\f421"; } +.bi-house-door-fill::before { content: "\f422"; } +.bi-house-door::before { content: "\f423"; } +.bi-house-fill::before { content: "\f424"; } +.bi-house::before { content: "\f425"; } +.bi-hr::before { content: "\f426"; } +.bi-hurricane::before { content: "\f427"; } +.bi-image-alt::before { content: "\f428"; } +.bi-image-fill::before { content: "\f429"; } +.bi-image::before { content: "\f42a"; } +.bi-images::before { content: "\f42b"; } +.bi-inbox-fill::before { content: "\f42c"; } +.bi-inbox::before { content: "\f42d"; } +.bi-inboxes-fill::before { content: "\f42e"; } +.bi-inboxes::before { content: "\f42f"; } +.bi-info-circle-fill::before { content: "\f430"; } +.bi-info-circle::before { content: "\f431"; } +.bi-info-square-fill::before { content: "\f432"; } +.bi-info-square::before { content: "\f433"; } +.bi-info::before { content: "\f434"; } +.bi-input-cursor-text::before { content: "\f435"; } +.bi-input-cursor::before { content: "\f436"; } +.bi-instagram::before { content: "\f437"; } +.bi-intersect::before { content: "\f438"; } +.bi-journal-album::before { content: "\f439"; } +.bi-journal-arrow-down::before { content: "\f43a"; } +.bi-journal-arrow-up::before { content: "\f43b"; } +.bi-journal-bookmark-fill::before { content: "\f43c"; } +.bi-journal-bookmark::before { content: "\f43d"; } +.bi-journal-check::before { content: "\f43e"; } +.bi-journal-code::before { content: "\f43f"; } +.bi-journal-medical::before { content: "\f440"; } +.bi-journal-minus::before { content: "\f441"; } +.bi-journal-plus::before { content: "\f442"; } +.bi-journal-richtext::before { content: "\f443"; } +.bi-journal-text::before { content: "\f444"; } +.bi-journal-x::before { content: "\f445"; } +.bi-journal::before { content: "\f446"; } +.bi-journals::before { content: "\f447"; } +.bi-joystick::before { content: "\f448"; } +.bi-justify-left::before { content: "\f449"; } +.bi-justify-right::before { content: "\f44a"; } +.bi-justify::before { content: "\f44b"; } +.bi-kanban-fill::before { content: "\f44c"; } +.bi-kanban::before { content: "\f44d"; } +.bi-key-fill::before { content: "\f44e"; } +.bi-key::before { content: "\f44f"; } +.bi-keyboard-fill::before { content: "\f450"; } +.bi-keyboard::before { content: "\f451"; } +.bi-ladder::before { content: "\f452"; } +.bi-lamp-fill::before { content: "\f453"; } +.bi-lamp::before { content: "\f454"; } +.bi-laptop-fill::before { content: "\f455"; } +.bi-laptop::before { content: "\f456"; } +.bi-layer-backward::before { content: "\f457"; } +.bi-layer-forward::before { content: "\f458"; } +.bi-layers-fill::before { content: "\f459"; } +.bi-layers-half::before { content: "\f45a"; } +.bi-layers::before { content: "\f45b"; } +.bi-layout-sidebar-inset-reverse::before { content: "\f45c"; } +.bi-layout-sidebar-inset::before { content: "\f45d"; } +.bi-layout-sidebar-reverse::before { content: "\f45e"; } +.bi-layout-sidebar::before { content: "\f45f"; } +.bi-layout-split::before { content: "\f460"; } +.bi-layout-text-sidebar-reverse::before { content: "\f461"; } +.bi-layout-text-sidebar::before { content: "\f462"; } +.bi-layout-text-window-reverse::before { content: "\f463"; } +.bi-layout-text-window::before { content: "\f464"; } +.bi-layout-three-columns::before { content: "\f465"; } +.bi-layout-wtf::before { content: "\f466"; } +.bi-life-preserver::before { content: "\f467"; } +.bi-lightbulb-fill::before { content: "\f468"; } +.bi-lightbulb-off-fill::before { content: "\f469"; } +.bi-lightbulb-off::before { content: "\f46a"; } +.bi-lightbulb::before { content: "\f46b"; } +.bi-lightning-charge-fill::before { content: "\f46c"; } +.bi-lightning-charge::before { content: "\f46d"; } +.bi-lightning-fill::before { content: "\f46e"; } +.bi-lightning::before { content: "\f46f"; } +.bi-link-45deg::before { content: "\f470"; } +.bi-link::before { content: "\f471"; } +.bi-linkedin::before { content: "\f472"; } +.bi-list-check::before { content: "\f473"; } +.bi-list-nested::before { content: "\f474"; } +.bi-list-ol::before { content: "\f475"; } +.bi-list-stars::before { content: "\f476"; } +.bi-list-task::before { content: "\f477"; } +.bi-list-ul::before { content: "\f478"; } +.bi-list::before { content: "\f479"; } +.bi-lock-fill::before { content: "\f47a"; } +.bi-lock::before { content: "\f47b"; } +.bi-mailbox::before { content: "\f47c"; } +.bi-mailbox2::before { content: "\f47d"; } +.bi-map-fill::before { content: "\f47e"; } +.bi-map::before { content: "\f47f"; } +.bi-markdown-fill::before { content: "\f480"; } +.bi-markdown::before { content: "\f481"; } +.bi-mask::before { content: "\f482"; } +.bi-megaphone-fill::before { content: "\f483"; } +.bi-megaphone::before { content: "\f484"; } +.bi-menu-app-fill::before { content: "\f485"; } +.bi-menu-app::before { content: "\f486"; } +.bi-menu-button-fill::before { content: "\f487"; } +.bi-menu-button-wide-fill::before { content: "\f488"; } +.bi-menu-button-wide::before { content: "\f489"; } +.bi-menu-button::before { content: "\f48a"; } +.bi-menu-down::before { content: "\f48b"; } +.bi-menu-up::before { content: "\f48c"; } +.bi-mic-fill::before { content: "\f48d"; } +.bi-mic-mute-fill::before { content: "\f48e"; } +.bi-mic-mute::before { content: "\f48f"; } +.bi-mic::before { content: "\f490"; } +.bi-minecart-loaded::before { content: "\f491"; } +.bi-minecart::before { content: "\f492"; } +.bi-moisture::before { content: "\f493"; } +.bi-moon-fill::before { content: "\f494"; } +.bi-moon-stars-fill::before { content: "\f495"; } +.bi-moon-stars::before { content: "\f496"; } +.bi-moon::before { content: "\f497"; } +.bi-mouse-fill::before { content: "\f498"; } +.bi-mouse::before { content: "\f499"; } +.bi-mouse2-fill::before { content: "\f49a"; } +.bi-mouse2::before { content: "\f49b"; } +.bi-mouse3-fill::before { content: "\f49c"; } +.bi-mouse3::before { content: "\f49d"; } +.bi-music-note-beamed::before { content: "\f49e"; } +.bi-music-note-list::before { content: "\f49f"; } +.bi-music-note::before { content: "\f4a0"; } +.bi-music-player-fill::before { content: "\f4a1"; } +.bi-music-player::before { content: "\f4a2"; } +.bi-newspaper::before { content: "\f4a3"; } +.bi-node-minus-fill::before { content: "\f4a4"; } +.bi-node-minus::before { content: "\f4a5"; } +.bi-node-plus-fill::before { content: "\f4a6"; } +.bi-node-plus::before { content: "\f4a7"; } +.bi-nut-fill::before { content: "\f4a8"; } +.bi-nut::before { content: "\f4a9"; } +.bi-octagon-fill::before { content: "\f4aa"; } +.bi-octagon-half::before { content: "\f4ab"; } +.bi-octagon::before { content: "\f4ac"; } +.bi-option::before { content: "\f4ad"; } +.bi-outlet::before { content: "\f4ae"; } +.bi-paint-bucket::before { content: "\f4af"; } +.bi-palette-fill::before { content: "\f4b0"; } +.bi-palette::before { content: "\f4b1"; } +.bi-palette2::before { content: "\f4b2"; } +.bi-paperclip::before { content: "\f4b3"; } +.bi-paragraph::before { content: "\f4b4"; } +.bi-patch-check-fill::before { content: "\f4b5"; } +.bi-patch-check::before { content: "\f4b6"; } +.bi-patch-exclamation-fill::before { content: "\f4b7"; } +.bi-patch-exclamation::before { content: "\f4b8"; } +.bi-patch-minus-fill::before { content: "\f4b9"; } +.bi-patch-minus::before { content: "\f4ba"; } +.bi-patch-plus-fill::before { content: "\f4bb"; } +.bi-patch-plus::before { content: "\f4bc"; } +.bi-patch-question-fill::before { content: "\f4bd"; } +.bi-patch-question::before { content: "\f4be"; } +.bi-pause-btn-fill::before { content: "\f4bf"; } +.bi-pause-btn::before { content: "\f4c0"; } +.bi-pause-circle-fill::before { content: "\f4c1"; } +.bi-pause-circle::before { content: "\f4c2"; } +.bi-pause-fill::before { content: "\f4c3"; } +.bi-pause::before { content: "\f4c4"; } +.bi-peace-fill::before { content: "\f4c5"; } +.bi-peace::before { content: "\f4c6"; } +.bi-pen-fill::before { content: "\f4c7"; } +.bi-pen::before { content: "\f4c8"; } +.bi-pencil-fill::before { content: "\f4c9"; } +.bi-pencil-square::before { content: "\f4ca"; } +.bi-pencil::before { content: "\f4cb"; } +.bi-pentagon-fill::before { content: "\f4cc"; } +.bi-pentagon-half::before { content: "\f4cd"; } +.bi-pentagon::before { content: "\f4ce"; } +.bi-people-fill::before { content: "\f4cf"; } +.bi-people::before { content: "\f4d0"; } +.bi-percent::before { content: "\f4d1"; } +.bi-person-badge-fill::before { content: "\f4d2"; } +.bi-person-badge::before { content: "\f4d3"; } +.bi-person-bounding-box::before { content: "\f4d4"; } +.bi-person-check-fill::before { content: "\f4d5"; } +.bi-person-check::before { content: "\f4d6"; } +.bi-person-circle::before { content: "\f4d7"; } +.bi-person-dash-fill::before { content: "\f4d8"; } +.bi-person-dash::before { content: "\f4d9"; } +.bi-person-fill::before { content: "\f4da"; } +.bi-person-lines-fill::before { content: "\f4db"; } +.bi-person-plus-fill::before { content: "\f4dc"; } +.bi-person-plus::before { content: "\f4dd"; } +.bi-person-square::before { content: "\f4de"; } +.bi-person-x-fill::before { content: "\f4df"; } +.bi-person-x::before { content: "\f4e0"; } +.bi-person::before { content: "\f4e1"; } +.bi-phone-fill::before { content: "\f4e2"; } +.bi-phone-landscape-fill::before { content: "\f4e3"; } +.bi-phone-landscape::before { content: "\f4e4"; } +.bi-phone-vibrate-fill::before { content: "\f4e5"; } +.bi-phone-vibrate::before { content: "\f4e6"; } +.bi-phone::before { content: "\f4e7"; } +.bi-pie-chart-fill::before { content: "\f4e8"; } +.bi-pie-chart::before { content: "\f4e9"; } +.bi-pin-angle-fill::before { content: "\f4ea"; } +.bi-pin-angle::before { content: "\f4eb"; } +.bi-pin-fill::before { content: "\f4ec"; } +.bi-pin::before { content: "\f4ed"; } +.bi-pip-fill::before { content: "\f4ee"; } +.bi-pip::before { content: "\f4ef"; } +.bi-play-btn-fill::before { content: "\f4f0"; } +.bi-play-btn::before { content: "\f4f1"; } +.bi-play-circle-fill::before { content: "\f4f2"; } +.bi-play-circle::before { content: "\f4f3"; } +.bi-play-fill::before { content: "\f4f4"; } +.bi-play::before { content: "\f4f5"; } +.bi-plug-fill::before { content: "\f4f6"; } +.bi-plug::before { content: "\f4f7"; } +.bi-plus-circle-dotted::before { content: "\f4f8"; } +.bi-plus-circle-fill::before { content: "\f4f9"; } +.bi-plus-circle::before { content: "\f4fa"; } +.bi-plus-square-dotted::before { content: "\f4fb"; } +.bi-plus-square-fill::before { content: "\f4fc"; } +.bi-plus-square::before { content: "\f4fd"; } +.bi-plus::before { content: "\f4fe"; } +.bi-power::before { content: "\f4ff"; } +.bi-printer-fill::before { content: "\f500"; } +.bi-printer::before { content: "\f501"; } +.bi-puzzle-fill::before { content: "\f502"; } +.bi-puzzle::before { content: "\f503"; } +.bi-question-circle-fill::before { content: "\f504"; } +.bi-question-circle::before { content: "\f505"; } +.bi-question-diamond-fill::before { content: "\f506"; } +.bi-question-diamond::before { content: "\f507"; } +.bi-question-octagon-fill::before { content: "\f508"; } +.bi-question-octagon::before { content: "\f509"; } +.bi-question-square-fill::before { content: "\f50a"; } +.bi-question-square::before { content: "\f50b"; } +.bi-question::before { content: "\f50c"; } +.bi-rainbow::before { content: "\f50d"; } +.bi-receipt-cutoff::before { content: "\f50e"; } +.bi-receipt::before { content: "\f50f"; } +.bi-reception-0::before { content: "\f510"; } +.bi-reception-1::before { content: "\f511"; } +.bi-reception-2::before { content: "\f512"; } +.bi-reception-3::before { content: "\f513"; } +.bi-reception-4::before { content: "\f514"; } +.bi-record-btn-fill::before { content: "\f515"; } +.bi-record-btn::before { content: "\f516"; } +.bi-record-circle-fill::before { content: "\f517"; } +.bi-record-circle::before { content: "\f518"; } +.bi-record-fill::before { content: "\f519"; } +.bi-record::before { content: "\f51a"; } +.bi-record2-fill::before { content: "\f51b"; } +.bi-record2::before { content: "\f51c"; } +.bi-reply-all-fill::before { content: "\f51d"; } +.bi-reply-all::before { content: "\f51e"; } +.bi-reply-fill::before { content: "\f51f"; } +.bi-reply::before { content: "\f520"; } +.bi-rss-fill::before { content: "\f521"; } +.bi-rss::before { content: "\f522"; } +.bi-rulers::before { content: "\f523"; } +.bi-save-fill::before { content: "\f524"; } +.bi-save::before { content: "\f525"; } +.bi-save2-fill::before { content: "\f526"; } +.bi-save2::before { content: "\f527"; } +.bi-scissors::before { content: "\f528"; } +.bi-screwdriver::before { content: "\f529"; } +.bi-search::before { content: "\f52a"; } +.bi-segmented-nav::before { content: "\f52b"; } +.bi-server::before { content: "\f52c"; } +.bi-share-fill::before { content: "\f52d"; } +.bi-share::before { content: "\f52e"; } +.bi-shield-check::before { content: "\f52f"; } +.bi-shield-exclamation::before { content: "\f530"; } +.bi-shield-fill-check::before { content: "\f531"; } +.bi-shield-fill-exclamation::before { content: "\f532"; } +.bi-shield-fill-minus::before { content: "\f533"; } +.bi-shield-fill-plus::before { content: "\f534"; } +.bi-shield-fill-x::before { content: "\f535"; } +.bi-shield-fill::before { content: "\f536"; } +.bi-shield-lock-fill::before { content: "\f537"; } +.bi-shield-lock::before { content: "\f538"; } +.bi-shield-minus::before { content: "\f539"; } +.bi-shield-plus::before { content: "\f53a"; } +.bi-shield-shaded::before { content: "\f53b"; } +.bi-shield-slash-fill::before { content: "\f53c"; } +.bi-shield-slash::before { content: "\f53d"; } +.bi-shield-x::before { content: "\f53e"; } +.bi-shield::before { content: "\f53f"; } +.bi-shift-fill::before { content: "\f540"; } +.bi-shift::before { content: "\f541"; } +.bi-shop-window::before { content: "\f542"; } +.bi-shop::before { content: "\f543"; } +.bi-shuffle::before { content: "\f544"; } +.bi-signpost-2-fill::before { content: "\f545"; } +.bi-signpost-2::before { content: "\f546"; } +.bi-signpost-fill::before { content: "\f547"; } +.bi-signpost-split-fill::before { content: "\f548"; } +.bi-signpost-split::before { content: "\f549"; } +.bi-signpost::before { content: "\f54a"; } +.bi-sim-fill::before { content: "\f54b"; } +.bi-sim::before { content: "\f54c"; } +.bi-skip-backward-btn-fill::before { content: "\f54d"; } +.bi-skip-backward-btn::before { content: "\f54e"; } +.bi-skip-backward-circle-fill::before { content: "\f54f"; } +.bi-skip-backward-circle::before { content: "\f550"; } +.bi-skip-backward-fill::before { content: "\f551"; } +.bi-skip-backward::before { content: "\f552"; } +.bi-skip-end-btn-fill::before { content: "\f553"; } +.bi-skip-end-btn::before { content: "\f554"; } +.bi-skip-end-circle-fill::before { content: "\f555"; } +.bi-skip-end-circle::before { content: "\f556"; } +.bi-skip-end-fill::before { content: "\f557"; } +.bi-skip-end::before { content: "\f558"; } +.bi-skip-forward-btn-fill::before { content: "\f559"; } +.bi-skip-forward-btn::before { content: "\f55a"; } +.bi-skip-forward-circle-fill::before { content: "\f55b"; } +.bi-skip-forward-circle::before { content: "\f55c"; } +.bi-skip-forward-fill::before { content: "\f55d"; } +.bi-skip-forward::before { content: "\f55e"; } +.bi-skip-start-btn-fill::before { content: "\f55f"; } +.bi-skip-start-btn::before { content: "\f560"; } +.bi-skip-start-circle-fill::before { content: "\f561"; } +.bi-skip-start-circle::before { content: "\f562"; } +.bi-skip-start-fill::before { content: "\f563"; } +.bi-skip-start::before { content: "\f564"; } +.bi-slack::before { content: "\f565"; } +.bi-slash-circle-fill::before { content: "\f566"; } +.bi-slash-circle::before { content: "\f567"; } +.bi-slash-square-fill::before { content: "\f568"; } +.bi-slash-square::before { content: "\f569"; } +.bi-slash::before { content: "\f56a"; } +.bi-sliders::before { content: "\f56b"; } +.bi-smartwatch::before { content: "\f56c"; } +.bi-snow::before { content: "\f56d"; } +.bi-snow2::before { content: "\f56e"; } +.bi-snow3::before { content: "\f56f"; } +.bi-sort-alpha-down-alt::before { content: "\f570"; } +.bi-sort-alpha-down::before { content: "\f571"; } +.bi-sort-alpha-up-alt::before { content: "\f572"; } +.bi-sort-alpha-up::before { content: "\f573"; } +.bi-sort-down-alt::before { content: "\f574"; } +.bi-sort-down::before { content: "\f575"; } +.bi-sort-numeric-down-alt::before { content: "\f576"; } +.bi-sort-numeric-down::before { content: "\f577"; } +.bi-sort-numeric-up-alt::before { content: "\f578"; } +.bi-sort-numeric-up::before { content: "\f579"; } +.bi-sort-up-alt::before { content: "\f57a"; } +.bi-sort-up::before { content: "\f57b"; } +.bi-soundwave::before { content: "\f57c"; } +.bi-speaker-fill::before { content: "\f57d"; } +.bi-speaker::before { content: "\f57e"; } +.bi-speedometer::before { content: "\f57f"; } +.bi-speedometer2::before { content: "\f580"; } +.bi-spellcheck::before { content: "\f581"; } +.bi-square-fill::before { content: "\f582"; } +.bi-square-half::before { content: "\f583"; } +.bi-square::before { content: "\f584"; } +.bi-stack::before { content: "\f585"; } +.bi-star-fill::before { content: "\f586"; } +.bi-star-half::before { content: "\f587"; } +.bi-star::before { content: "\f588"; } +.bi-stars::before { content: "\f589"; } +.bi-stickies-fill::before { content: "\f58a"; } +.bi-stickies::before { content: "\f58b"; } +.bi-sticky-fill::before { content: "\f58c"; } +.bi-sticky::before { content: "\f58d"; } +.bi-stop-btn-fill::before { content: "\f58e"; } +.bi-stop-btn::before { content: "\f58f"; } +.bi-stop-circle-fill::before { content: "\f590"; } +.bi-stop-circle::before { content: "\f591"; } +.bi-stop-fill::before { content: "\f592"; } +.bi-stop::before { content: "\f593"; } +.bi-stoplights-fill::before { content: "\f594"; } +.bi-stoplights::before { content: "\f595"; } +.bi-stopwatch-fill::before { content: "\f596"; } +.bi-stopwatch::before { content: "\f597"; } +.bi-subtract::before { content: "\f598"; } +.bi-suit-club-fill::before { content: "\f599"; } +.bi-suit-club::before { content: "\f59a"; } +.bi-suit-diamond-fill::before { content: "\f59b"; } +.bi-suit-diamond::before { content: "\f59c"; } +.bi-suit-heart-fill::before { content: "\f59d"; } +.bi-suit-heart::before { content: "\f59e"; } +.bi-suit-spade-fill::before { content: "\f59f"; } +.bi-suit-spade::before { content: "\f5a0"; } +.bi-sun-fill::before { content: "\f5a1"; } +.bi-sun::before { content: "\f5a2"; } +.bi-sunglasses::before { content: "\f5a3"; } +.bi-sunrise-fill::before { content: "\f5a4"; } +.bi-sunrise::before { content: "\f5a5"; } +.bi-sunset-fill::before { content: "\f5a6"; } +.bi-sunset::before { content: "\f5a7"; } +.bi-symmetry-horizontal::before { content: "\f5a8"; } +.bi-symmetry-vertical::before { content: "\f5a9"; } +.bi-table::before { content: "\f5aa"; } +.bi-tablet-fill::before { content: "\f5ab"; } +.bi-tablet-landscape-fill::before { content: "\f5ac"; } +.bi-tablet-landscape::before { content: "\f5ad"; } +.bi-tablet::before { content: "\f5ae"; } +.bi-tag-fill::before { content: "\f5af"; } +.bi-tag::before { content: "\f5b0"; } +.bi-tags-fill::before { content: "\f5b1"; } +.bi-tags::before { content: "\f5b2"; } +.bi-telegram::before { content: "\f5b3"; } +.bi-telephone-fill::before { content: "\f5b4"; } +.bi-telephone-forward-fill::before { content: "\f5b5"; } +.bi-telephone-forward::before { content: "\f5b6"; } +.bi-telephone-inbound-fill::before { content: "\f5b7"; } +.bi-telephone-inbound::before { content: "\f5b8"; } +.bi-telephone-minus-fill::before { content: "\f5b9"; } +.bi-telephone-minus::before { content: "\f5ba"; } +.bi-telephone-outbound-fill::before { content: "\f5bb"; } +.bi-telephone-outbound::before { content: "\f5bc"; } +.bi-telephone-plus-fill::before { content: "\f5bd"; } +.bi-telephone-plus::before { content: "\f5be"; } +.bi-telephone-x-fill::before { content: "\f5bf"; } +.bi-telephone-x::before { content: "\f5c0"; } +.bi-telephone::before { content: "\f5c1"; } +.bi-terminal-fill::before { content: "\f5c2"; } +.bi-terminal::before { content: "\f5c3"; } +.bi-text-center::before { content: "\f5c4"; } +.bi-text-indent-left::before { content: "\f5c5"; } +.bi-text-indent-right::before { content: "\f5c6"; } +.bi-text-left::before { content: "\f5c7"; } +.bi-text-paragraph::before { content: "\f5c8"; } +.bi-text-right::before { content: "\f5c9"; } +.bi-textarea-resize::before { content: "\f5ca"; } +.bi-textarea-t::before { content: "\f5cb"; } +.bi-textarea::before { content: "\f5cc"; } +.bi-thermometer-half::before { content: "\f5cd"; } +.bi-thermometer-high::before { content: "\f5ce"; } +.bi-thermometer-low::before { content: "\f5cf"; } +.bi-thermometer-snow::before { content: "\f5d0"; } +.bi-thermometer-sun::before { content: "\f5d1"; } +.bi-thermometer::before { content: "\f5d2"; } +.bi-three-dots-vertical::before { content: "\f5d3"; } +.bi-three-dots::before { content: "\f5d4"; } +.bi-toggle-off::before { content: "\f5d5"; } +.bi-toggle-on::before { content: "\f5d6"; } +.bi-toggle2-off::before { content: "\f5d7"; } +.bi-toggle2-on::before { content: "\f5d8"; } +.bi-toggles::before { content: "\f5d9"; } +.bi-toggles2::before { content: "\f5da"; } +.bi-tools::before { content: "\f5db"; } +.bi-tornado::before { content: "\f5dc"; } +.bi-trash-fill::before { content: "\f5dd"; } +.bi-trash::before { content: "\f5de"; } +.bi-trash2-fill::before { content: "\f5df"; } +.bi-trash2::before { content: "\f5e0"; } +.bi-tree-fill::before { content: "\f5e1"; } +.bi-tree::before { content: "\f5e2"; } +.bi-triangle-fill::before { content: "\f5e3"; } +.bi-triangle-half::before { content: "\f5e4"; } +.bi-triangle::before { content: "\f5e5"; } +.bi-trophy-fill::before { content: "\f5e6"; } +.bi-trophy::before { content: "\f5e7"; } +.bi-tropical-storm::before { content: "\f5e8"; } +.bi-truck-flatbed::before { content: "\f5e9"; } +.bi-truck::before { content: "\f5ea"; } +.bi-tsunami::before { content: "\f5eb"; } +.bi-tv-fill::before { content: "\f5ec"; } +.bi-tv::before { content: "\f5ed"; } +.bi-twitch::before { content: "\f5ee"; } +.bi-twitter::before { content: "\f5ef"; } +.bi-type-bold::before { content: "\f5f0"; } +.bi-type-h1::before { content: "\f5f1"; } +.bi-type-h2::before { content: "\f5f2"; } +.bi-type-h3::before { content: "\f5f3"; } +.bi-type-italic::before { content: "\f5f4"; } +.bi-type-strikethrough::before { content: "\f5f5"; } +.bi-type-underline::before { content: "\f5f6"; } +.bi-type::before { content: "\f5f7"; } +.bi-ui-checks-grid::before { content: "\f5f8"; } +.bi-ui-checks::before { content: "\f5f9"; } +.bi-ui-radios-grid::before { content: "\f5fa"; } +.bi-ui-radios::before { content: "\f5fb"; } +.bi-umbrella-fill::before { content: "\f5fc"; } +.bi-umbrella::before { content: "\f5fd"; } +.bi-union::before { content: "\f5fe"; } +.bi-unlock-fill::before { content: "\f5ff"; } +.bi-unlock::before { content: "\f600"; } +.bi-upc-scan::before { content: "\f601"; } +.bi-upc::before { content: "\f602"; } +.bi-upload::before { content: "\f603"; } +.bi-vector-pen::before { content: "\f604"; } +.bi-view-list::before { content: "\f605"; } +.bi-view-stacked::before { content: "\f606"; } +.bi-vinyl-fill::before { content: "\f607"; } +.bi-vinyl::before { content: "\f608"; } +.bi-voicemail::before { content: "\f609"; } +.bi-volume-down-fill::before { content: "\f60a"; } +.bi-volume-down::before { content: "\f60b"; } +.bi-volume-mute-fill::before { content: "\f60c"; } +.bi-volume-mute::before { content: "\f60d"; } +.bi-volume-off-fill::before { content: "\f60e"; } +.bi-volume-off::before { content: "\f60f"; } +.bi-volume-up-fill::before { content: "\f610"; } +.bi-volume-up::before { content: "\f611"; } +.bi-vr::before { content: "\f612"; } +.bi-wallet-fill::before { content: "\f613"; } +.bi-wallet::before { content: "\f614"; } +.bi-wallet2::before { content: "\f615"; } +.bi-watch::before { content: "\f616"; } +.bi-water::before { content: "\f617"; } +.bi-whatsapp::before { content: "\f618"; } +.bi-wifi-1::before { content: "\f619"; } +.bi-wifi-2::before { content: "\f61a"; } +.bi-wifi-off::before { content: "\f61b"; } +.bi-wifi::before { content: "\f61c"; } +.bi-wind::before { content: "\f61d"; } +.bi-window-dock::before { content: "\f61e"; } +.bi-window-sidebar::before { content: "\f61f"; } +.bi-window::before { content: "\f620"; } +.bi-wrench::before { content: "\f621"; } +.bi-x-circle-fill::before { content: "\f622"; } +.bi-x-circle::before { content: "\f623"; } +.bi-x-diamond-fill::before { content: "\f624"; } +.bi-x-diamond::before { content: "\f625"; } +.bi-x-octagon-fill::before { content: "\f626"; } +.bi-x-octagon::before { content: "\f627"; } +.bi-x-square-fill::before { content: "\f628"; } +.bi-x-square::before { content: "\f629"; } +.bi-x::before { content: "\f62a"; } +.bi-youtube::before { content: "\f62b"; } +.bi-zoom-in::before { content: "\f62c"; } +.bi-zoom-out::before { content: "\f62d"; } +.bi-bank::before { content: "\f62e"; } +.bi-bank2::before { content: "\f62f"; } +.bi-bell-slash-fill::before { content: "\f630"; } +.bi-bell-slash::before { content: "\f631"; } +.bi-cash-coin::before { content: "\f632"; } +.bi-check-lg::before { content: "\f633"; } +.bi-coin::before { content: "\f634"; } +.bi-currency-bitcoin::before { content: "\f635"; } +.bi-currency-dollar::before { content: "\f636"; } +.bi-currency-euro::before { content: "\f637"; } +.bi-currency-exchange::before { content: "\f638"; } +.bi-currency-pound::before { content: "\f639"; } +.bi-currency-yen::before { content: "\f63a"; } +.bi-dash-lg::before { content: "\f63b"; } +.bi-exclamation-lg::before { content: "\f63c"; } +.bi-file-earmark-pdf-fill::before { content: "\f63d"; } +.bi-file-earmark-pdf::before { content: "\f63e"; } +.bi-file-pdf-fill::before { content: "\f63f"; } +.bi-file-pdf::before { content: "\f640"; } +.bi-gender-ambiguous::before { content: "\f641"; } +.bi-gender-female::before { content: "\f642"; } +.bi-gender-male::before { content: "\f643"; } +.bi-gender-trans::before { content: "\f644"; } +.bi-headset-vr::before { content: "\f645"; } +.bi-info-lg::before { content: "\f646"; } +.bi-mastodon::before { content: "\f647"; } +.bi-messenger::before { content: "\f648"; } +.bi-piggy-bank-fill::before { content: "\f649"; } +.bi-piggy-bank::before { content: "\f64a"; } +.bi-pin-map-fill::before { content: "\f64b"; } +.bi-pin-map::before { content: "\f64c"; } +.bi-plus-lg::before { content: "\f64d"; } +.bi-question-lg::before { content: "\f64e"; } +.bi-recycle::before { content: "\f64f"; } +.bi-reddit::before { content: "\f650"; } +.bi-safe-fill::before { content: "\f651"; } +.bi-safe2-fill::before { content: "\f652"; } +.bi-safe2::before { content: "\f653"; } +.bi-sd-card-fill::before { content: "\f654"; } +.bi-sd-card::before { content: "\f655"; } +.bi-skype::before { content: "\f656"; } +.bi-slash-lg::before { content: "\f657"; } +.bi-translate::before { content: "\f658"; } +.bi-x-lg::before { content: "\f659"; } +.bi-safe::before { content: "\f65a"; } +.bi-apple::before { content: "\f65b"; } +.bi-microsoft::before { content: "\f65d"; } +.bi-windows::before { content: "\f65e"; } +.bi-behance::before { content: "\f65c"; } +.bi-dribbble::before { content: "\f65f"; } +.bi-line::before { content: "\f660"; } +.bi-medium::before { content: "\f661"; } +.bi-paypal::before { content: "\f662"; } +.bi-pinterest::before { content: "\f663"; } +.bi-signal::before { content: "\f664"; } +.bi-snapchat::before { content: "\f665"; } +.bi-spotify::before { content: "\f666"; } +.bi-stack-overflow::before { content: "\f667"; } +.bi-strava::before { content: "\f668"; } +.bi-wordpress::before { content: "\f669"; } +.bi-vimeo::before { content: "\f66a"; } +.bi-activity::before { content: "\f66b"; } +.bi-easel2-fill::before { content: "\f66c"; } +.bi-easel2::before { content: "\f66d"; } +.bi-easel3-fill::before { content: "\f66e"; } +.bi-easel3::before { content: "\f66f"; } +.bi-fan::before { content: "\f670"; } +.bi-fingerprint::before { content: "\f671"; } +.bi-graph-down-arrow::before { content: "\f672"; } +.bi-graph-up-arrow::before { content: "\f673"; } +.bi-hypnotize::before { content: "\f674"; } +.bi-magic::before { content: "\f675"; } +.bi-person-rolodex::before { content: "\f676"; } +.bi-person-video::before { content: "\f677"; } +.bi-person-video2::before { content: "\f678"; } +.bi-person-video3::before { content: "\f679"; } +.bi-person-workspace::before { content: "\f67a"; } +.bi-radioactive::before { content: "\f67b"; } +.bi-webcam-fill::before { content: "\f67c"; } +.bi-webcam::before { content: "\f67d"; } +.bi-yin-yang::before { content: "\f67e"; } +.bi-bandaid-fill::before { content: "\f680"; } +.bi-bandaid::before { content: "\f681"; } +.bi-bluetooth::before { content: "\f682"; } +.bi-body-text::before { content: "\f683"; } +.bi-boombox::before { content: "\f684"; } +.bi-boxes::before { content: "\f685"; } +.bi-dpad-fill::before { content: "\f686"; } +.bi-dpad::before { content: "\f687"; } +.bi-ear-fill::before { content: "\f688"; } +.bi-ear::before { content: "\f689"; } +.bi-envelope-check-fill::before { content: "\f68b"; } +.bi-envelope-check::before { content: "\f68c"; } +.bi-envelope-dash-fill::before { content: "\f68e"; } +.bi-envelope-dash::before { content: "\f68f"; } +.bi-envelope-exclamation-fill::before { content: "\f691"; } +.bi-envelope-exclamation::before { content: "\f692"; } +.bi-envelope-plus-fill::before { content: "\f693"; } +.bi-envelope-plus::before { content: "\f694"; } +.bi-envelope-slash-fill::before { content: "\f696"; } +.bi-envelope-slash::before { content: "\f697"; } +.bi-envelope-x-fill::before { content: "\f699"; } +.bi-envelope-x::before { content: "\f69a"; } +.bi-explicit-fill::before { content: "\f69b"; } +.bi-explicit::before { content: "\f69c"; } +.bi-git::before { content: "\f69d"; } +.bi-infinity::before { content: "\f69e"; } +.bi-list-columns-reverse::before { content: "\f69f"; } +.bi-list-columns::before { content: "\f6a0"; } +.bi-meta::before { content: "\f6a1"; } +.bi-nintendo-switch::before { content: "\f6a4"; } +.bi-pc-display-horizontal::before { content: "\f6a5"; } +.bi-pc-display::before { content: "\f6a6"; } +.bi-pc-horizontal::before { content: "\f6a7"; } +.bi-pc::before { content: "\f6a8"; } +.bi-playstation::before { content: "\f6a9"; } +.bi-plus-slash-minus::before { content: "\f6aa"; } +.bi-projector-fill::before { content: "\f6ab"; } +.bi-projector::before { content: "\f6ac"; } +.bi-qr-code-scan::before { content: "\f6ad"; } +.bi-qr-code::before { content: "\f6ae"; } +.bi-quora::before { content: "\f6af"; } +.bi-quote::before { content: "\f6b0"; } +.bi-robot::before { content: "\f6b1"; } +.bi-send-check-fill::before { content: "\f6b2"; } +.bi-send-check::before { content: "\f6b3"; } +.bi-send-dash-fill::before { content: "\f6b4"; } +.bi-send-dash::before { content: "\f6b5"; } +.bi-send-exclamation-fill::before { content: "\f6b7"; } +.bi-send-exclamation::before { content: "\f6b8"; } +.bi-send-fill::before { content: "\f6b9"; } +.bi-send-plus-fill::before { content: "\f6ba"; } +.bi-send-plus::before { content: "\f6bb"; } +.bi-send-slash-fill::before { content: "\f6bc"; } +.bi-send-slash::before { content: "\f6bd"; } +.bi-send-x-fill::before { content: "\f6be"; } +.bi-send-x::before { content: "\f6bf"; } +.bi-send::before { content: "\f6c0"; } +.bi-steam::before { content: "\f6c1"; } +.bi-terminal-dash::before { content: "\f6c3"; } +.bi-terminal-plus::before { content: "\f6c4"; } +.bi-terminal-split::before { content: "\f6c5"; } +.bi-ticket-detailed-fill::before { content: "\f6c6"; } +.bi-ticket-detailed::before { content: "\f6c7"; } +.bi-ticket-fill::before { content: "\f6c8"; } +.bi-ticket-perforated-fill::before { content: "\f6c9"; } +.bi-ticket-perforated::before { content: "\f6ca"; } +.bi-ticket::before { content: "\f6cb"; } +.bi-tiktok::before { content: "\f6cc"; } +.bi-window-dash::before { content: "\f6cd"; } +.bi-window-desktop::before { content: "\f6ce"; } +.bi-window-fullscreen::before { content: "\f6cf"; } +.bi-window-plus::before { content: "\f6d0"; } +.bi-window-split::before { content: "\f6d1"; } +.bi-window-stack::before { content: "\f6d2"; } +.bi-window-x::before { content: "\f6d3"; } +.bi-xbox::before { content: "\f6d4"; } +.bi-ethernet::before { content: "\f6d5"; } +.bi-hdmi-fill::before { content: "\f6d6"; } +.bi-hdmi::before { content: "\f6d7"; } +.bi-usb-c-fill::before { content: "\f6d8"; } +.bi-usb-c::before { content: "\f6d9"; } +.bi-usb-fill::before { content: "\f6da"; } +.bi-usb-plug-fill::before { content: "\f6db"; } +.bi-usb-plug::before { content: "\f6dc"; } +.bi-usb-symbol::before { content: "\f6dd"; } +.bi-usb::before { content: "\f6de"; } +.bi-boombox-fill::before { content: "\f6df"; } +.bi-displayport::before { content: "\f6e1"; } +.bi-gpu-card::before { content: "\f6e2"; } +.bi-memory::before { content: "\f6e3"; } +.bi-modem-fill::before { content: "\f6e4"; } +.bi-modem::before { content: "\f6e5"; } +.bi-motherboard-fill::before { content: "\f6e6"; } +.bi-motherboard::before { content: "\f6e7"; } +.bi-optical-audio-fill::before { content: "\f6e8"; } +.bi-optical-audio::before { content: "\f6e9"; } +.bi-pci-card::before { content: "\f6ea"; } +.bi-router-fill::before { content: "\f6eb"; } +.bi-router::before { content: "\f6ec"; } +.bi-thunderbolt-fill::before { content: "\f6ef"; } +.bi-thunderbolt::before { content: "\f6f0"; } +.bi-usb-drive-fill::before { content: "\f6f1"; } +.bi-usb-drive::before { content: "\f6f2"; } +.bi-usb-micro-fill::before { content: "\f6f3"; } +.bi-usb-micro::before { content: "\f6f4"; } +.bi-usb-mini-fill::before { content: "\f6f5"; } +.bi-usb-mini::before { content: "\f6f6"; } +.bi-cloud-haze2::before { content: "\f6f7"; } +.bi-device-hdd-fill::before { content: "\f6f8"; } +.bi-device-hdd::before { content: "\f6f9"; } +.bi-device-ssd-fill::before { content: "\f6fa"; } +.bi-device-ssd::before { content: "\f6fb"; } +.bi-displayport-fill::before { content: "\f6fc"; } +.bi-mortarboard-fill::before { content: "\f6fd"; } +.bi-mortarboard::before { content: "\f6fe"; } +.bi-terminal-x::before { content: "\f6ff"; } +.bi-arrow-through-heart-fill::before { content: "\f700"; } +.bi-arrow-through-heart::before { content: "\f701"; } +.bi-badge-sd-fill::before { content: "\f702"; } +.bi-badge-sd::before { content: "\f703"; } +.bi-bag-heart-fill::before { content: "\f704"; } +.bi-bag-heart::before { content: "\f705"; } +.bi-balloon-fill::before { content: "\f706"; } +.bi-balloon-heart-fill::before { content: "\f707"; } +.bi-balloon-heart::before { content: "\f708"; } +.bi-balloon::before { content: "\f709"; } +.bi-box2-fill::before { content: "\f70a"; } +.bi-box2-heart-fill::before { content: "\f70b"; } +.bi-box2-heart::before { content: "\f70c"; } +.bi-box2::before { content: "\f70d"; } +.bi-braces-asterisk::before { content: "\f70e"; } +.bi-calendar-heart-fill::before { content: "\f70f"; } +.bi-calendar-heart::before { content: "\f710"; } +.bi-calendar2-heart-fill::before { content: "\f711"; } +.bi-calendar2-heart::before { content: "\f712"; } +.bi-chat-heart-fill::before { content: "\f713"; } +.bi-chat-heart::before { content: "\f714"; } +.bi-chat-left-heart-fill::before { content: "\f715"; } +.bi-chat-left-heart::before { content: "\f716"; } +.bi-chat-right-heart-fill::before { content: "\f717"; } +.bi-chat-right-heart::before { content: "\f718"; } +.bi-chat-square-heart-fill::before { content: "\f719"; } +.bi-chat-square-heart::before { content: "\f71a"; } +.bi-clipboard-check-fill::before { content: "\f71b"; } +.bi-clipboard-data-fill::before { content: "\f71c"; } +.bi-clipboard-fill::before { content: "\f71d"; } +.bi-clipboard-heart-fill::before { content: "\f71e"; } +.bi-clipboard-heart::before { content: "\f71f"; } +.bi-clipboard-minus-fill::before { content: "\f720"; } +.bi-clipboard-plus-fill::before { content: "\f721"; } +.bi-clipboard-pulse::before { content: "\f722"; } +.bi-clipboard-x-fill::before { content: "\f723"; } +.bi-clipboard2-check-fill::before { content: "\f724"; } +.bi-clipboard2-check::before { content: "\f725"; } +.bi-clipboard2-data-fill::before { content: "\f726"; } +.bi-clipboard2-data::before { content: "\f727"; } +.bi-clipboard2-fill::before { content: "\f728"; } +.bi-clipboard2-heart-fill::before { content: "\f729"; } +.bi-clipboard2-heart::before { content: "\f72a"; } +.bi-clipboard2-minus-fill::before { content: "\f72b"; } +.bi-clipboard2-minus::before { content: "\f72c"; } +.bi-clipboard2-plus-fill::before { content: "\f72d"; } +.bi-clipboard2-plus::before { content: "\f72e"; } +.bi-clipboard2-pulse-fill::before { content: "\f72f"; } +.bi-clipboard2-pulse::before { content: "\f730"; } +.bi-clipboard2-x-fill::before { content: "\f731"; } +.bi-clipboard2-x::before { content: "\f732"; } +.bi-clipboard2::before { content: "\f733"; } +.bi-emoji-kiss-fill::before { content: "\f734"; } +.bi-emoji-kiss::before { content: "\f735"; } +.bi-envelope-heart-fill::before { content: "\f736"; } +.bi-envelope-heart::before { content: "\f737"; } +.bi-envelope-open-heart-fill::before { content: "\f738"; } +.bi-envelope-open-heart::before { content: "\f739"; } +.bi-envelope-paper-fill::before { content: "\f73a"; } +.bi-envelope-paper-heart-fill::before { content: "\f73b"; } +.bi-envelope-paper-heart::before { content: "\f73c"; } +.bi-envelope-paper::before { content: "\f73d"; } +.bi-filetype-aac::before { content: "\f73e"; } +.bi-filetype-ai::before { content: "\f73f"; } +.bi-filetype-bmp::before { content: "\f740"; } +.bi-filetype-cs::before { content: "\f741"; } +.bi-filetype-css::before { content: "\f742"; } +.bi-filetype-csv::before { content: "\f743"; } +.bi-filetype-doc::before { content: "\f744"; } +.bi-filetype-docx::before { content: "\f745"; } +.bi-filetype-exe::before { content: "\f746"; } +.bi-filetype-gif::before { content: "\f747"; } +.bi-filetype-heic::before { content: "\f748"; } +.bi-filetype-html::before { content: "\f749"; } +.bi-filetype-java::before { content: "\f74a"; } +.bi-filetype-jpg::before { content: "\f74b"; } +.bi-filetype-js::before { content: "\f74c"; } +.bi-filetype-jsx::before { content: "\f74d"; } +.bi-filetype-key::before { content: "\f74e"; } +.bi-filetype-m4p::before { content: "\f74f"; } +.bi-filetype-md::before { content: "\f750"; } +.bi-filetype-mdx::before { content: "\f751"; } +.bi-filetype-mov::before { content: "\f752"; } +.bi-filetype-mp3::before { content: "\f753"; } +.bi-filetype-mp4::before { content: "\f754"; } +.bi-filetype-otf::before { content: "\f755"; } +.bi-filetype-pdf::before { content: "\f756"; } +.bi-filetype-php::before { content: "\f757"; } +.bi-filetype-png::before { content: "\f758"; } +.bi-filetype-ppt::before { content: "\f75a"; } +.bi-filetype-psd::before { content: "\f75b"; } +.bi-filetype-py::before { content: "\f75c"; } +.bi-filetype-raw::before { content: "\f75d"; } +.bi-filetype-rb::before { content: "\f75e"; } +.bi-filetype-sass::before { content: "\f75f"; } +.bi-filetype-scss::before { content: "\f760"; } +.bi-filetype-sh::before { content: "\f761"; } +.bi-filetype-svg::before { content: "\f762"; } +.bi-filetype-tiff::before { content: "\f763"; } +.bi-filetype-tsx::before { content: "\f764"; } +.bi-filetype-ttf::before { content: "\f765"; } +.bi-filetype-txt::before { content: "\f766"; } +.bi-filetype-wav::before { content: "\f767"; } +.bi-filetype-woff::before { content: "\f768"; } +.bi-filetype-xls::before { content: "\f76a"; } +.bi-filetype-xml::before { content: "\f76b"; } +.bi-filetype-yml::before { content: "\f76c"; } +.bi-heart-arrow::before { content: "\f76d"; } +.bi-heart-pulse-fill::before { content: "\f76e"; } +.bi-heart-pulse::before { content: "\f76f"; } +.bi-heartbreak-fill::before { content: "\f770"; } +.bi-heartbreak::before { content: "\f771"; } +.bi-hearts::before { content: "\f772"; } +.bi-hospital-fill::before { content: "\f773"; } +.bi-hospital::before { content: "\f774"; } +.bi-house-heart-fill::before { content: "\f775"; } +.bi-house-heart::before { content: "\f776"; } +.bi-incognito::before { content: "\f777"; } +.bi-magnet-fill::before { content: "\f778"; } +.bi-magnet::before { content: "\f779"; } +.bi-person-heart::before { content: "\f77a"; } +.bi-person-hearts::before { content: "\f77b"; } +.bi-phone-flip::before { content: "\f77c"; } +.bi-plugin::before { content: "\f77d"; } +.bi-postage-fill::before { content: "\f77e"; } +.bi-postage-heart-fill::before { content: "\f77f"; } +.bi-postage-heart::before { content: "\f780"; } +.bi-postage::before { content: "\f781"; } +.bi-postcard-fill::before { content: "\f782"; } +.bi-postcard-heart-fill::before { content: "\f783"; } +.bi-postcard-heart::before { content: "\f784"; } +.bi-postcard::before { content: "\f785"; } +.bi-search-heart-fill::before { content: "\f786"; } +.bi-search-heart::before { content: "\f787"; } +.bi-sliders2-vertical::before { content: "\f788"; } +.bi-sliders2::before { content: "\f789"; } +.bi-trash3-fill::before { content: "\f78a"; } +.bi-trash3::before { content: "\f78b"; } +.bi-valentine::before { content: "\f78c"; } +.bi-valentine2::before { content: "\f78d"; } +.bi-wrench-adjustable-circle-fill::before { content: "\f78e"; } +.bi-wrench-adjustable-circle::before { content: "\f78f"; } +.bi-wrench-adjustable::before { content: "\f790"; } +.bi-filetype-json::before { content: "\f791"; } +.bi-filetype-pptx::before { content: "\f792"; } +.bi-filetype-xlsx::before { content: "\f793"; } +.bi-1-circle-fill::before { content: "\f796"; } +.bi-1-circle::before { content: "\f797"; } +.bi-1-square-fill::before { content: "\f798"; } +.bi-1-square::before { content: "\f799"; } +.bi-2-circle-fill::before { content: "\f79c"; } +.bi-2-circle::before { content: "\f79d"; } +.bi-2-square-fill::before { content: "\f79e"; } +.bi-2-square::before { content: "\f79f"; } +.bi-3-circle-fill::before { content: "\f7a2"; } +.bi-3-circle::before { content: "\f7a3"; } +.bi-3-square-fill::before { content: "\f7a4"; } +.bi-3-square::before { content: "\f7a5"; } +.bi-4-circle-fill::before { content: "\f7a8"; } +.bi-4-circle::before { content: "\f7a9"; } +.bi-4-square-fill::before { content: "\f7aa"; } +.bi-4-square::before { content: "\f7ab"; } +.bi-5-circle-fill::before { content: "\f7ae"; } +.bi-5-circle::before { content: "\f7af"; } +.bi-5-square-fill::before { content: "\f7b0"; } +.bi-5-square::before { content: "\f7b1"; } +.bi-6-circle-fill::before { content: "\f7b4"; } +.bi-6-circle::before { content: "\f7b5"; } +.bi-6-square-fill::before { content: "\f7b6"; } +.bi-6-square::before { content: "\f7b7"; } +.bi-7-circle-fill::before { content: "\f7ba"; } +.bi-7-circle::before { content: "\f7bb"; } +.bi-7-square-fill::before { content: "\f7bc"; } +.bi-7-square::before { content: "\f7bd"; } +.bi-8-circle-fill::before { content: "\f7c0"; } +.bi-8-circle::before { content: "\f7c1"; } +.bi-8-square-fill::before { content: "\f7c2"; } +.bi-8-square::before { content: "\f7c3"; } +.bi-9-circle-fill::before { content: "\f7c6"; } +.bi-9-circle::before { content: "\f7c7"; } +.bi-9-square-fill::before { content: "\f7c8"; } +.bi-9-square::before { content: "\f7c9"; } +.bi-airplane-engines-fill::before { content: "\f7ca"; } +.bi-airplane-engines::before { content: "\f7cb"; } +.bi-airplane-fill::before { content: "\f7cc"; } +.bi-airplane::before { content: "\f7cd"; } +.bi-alexa::before { content: "\f7ce"; } +.bi-alipay::before { content: "\f7cf"; } +.bi-android::before { content: "\f7d0"; } +.bi-android2::before { content: "\f7d1"; } +.bi-box-fill::before { content: "\f7d2"; } +.bi-box-seam-fill::before { content: "\f7d3"; } +.bi-browser-chrome::before { content: "\f7d4"; } +.bi-browser-edge::before { content: "\f7d5"; } +.bi-browser-firefox::before { content: "\f7d6"; } +.bi-browser-safari::before { content: "\f7d7"; } +.bi-c-circle-fill::before { content: "\f7da"; } +.bi-c-circle::before { content: "\f7db"; } +.bi-c-square-fill::before { content: "\f7dc"; } +.bi-c-square::before { content: "\f7dd"; } +.bi-capsule-pill::before { content: "\f7de"; } +.bi-capsule::before { content: "\f7df"; } +.bi-car-front-fill::before { content: "\f7e0"; } +.bi-car-front::before { content: "\f7e1"; } +.bi-cassette-fill::before { content: "\f7e2"; } +.bi-cassette::before { content: "\f7e3"; } +.bi-cc-circle-fill::before { content: "\f7e6"; } +.bi-cc-circle::before { content: "\f7e7"; } +.bi-cc-square-fill::before { content: "\f7e8"; } +.bi-cc-square::before { content: "\f7e9"; } +.bi-cup-hot-fill::before { content: "\f7ea"; } +.bi-cup-hot::before { content: "\f7eb"; } +.bi-currency-rupee::before { content: "\f7ec"; } +.bi-dropbox::before { content: "\f7ed"; } +.bi-escape::before { content: "\f7ee"; } +.bi-fast-forward-btn-fill::before { content: "\f7ef"; } +.bi-fast-forward-btn::before { content: "\f7f0"; } +.bi-fast-forward-circle-fill::before { content: "\f7f1"; } +.bi-fast-forward-circle::before { content: "\f7f2"; } +.bi-fast-forward-fill::before { content: "\f7f3"; } +.bi-fast-forward::before { content: "\f7f4"; } +.bi-filetype-sql::before { content: "\f7f5"; } +.bi-fire::before { content: "\f7f6"; } +.bi-google-play::before { content: "\f7f7"; } +.bi-h-circle-fill::before { content: "\f7fa"; } +.bi-h-circle::before { content: "\f7fb"; } +.bi-h-square-fill::before { content: "\f7fc"; } +.bi-h-square::before { content: "\f7fd"; } +.bi-indent::before { content: "\f7fe"; } +.bi-lungs-fill::before { content: "\f7ff"; } +.bi-lungs::before { content: "\f800"; } +.bi-microsoft-teams::before { content: "\f801"; } +.bi-p-circle-fill::before { content: "\f804"; } +.bi-p-circle::before { content: "\f805"; } +.bi-p-square-fill::before { content: "\f806"; } +.bi-p-square::before { content: "\f807"; } +.bi-pass-fill::before { content: "\f808"; } +.bi-pass::before { content: "\f809"; } +.bi-prescription::before { content: "\f80a"; } +.bi-prescription2::before { content: "\f80b"; } +.bi-r-circle-fill::before { content: "\f80e"; } +.bi-r-circle::before { content: "\f80f"; } +.bi-r-square-fill::before { content: "\f810"; } +.bi-r-square::before { content: "\f811"; } +.bi-repeat-1::before { content: "\f812"; } +.bi-repeat::before { content: "\f813"; } +.bi-rewind-btn-fill::before { content: "\f814"; } +.bi-rewind-btn::before { content: "\f815"; } +.bi-rewind-circle-fill::before { content: "\f816"; } +.bi-rewind-circle::before { content: "\f817"; } +.bi-rewind-fill::before { content: "\f818"; } +.bi-rewind::before { content: "\f819"; } +.bi-train-freight-front-fill::before { content: "\f81a"; } +.bi-train-freight-front::before { content: "\f81b"; } +.bi-train-front-fill::before { content: "\f81c"; } +.bi-train-front::before { content: "\f81d"; } +.bi-train-lightrail-front-fill::before { content: "\f81e"; } +.bi-train-lightrail-front::before { content: "\f81f"; } +.bi-truck-front-fill::before { content: "\f820"; } +.bi-truck-front::before { content: "\f821"; } +.bi-ubuntu::before { content: "\f822"; } +.bi-unindent::before { content: "\f823"; } +.bi-unity::before { content: "\f824"; } +.bi-universal-access-circle::before { content: "\f825"; } +.bi-universal-access::before { content: "\f826"; } +.bi-virus::before { content: "\f827"; } +.bi-virus2::before { content: "\f828"; } +.bi-wechat::before { content: "\f829"; } +.bi-yelp::before { content: "\f82a"; } +.bi-sign-stop-fill::before { content: "\f82b"; } +.bi-sign-stop-lights-fill::before { content: "\f82c"; } +.bi-sign-stop-lights::before { content: "\f82d"; } +.bi-sign-stop::before { content: "\f82e"; } +.bi-sign-turn-left-fill::before { content: "\f82f"; } +.bi-sign-turn-left::before { content: "\f830"; } +.bi-sign-turn-right-fill::before { content: "\f831"; } +.bi-sign-turn-right::before { content: "\f832"; } +.bi-sign-turn-slight-left-fill::before { content: "\f833"; } +.bi-sign-turn-slight-left::before { content: "\f834"; } +.bi-sign-turn-slight-right-fill::before { content: "\f835"; } +.bi-sign-turn-slight-right::before { content: "\f836"; } +.bi-sign-yield-fill::before { content: "\f837"; } +.bi-sign-yield::before { content: "\f838"; } +.bi-ev-station-fill::before { content: "\f839"; } +.bi-ev-station::before { content: "\f83a"; } +.bi-fuel-pump-diesel-fill::before { content: "\f83b"; } +.bi-fuel-pump-diesel::before { content: "\f83c"; } +.bi-fuel-pump-fill::before { content: "\f83d"; } +.bi-fuel-pump::before { content: "\f83e"; } +.bi-0-circle-fill::before { content: "\f83f"; } +.bi-0-circle::before { content: "\f840"; } +.bi-0-square-fill::before { content: "\f841"; } +.bi-0-square::before { content: "\f842"; } +.bi-rocket-fill::before { content: "\f843"; } +.bi-rocket-takeoff-fill::before { content: "\f844"; } +.bi-rocket-takeoff::before { content: "\f845"; } +.bi-rocket::before { content: "\f846"; } +.bi-stripe::before { content: "\f847"; } +.bi-subscript::before { content: "\f848"; } +.bi-superscript::before { content: "\f849"; } +.bi-trello::before { content: "\f84a"; } +.bi-envelope-at-fill::before { content: "\f84b"; } +.bi-envelope-at::before { content: "\f84c"; } +.bi-regex::before { content: "\f84d"; } +.bi-text-wrap::before { content: "\f84e"; } +.bi-sign-dead-end-fill::before { content: "\f84f"; } +.bi-sign-dead-end::before { content: "\f850"; } +.bi-sign-do-not-enter-fill::before { content: "\f851"; } +.bi-sign-do-not-enter::before { content: "\f852"; } +.bi-sign-intersection-fill::before { content: "\f853"; } +.bi-sign-intersection-side-fill::before { content: "\f854"; } +.bi-sign-intersection-side::before { content: "\f855"; } +.bi-sign-intersection-t-fill::before { content: "\f856"; } +.bi-sign-intersection-t::before { content: "\f857"; } +.bi-sign-intersection-y-fill::before { content: "\f858"; } +.bi-sign-intersection-y::before { content: "\f859"; } +.bi-sign-intersection::before { content: "\f85a"; } +.bi-sign-merge-left-fill::before { content: "\f85b"; } +.bi-sign-merge-left::before { content: "\f85c"; } +.bi-sign-merge-right-fill::before { content: "\f85d"; } +.bi-sign-merge-right::before { content: "\f85e"; } +.bi-sign-no-left-turn-fill::before { content: "\f85f"; } +.bi-sign-no-left-turn::before { content: "\f860"; } +.bi-sign-no-parking-fill::before { content: "\f861"; } +.bi-sign-no-parking::before { content: "\f862"; } +.bi-sign-no-right-turn-fill::before { content: "\f863"; } +.bi-sign-no-right-turn::before { content: "\f864"; } +.bi-sign-railroad-fill::before { content: "\f865"; } +.bi-sign-railroad::before { content: "\f866"; } +.bi-building-add::before { content: "\f867"; } +.bi-building-check::before { content: "\f868"; } +.bi-building-dash::before { content: "\f869"; } +.bi-building-down::before { content: "\f86a"; } +.bi-building-exclamation::before { content: "\f86b"; } +.bi-building-fill-add::before { content: "\f86c"; } +.bi-building-fill-check::before { content: "\f86d"; } +.bi-building-fill-dash::before { content: "\f86e"; } +.bi-building-fill-down::before { content: "\f86f"; } +.bi-building-fill-exclamation::before { content: "\f870"; } +.bi-building-fill-gear::before { content: "\f871"; } +.bi-building-fill-lock::before { content: "\f872"; } +.bi-building-fill-slash::before { content: "\f873"; } +.bi-building-fill-up::before { content: "\f874"; } +.bi-building-fill-x::before { content: "\f875"; } +.bi-building-fill::before { content: "\f876"; } +.bi-building-gear::before { content: "\f877"; } +.bi-building-lock::before { content: "\f878"; } +.bi-building-slash::before { content: "\f879"; } +.bi-building-up::before { content: "\f87a"; } +.bi-building-x::before { content: "\f87b"; } +.bi-buildings-fill::before { content: "\f87c"; } +.bi-buildings::before { content: "\f87d"; } +.bi-bus-front-fill::before { content: "\f87e"; } +.bi-bus-front::before { content: "\f87f"; } +.bi-ev-front-fill::before { content: "\f880"; } +.bi-ev-front::before { content: "\f881"; } +.bi-globe-americas::before { content: "\f882"; } +.bi-globe-asia-australia::before { content: "\f883"; } +.bi-globe-central-south-asia::before { content: "\f884"; } +.bi-globe-europe-africa::before { content: "\f885"; } +.bi-house-add-fill::before { content: "\f886"; } +.bi-house-add::before { content: "\f887"; } +.bi-house-check-fill::before { content: "\f888"; } +.bi-house-check::before { content: "\f889"; } +.bi-house-dash-fill::before { content: "\f88a"; } +.bi-house-dash::before { content: "\f88b"; } +.bi-house-down-fill::before { content: "\f88c"; } +.bi-house-down::before { content: "\f88d"; } +.bi-house-exclamation-fill::before { content: "\f88e"; } +.bi-house-exclamation::before { content: "\f88f"; } +.bi-house-gear-fill::before { content: "\f890"; } +.bi-house-gear::before { content: "\f891"; } +.bi-house-lock-fill::before { content: "\f892"; } +.bi-house-lock::before { content: "\f893"; } +.bi-house-slash-fill::before { content: "\f894"; } +.bi-house-slash::before { content: "\f895"; } +.bi-house-up-fill::before { content: "\f896"; } +.bi-house-up::before { content: "\f897"; } +.bi-house-x-fill::before { content: "\f898"; } +.bi-house-x::before { content: "\f899"; } +.bi-person-add::before { content: "\f89a"; } +.bi-person-down::before { content: "\f89b"; } +.bi-person-exclamation::before { content: "\f89c"; } +.bi-person-fill-add::before { content: "\f89d"; } +.bi-person-fill-check::before { content: "\f89e"; } +.bi-person-fill-dash::before { content: "\f89f"; } +.bi-person-fill-down::before { content: "\f8a0"; } +.bi-person-fill-exclamation::before { content: "\f8a1"; } +.bi-person-fill-gear::before { content: "\f8a2"; } +.bi-person-fill-lock::before { content: "\f8a3"; } +.bi-person-fill-slash::before { content: "\f8a4"; } +.bi-person-fill-up::before { content: "\f8a5"; } +.bi-person-fill-x::before { content: "\f8a6"; } +.bi-person-gear::before { content: "\f8a7"; } +.bi-person-lock::before { content: "\f8a8"; } +.bi-person-slash::before { content: "\f8a9"; } +.bi-person-up::before { content: "\f8aa"; } +.bi-scooter::before { content: "\f8ab"; } +.bi-taxi-front-fill::before { content: "\f8ac"; } +.bi-taxi-front::before { content: "\f8ad"; } +.bi-amd::before { content: "\f8ae"; } +.bi-database-add::before { content: "\f8af"; } +.bi-database-check::before { content: "\f8b0"; } +.bi-database-dash::before { content: "\f8b1"; } +.bi-database-down::before { content: "\f8b2"; } +.bi-database-exclamation::before { content: "\f8b3"; } +.bi-database-fill-add::before { content: "\f8b4"; } +.bi-database-fill-check::before { content: "\f8b5"; } +.bi-database-fill-dash::before { content: "\f8b6"; } +.bi-database-fill-down::before { content: "\f8b7"; } +.bi-database-fill-exclamation::before { content: "\f8b8"; } +.bi-database-fill-gear::before { content: "\f8b9"; } +.bi-database-fill-lock::before { content: "\f8ba"; } +.bi-database-fill-slash::before { content: "\f8bb"; } +.bi-database-fill-up::before { content: "\f8bc"; } +.bi-database-fill-x::before { content: "\f8bd"; } +.bi-database-fill::before { content: "\f8be"; } +.bi-database-gear::before { content: "\f8bf"; } +.bi-database-lock::before { content: "\f8c0"; } +.bi-database-slash::before { content: "\f8c1"; } +.bi-database-up::before { content: "\f8c2"; } +.bi-database-x::before { content: "\f8c3"; } +.bi-database::before { content: "\f8c4"; } +.bi-houses-fill::before { content: "\f8c5"; } +.bi-houses::before { content: "\f8c6"; } +.bi-nvidia::before { content: "\f8c7"; } +.bi-person-vcard-fill::before { content: "\f8c8"; } +.bi-person-vcard::before { content: "\f8c9"; } +.bi-sina-weibo::before { content: "\f8ca"; } +.bi-tencent-qq::before { content: "\f8cb"; } +.bi-wikipedia::before { content: "\f8cc"; } +.bi-alphabet-uppercase::before { content: "\f2a5"; } +.bi-alphabet::before { content: "\f68a"; } +.bi-amazon::before { content: "\f68d"; } +.bi-arrows-collapse-vertical::before { content: "\f690"; } +.bi-arrows-expand-vertical::before { content: "\f695"; } +.bi-arrows-vertical::before { content: "\f698"; } +.bi-arrows::before { content: "\f6a2"; } +.bi-ban-fill::before { content: "\f6a3"; } +.bi-ban::before { content: "\f6b6"; } +.bi-bing::before { content: "\f6c2"; } +.bi-cake::before { content: "\f6e0"; } +.bi-cake2::before { content: "\f6ed"; } +.bi-cookie::before { content: "\f6ee"; } +.bi-copy::before { content: "\f759"; } +.bi-crosshair::before { content: "\f769"; } +.bi-crosshair2::before { content: "\f794"; } +.bi-emoji-astonished-fill::before { content: "\f795"; } +.bi-emoji-astonished::before { content: "\f79a"; } +.bi-emoji-grimace-fill::before { content: "\f79b"; } +.bi-emoji-grimace::before { content: "\f7a0"; } +.bi-emoji-grin-fill::before { content: "\f7a1"; } +.bi-emoji-grin::before { content: "\f7a6"; } +.bi-emoji-surprise-fill::before { content: "\f7a7"; } +.bi-emoji-surprise::before { content: "\f7ac"; } +.bi-emoji-tear-fill::before { content: "\f7ad"; } +.bi-emoji-tear::before { content: "\f7b2"; } +.bi-envelope-arrow-down-fill::before { content: "\f7b3"; } +.bi-envelope-arrow-down::before { content: "\f7b8"; } +.bi-envelope-arrow-up-fill::before { content: "\f7b9"; } +.bi-envelope-arrow-up::before { content: "\f7be"; } +.bi-feather::before { content: "\f7bf"; } +.bi-feather2::before { content: "\f7c4"; } +.bi-floppy-fill::before { content: "\f7c5"; } +.bi-floppy::before { content: "\f7d8"; } +.bi-floppy2-fill::before { content: "\f7d9"; } +.bi-floppy2::before { content: "\f7e4"; } +.bi-gitlab::before { content: "\f7e5"; } +.bi-highlighter::before { content: "\f7f8"; } +.bi-marker-tip::before { content: "\f802"; } +.bi-nvme-fill::before { content: "\f803"; } +.bi-nvme::before { content: "\f80c"; } +.bi-opencollective::before { content: "\f80d"; } +.bi-pci-card-network::before { content: "\f8cd"; } +.bi-pci-card-sound::before { content: "\f8ce"; } +.bi-radar::before { content: "\f8cf"; } +.bi-send-arrow-down-fill::before { content: "\f8d0"; } +.bi-send-arrow-down::before { content: "\f8d1"; } +.bi-send-arrow-up-fill::before { content: "\f8d2"; } +.bi-send-arrow-up::before { content: "\f8d3"; } +.bi-sim-slash-fill::before { content: "\f8d4"; } +.bi-sim-slash::before { content: "\f8d5"; } +.bi-sourceforge::before { content: "\f8d6"; } +.bi-substack::before { content: "\f8d7"; } +.bi-threads-fill::before { content: "\f8d8"; } +.bi-threads::before { content: "\f8d9"; } +.bi-transparency::before { content: "\f8da"; } +.bi-twitter-x::before { content: "\f8db"; } +.bi-type-h4::before { content: "\f8dc"; } +.bi-type-h5::before { content: "\f8dd"; } +.bi-type-h6::before { content: "\f8de"; } +.bi-backpack-fill::before { content: "\f8df"; } +.bi-backpack::before { content: "\f8e0"; } +.bi-backpack2-fill::before { content: "\f8e1"; } +.bi-backpack2::before { content: "\f8e2"; } +.bi-backpack3-fill::before { content: "\f8e3"; } +.bi-backpack3::before { content: "\f8e4"; } +.bi-backpack4-fill::before { content: "\f8e5"; } +.bi-backpack4::before { content: "\f8e6"; } +.bi-brilliance::before { content: "\f8e7"; } +.bi-cake-fill::before { content: "\f8e8"; } +.bi-cake2-fill::before { content: "\f8e9"; } +.bi-duffle-fill::before { content: "\f8ea"; } +.bi-duffle::before { content: "\f8eb"; } +.bi-exposure::before { content: "\f8ec"; } +.bi-gender-neuter::before { content: "\f8ed"; } +.bi-highlights::before { content: "\f8ee"; } +.bi-luggage-fill::before { content: "\f8ef"; } +.bi-luggage::before { content: "\f8f0"; } +.bi-mailbox-flag::before { content: "\f8f1"; } +.bi-mailbox2-flag::before { content: "\f8f2"; } +.bi-noise-reduction::before { content: "\f8f3"; } +.bi-passport-fill::before { content: "\f8f4"; } +.bi-passport::before { content: "\f8f5"; } +.bi-person-arms-up::before { content: "\f8f6"; } +.bi-person-raised-hand::before { content: "\f8f7"; } +.bi-person-standing-dress::before { content: "\f8f8"; } +.bi-person-standing::before { content: "\f8f9"; } +.bi-person-walking::before { content: "\f8fa"; } +.bi-person-wheelchair::before { content: "\f8fb"; } +.bi-shadows::before { content: "\f8fc"; } +.bi-suitcase-fill::before { content: "\f8fd"; } +.bi-suitcase-lg-fill::before { content: "\f8fe"; } +.bi-suitcase-lg::before { content: "\f8ff"; } +.bi-suitcase::before { content: "\f900"; } +.bi-suitcase2-fill::before { content: "\f901"; } +.bi-suitcase2::before { content: "\f902"; } +.bi-vignette::before { content: "\f903"; } diff --git a/site_libs/bootstrap/bootstrap-icons.woff b/site_libs/bootstrap/bootstrap-icons.woff new file mode 100644 index 0000000000000000000000000000000000000000..dbeeb055674125ad78fda0f3d166b36e5cc92336 GIT binary patch literal 176200 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

2+UrCZlNAp`)&G@jg{m-!Dn; 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πR(XEyYV3az{Ot$Y*f93mYq^Lnu{qbYp!X!TA9&! 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 F?~0aZ)an2L literal 0 HcmV?d00001 diff --git a/site_libs/bootstrap/bootstrap.min.js b/site_libs/bootstrap/bootstrap.min.js new file mode 100644 index 00000000..e8f21f70 --- /dev/null +++ b/site_libs/bootstrap/bootstrap.min.js @@ -0,0 +1,7 @@ +/*! + * Bootstrap v5.3.1 (https://getbootstrap.com/) + * Copyright 2011-2023 The Bootstrap Authors (https://github.com/twbs/bootstrap/graphs/contributors) + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */ +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t="undefined"!=typeof globalThis?globalThis:t||self).bootstrap=e()}(this,(function(){"use strict";const t=new Map,e={set(e,i,n){t.has(e)||t.set(e,new Map);const s=t.get(e);s.has(i)||0===s.size?s.set(i,n):console.error(`Bootstrap doesn't allow more than one instance per element. Bound instance: ${Array.from(s.keys())[0]}.`)},get:(e,i)=>t.has(e)&&t.get(e).get(i)||null,remove(e,i){if(!t.has(e))return;const n=t.get(e);n.delete(i),0===n.size&&t.delete(e)}},i="transitionend",n=t=>(t&&window.CSS&&window.CSS.escape&&(t=t.replace(/#([^\s"#']+)/g,((t,e)=>`#${CSS.escape(e)}`))),t),s=t=>{t.dispatchEvent(new Event(i))},o=t=>!(!t||"object"!=typeof t)&&(void 0!==t.jquery&&(t=t[0]),void 0!==t.nodeType),r=t=>o(t)?t.jquery?t[0]:t:"string"==typeof t&&t.length>0?document.querySelector(n(t)):null,a=t=>{if(!o(t)||0===t.getClientRects().length)return!1;const e="visible"===getComputedStyle(t).getPropertyValue("visibility"),i=t.closest("details:not([open])");if(!i)return e;if(i!==t){const e=t.closest("summary");if(e&&e.parentNode!==i)return!1;if(null===e)return!1}return e},l=t=>!t||t.nodeType!==Node.ELEMENT_NODE||!!t.classList.contains("disabled")||(void 0!==t.disabled?t.disabled:t.hasAttribute("disabled")&&"false"!==t.getAttribute("disabled")),c=t=>{if(!document.documentElement.attachShadow)return null;if("function"==typeof t.getRootNode){const e=t.getRootNode();return e instanceof ShadowRoot?e:null}return t instanceof ShadowRoot?t:t.parentNode?c(t.parentNode):null},h=()=>{},d=t=>{t.offsetHeight},u=()=>window.jQuery&&!document.body.hasAttribute("data-bs-no-jquery")?window.jQuery:null,f=[],p=()=>"rtl"===document.documentElement.dir,m=t=>{var e;e=()=>{const e=u();if(e){const i=t.NAME,n=e.fn[i];e.fn[i]=t.jQueryInterface,e.fn[i].Constructor=t,e.fn[i].noConflict=()=>(e.fn[i]=n,t.jQueryInterface)}},"loading"===document.readyState?(f.length||document.addEventListener("DOMContentLoaded",(()=>{for(const t of f)t()})),f.push(e)):e()},g=(t,e=[],i=t)=>"function"==typeof t?t(...e):i,_=(t,e,n=!0)=>{if(!n)return void g(t);const o=(t=>{if(!t)return 0;let{transitionDuration:e,transitionDelay:i}=window.getComputedStyle(t);const n=Number.parseFloat(e),s=Number.parseFloat(i);return n||s?(e=e.split(",")[0],i=i.split(",")[0],1e3*(Number.parseFloat(e)+Number.parseFloat(i))):0})(e)+5;let r=!1;const a=({target:n})=>{n===e&&(r=!0,e.removeEventListener(i,a),g(t))};e.addEventListener(i,a),setTimeout((()=>{r||s(e)}),o)},b=(t,e,i,n)=>{const s=t.length;let o=t.indexOf(e);return-1===o?!i&&n?t[s-1]:t[0]:(o+=i?1:-1,n&&(o=(o+s)%s),t[Math.max(0,Math.min(o,s-1))])},v=/[^.]*(?=\..*)\.|.*/,y=/\..*/,w=/::\d+$/,A={};let E=1;const T={mouseenter:"mouseover",mouseleave:"mouseout"},C=new Set(["click","dblclick","mouseup","mousedown","contextmenu","mousewheel","DOMMouseScroll","mouseover","mouseout","mousemove","selectstart","selectend","keydown","keypress","keyup","orientationchange","touchstart","touchmove","touchend","touchcancel","pointerdown","pointermove","pointerup","pointerleave","pointercancel","gesturestart","gesturechange","gestureend","focus","blur","change","reset","select","submit","focusin","focusout","load","unload","beforeunload","resize","move","DOMContentLoaded","readystatechange","error","abort","scroll"]);function O(t,e){return e&&`${e}::${E++}`||t.uidEvent||E++}function x(t){const e=O(t);return t.uidEvent=e,A[e]=A[e]||{},A[e]}function k(t,e,i=null){return Object.values(t).find((t=>t.callable===e&&t.delegationSelector===i))}function L(t,e,i){const n="string"==typeof e,s=n?i:e||i;let o=I(t);return C.has(o)||(o=t),[n,s,o]}function S(t,e,i,n,s){if("string"!=typeof e||!t)return;let[o,r,a]=L(e,i,n);if(e in T){const t=t=>function(e){if(!e.relatedTarget||e.relatedTarget!==e.delegateTarget&&!e.delegateTarget.contains(e.relatedTarget))return t.call(this,e)};r=t(r)}const l=x(t),c=l[a]||(l[a]={}),h=k(c,r,o?i:null);if(h)return void(h.oneOff=h.oneOff&&s);const d=O(r,e.replace(v,"")),u=o?function(t,e,i){return function n(s){const o=t.querySelectorAll(e);for(let{target:r}=s;r&&r!==this;r=r.parentNode)for(const a of o)if(a===r)return P(s,{delegateTarget:r}),n.oneOff&&N.off(t,s.type,e,i),i.apply(r,[s])}}(t,i,r):function(t,e){return function i(n){return P(n,{delegateTarget:t}),i.oneOff&&N.off(t,n.type,e),e.apply(t,[n])}}(t,r);u.delegationSelector=o?i:null,u.callable=r,u.oneOff=s,u.uidEvent=d,c[d]=u,t.addEventListener(a,u,o)}function D(t,e,i,n,s){const o=k(e[i],n,s);o&&(t.removeEventListener(i,o,Boolean(s)),delete e[i][o.uidEvent])}function $(t,e,i,n){const s=e[i]||{};for(const[o,r]of Object.entries(s))o.includes(n)&&D(t,e,i,r.callable,r.delegationSelector)}function I(t){return t=t.replace(y,""),T[t]||t}const N={on(t,e,i,n){S(t,e,i,n,!1)},one(t,e,i,n){S(t,e,i,n,!0)},off(t,e,i,n){if("string"!=typeof e||!t)return;const[s,o,r]=L(e,i,n),a=r!==e,l=x(t),c=l[r]||{},h=e.startsWith(".");if(void 0===o){if(h)for(const i of Object.keys(l))$(t,l,i,e.slice(1));for(const[i,n]of Object.entries(c)){const s=i.replace(w,"");a&&!e.includes(s)||D(t,l,r,n.callable,n.delegationSelector)}}else{if(!Object.keys(c).length)return;D(t,l,r,o,s?i:null)}},trigger(t,e,i){if("string"!=typeof e||!t)return null;const n=u();let s=null,o=!0,r=!0,a=!1;e!==I(e)&&n&&(s=n.Event(e,i),n(t).trigger(s),o=!s.isPropagationStopped(),r=!s.isImmediatePropagationStopped(),a=s.isDefaultPrevented());const l=P(new Event(e,{bubbles:o,cancelable:!0}),i);return a&&l.preventDefault(),r&&t.dispatchEvent(l),l.defaultPrevented&&s&&s.preventDefault(),l}};function P(t,e={}){for(const[i,n]of Object.entries(e))try{t[i]=n}catch(e){Object.defineProperty(t,i,{configurable:!0,get:()=>n})}return t}function M(t){if("true"===t)return!0;if("false"===t)return!1;if(t===Number(t).toString())return Number(t);if(""===t||"null"===t)return null;if("string"!=typeof t)return t;try{return JSON.parse(decodeURIComponent(t))}catch(e){return t}}function j(t){return t.replace(/[A-Z]/g,(t=>`-${t.toLowerCase()}`))}const F={setDataAttribute(t,e,i){t.setAttribute(`data-bs-${j(e)}`,i)},removeDataAttribute(t,e){t.removeAttribute(`data-bs-${j(e)}`)},getDataAttributes(t){if(!t)return{};const e={},i=Object.keys(t.dataset).filter((t=>t.startsWith("bs")&&!t.startsWith("bsConfig")));for(const n of i){let i=n.replace(/^bs/,"");i=i.charAt(0).toLowerCase()+i.slice(1,i.length),e[i]=M(t.dataset[n])}return e},getDataAttribute:(t,e)=>M(t.getAttribute(`data-bs-${j(e)}`))};class H{static get Default(){return{}}static get DefaultType(){return{}}static get NAME(){throw new Error('You have to implement the static method "NAME", for each component!')}_getConfig(t){return t=this._mergeConfigObj(t),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}_configAfterMerge(t){return t}_mergeConfigObj(t,e){const i=o(e)?F.getDataAttribute(e,"config"):{};return{...this.constructor.Default,..."object"==typeof i?i:{},...o(e)?F.getDataAttributes(e):{},..."object"==typeof t?t:{}}}_typeCheckConfig(t,e=this.constructor.DefaultType){for(const[n,s]of Object.entries(e)){const e=t[n],r=o(e)?"element":null==(i=e)?`${i}`:Object.prototype.toString.call(i).match(/\s([a-z]+)/i)[1].toLowerCase();if(!new RegExp(s).test(r))throw new TypeError(`${this.constructor.NAME.toUpperCase()}: Option "${n}" provided type "${r}" but expected type "${s}".`)}var i}}class W extends H{constructor(t,i){super(),(t=r(t))&&(this._element=t,this._config=this._getConfig(i),e.set(this._element,this.constructor.DATA_KEY,this))}dispose(){e.remove(this._element,this.constructor.DATA_KEY),N.off(this._element,this.constructor.EVENT_KEY);for(const t of Object.getOwnPropertyNames(this))this[t]=null}_queueCallback(t,e,i=!0){_(t,e,i)}_getConfig(t){return t=this._mergeConfigObj(t,this._element),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}static getInstance(t){return e.get(r(t),this.DATA_KEY)}static getOrCreateInstance(t,e={}){return this.getInstance(t)||new this(t,"object"==typeof e?e:null)}static get VERSION(){return"5.3.1"}static get DATA_KEY(){return`bs.${this.NAME}`}static get EVENT_KEY(){return`.${this.DATA_KEY}`}static eventName(t){return`${t}${this.EVENT_KEY}`}}const B=t=>{let e=t.getAttribute("data-bs-target");if(!e||"#"===e){let i=t.getAttribute("href");if(!i||!i.includes("#")&&!i.startsWith("."))return null;i.includes("#")&&!i.startsWith("#")&&(i=`#${i.split("#")[1]}`),e=i&&"#"!==i?i.trim():null}return n(e)},z={find:(t,e=document.documentElement)=>[].concat(...Element.prototype.querySelectorAll.call(e,t)),findOne:(t,e=document.documentElement)=>Element.prototype.querySelector.call(e,t),children:(t,e)=>[].concat(...t.children).filter((t=>t.matches(e))),parents(t,e){const i=[];let n=t.parentNode.closest(e);for(;n;)i.push(n),n=n.parentNode.closest(e);return i},prev(t,e){let i=t.previousElementSibling;for(;i;){if(i.matches(e))return[i];i=i.previousElementSibling}return[]},next(t,e){let i=t.nextElementSibling;for(;i;){if(i.matches(e))return[i];i=i.nextElementSibling}return[]},focusableChildren(t){const e=["a","button","input","textarea","select","details","[tabindex]",'[contenteditable="true"]'].map((t=>`${t}:not([tabindex^="-"])`)).join(",");return this.find(e,t).filter((t=>!l(t)&&a(t)))},getSelectorFromElement(t){const e=B(t);return e&&z.findOne(e)?e:null},getElementFromSelector(t){const e=B(t);return e?z.findOne(e):null},getMultipleElementsFromSelector(t){const e=B(t);return e?z.find(e):[]}},R=(t,e="hide")=>{const i=`click.dismiss${t.EVENT_KEY}`,n=t.NAME;N.on(document,i,`[data-bs-dismiss="${n}"]`,(function(i){if(["A","AREA"].includes(this.tagName)&&i.preventDefault(),l(this))return;const s=z.getElementFromSelector(this)||this.closest(`.${n}`);t.getOrCreateInstance(s)[e]()}))},q=".bs.alert",V=`close${q}`,K=`closed${q}`;class Q extends W{static get NAME(){return"alert"}close(){if(N.trigger(this._element,V).defaultPrevented)return;this._element.classList.remove("show");const t=this._element.classList.contains("fade");this._queueCallback((()=>this._destroyElement()),this._element,t)}_destroyElement(){this._element.remove(),N.trigger(this._element,K),this.dispose()}static jQueryInterface(t){return this.each((function(){const e=Q.getOrCreateInstance(this);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}R(Q,"close"),m(Q);const X='[data-bs-toggle="button"]';class Y extends W{static get NAME(){return"button"}toggle(){this._element.setAttribute("aria-pressed",this._element.classList.toggle("active"))}static jQueryInterface(t){return this.each((function(){const e=Y.getOrCreateInstance(this);"toggle"===t&&e[t]()}))}}N.on(document,"click.bs.button.data-api",X,(t=>{t.preventDefault();const e=t.target.closest(X);Y.getOrCreateInstance(e).toggle()})),m(Y);const U=".bs.swipe",G=`touchstart${U}`,J=`touchmove${U}`,Z=`touchend${U}`,tt=`pointerdown${U}`,et=`pointerup${U}`,it={endCallback:null,leftCallback:null,rightCallback:null},nt={endCallback:"(function|null)",leftCallback:"(function|null)",rightCallback:"(function|null)"};class st extends H{constructor(t,e){super(),this._element=t,t&&st.isSupported()&&(this._config=this._getConfig(e),this._deltaX=0,this._supportPointerEvents=Boolean(window.PointerEvent),this._initEvents())}static get Default(){return it}static get DefaultType(){return nt}static get NAME(){return"swipe"}dispose(){N.off(this._element,U)}_start(t){this._supportPointerEvents?this._eventIsPointerPenTouch(t)&&(this._deltaX=t.clientX):this._deltaX=t.touches[0].clientX}_end(t){this._eventIsPointerPenTouch(t)&&(this._deltaX=t.clientX-this._deltaX),this._handleSwipe(),g(this._config.endCallback)}_move(t){this._deltaX=t.touches&&t.touches.length>1?0:t.touches[0].clientX-this._deltaX}_handleSwipe(){const t=Math.abs(this._deltaX);if(t<=40)return;const e=t/this._deltaX;this._deltaX=0,e&&g(e>0?this._config.rightCallback:this._config.leftCallback)}_initEvents(){this._supportPointerEvents?(N.on(this._element,tt,(t=>this._start(t))),N.on(this._element,et,(t=>this._end(t))),this._element.classList.add("pointer-event")):(N.on(this._element,G,(t=>this._start(t))),N.on(this._element,J,(t=>this._move(t))),N.on(this._element,Z,(t=>this._end(t))))}_eventIsPointerPenTouch(t){return this._supportPointerEvents&&("pen"===t.pointerType||"touch"===t.pointerType)}static isSupported(){return"ontouchstart"in document.documentElement||navigator.maxTouchPoints>0}}const ot=".bs.carousel",rt=".data-api",at="next",lt="prev",ct="left",ht="right",dt=`slide${ot}`,ut=`slid${ot}`,ft=`keydown${ot}`,pt=`mouseenter${ot}`,mt=`mouseleave${ot}`,gt=`dragstart${ot}`,_t=`load${ot}${rt}`,bt=`click${ot}${rt}`,vt="carousel",yt="active",wt=".active",At=".carousel-item",Et=wt+At,Tt={ArrowLeft:ht,ArrowRight:ct},Ct={interval:5e3,keyboard:!0,pause:"hover",ride:!1,touch:!0,wrap:!0},Ot={interval:"(number|boolean)",keyboard:"boolean",pause:"(string|boolean)",ride:"(boolean|string)",touch:"boolean",wrap:"boolean"};class xt extends W{constructor(t,e){super(t,e),this._interval=null,this._activeElement=null,this._isSliding=!1,this.touchTimeout=null,this._swipeHelper=null,this._indicatorsElement=z.findOne(".carousel-indicators",this._element),this._addEventListeners(),this._config.ride===vt&&this.cycle()}static get Default(){return Ct}static get DefaultType(){return Ot}static get NAME(){return"carousel"}next(){this._slide(at)}nextWhenVisible(){!document.hidden&&a(this._element)&&this.next()}prev(){this._slide(lt)}pause(){this._isSliding&&s(this._element),this._clearInterval()}cycle(){this._clearInterval(),this._updateInterval(),this._interval=setInterval((()=>this.nextWhenVisible()),this._config.interval)}_maybeEnableCycle(){this._config.ride&&(this._isSliding?N.one(this._element,ut,(()=>this.cycle())):this.cycle())}to(t){const e=this._getItems();if(t>e.length-1||t<0)return;if(this._isSliding)return void N.one(this._element,ut,(()=>this.to(t)));const i=this._getItemIndex(this._getActive());if(i===t)return;const n=t>i?at:lt;this._slide(n,e[t])}dispose(){this._swipeHelper&&this._swipeHelper.dispose(),super.dispose()}_configAfterMerge(t){return t.defaultInterval=t.interval,t}_addEventListeners(){this._config.keyboard&&N.on(this._element,ft,(t=>this._keydown(t))),"hover"===this._config.pause&&(N.on(this._element,pt,(()=>this.pause())),N.on(this._element,mt,(()=>this._maybeEnableCycle()))),this._config.touch&&st.isSupported()&&this._addTouchEventListeners()}_addTouchEventListeners(){for(const t of z.find(".carousel-item img",this._element))N.on(t,gt,(t=>t.preventDefault()));const t={leftCallback:()=>this._slide(this._directionToOrder(ct)),rightCallback:()=>this._slide(this._directionToOrder(ht)),endCallback:()=>{"hover"===this._config.pause&&(this.pause(),this.touchTimeout&&clearTimeout(this.touchTimeout),this.touchTimeout=setTimeout((()=>this._maybeEnableCycle()),500+this._config.interval))}};this._swipeHelper=new st(this._element,t)}_keydown(t){if(/input|textarea/i.test(t.target.tagName))return;const e=Tt[t.key];e&&(t.preventDefault(),this._slide(this._directionToOrder(e)))}_getItemIndex(t){return this._getItems().indexOf(t)}_setActiveIndicatorElement(t){if(!this._indicatorsElement)return;const e=z.findOne(wt,this._indicatorsElement);e.classList.remove(yt),e.removeAttribute("aria-current");const i=z.findOne(`[data-bs-slide-to="${t}"]`,this._indicatorsElement);i&&(i.classList.add(yt),i.setAttribute("aria-current","true"))}_updateInterval(){const t=this._activeElement||this._getActive();if(!t)return;const e=Number.parseInt(t.getAttribute("data-bs-interval"),10);this._config.interval=e||this._config.defaultInterval}_slide(t,e=null){if(this._isSliding)return;const i=this._getActive(),n=t===at,s=e||b(this._getItems(),i,n,this._config.wrap);if(s===i)return;const o=this._getItemIndex(s),r=e=>N.trigger(this._element,e,{relatedTarget:s,direction:this._orderToDirection(t),from:this._getItemIndex(i),to:o});if(r(dt).defaultPrevented)return;if(!i||!s)return;const a=Boolean(this._interval);this.pause(),this._isSliding=!0,this._setActiveIndicatorElement(o),this._activeElement=s;const l=n?"carousel-item-start":"carousel-item-end",c=n?"carousel-item-next":"carousel-item-prev";s.classList.add(c),d(s),i.classList.add(l),s.classList.add(l),this._queueCallback((()=>{s.classList.remove(l,c),s.classList.add(yt),i.classList.remove(yt,c,l),this._isSliding=!1,r(ut)}),i,this._isAnimated()),a&&this.cycle()}_isAnimated(){return this._element.classList.contains("slide")}_getActive(){return z.findOne(Et,this._element)}_getItems(){return z.find(At,this._element)}_clearInterval(){this._interval&&(clearInterval(this._interval),this._interval=null)}_directionToOrder(t){return p()?t===ct?lt:at:t===ct?at:lt}_orderToDirection(t){return p()?t===lt?ct:ht:t===lt?ht:ct}static jQueryInterface(t){return this.each((function(){const e=xt.getOrCreateInstance(this,t);if("number"!=typeof t){if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}else e.to(t)}))}}N.on(document,bt,"[data-bs-slide], [data-bs-slide-to]",(function(t){const e=z.getElementFromSelector(this);if(!e||!e.classList.contains(vt))return;t.preventDefault();const i=xt.getOrCreateInstance(e),n=this.getAttribute("data-bs-slide-to");return n?(i.to(n),void i._maybeEnableCycle()):"next"===F.getDataAttribute(this,"slide")?(i.next(),void i._maybeEnableCycle()):(i.prev(),void i._maybeEnableCycle())})),N.on(window,_t,(()=>{const t=z.find('[data-bs-ride="carousel"]');for(const e of t)xt.getOrCreateInstance(e)})),m(xt);const kt=".bs.collapse",Lt=`show${kt}`,St=`shown${kt}`,Dt=`hide${kt}`,$t=`hidden${kt}`,It=`click${kt}.data-api`,Nt="show",Pt="collapse",Mt="collapsing",jt=`:scope .${Pt} .${Pt}`,Ft='[data-bs-toggle="collapse"]',Ht={parent:null,toggle:!0},Wt={parent:"(null|element)",toggle:"boolean"};class Bt extends W{constructor(t,e){super(t,e),this._isTransitioning=!1,this._triggerArray=[];const i=z.find(Ft);for(const t of i){const e=z.getSelectorFromElement(t),i=z.find(e).filter((t=>t===this._element));null!==e&&i.length&&this._triggerArray.push(t)}this._initializeChildren(),this._config.parent||this._addAriaAndCollapsedClass(this._triggerArray,this._isShown()),this._config.toggle&&this.toggle()}static get Default(){return Ht}static get DefaultType(){return Wt}static get NAME(){return"collapse"}toggle(){this._isShown()?this.hide():this.show()}show(){if(this._isTransitioning||this._isShown())return;let t=[];if(this._config.parent&&(t=this._getFirstLevelChildren(".collapse.show, .collapse.collapsing").filter((t=>t!==this._element)).map((t=>Bt.getOrCreateInstance(t,{toggle:!1})))),t.length&&t[0]._isTransitioning)return;if(N.trigger(this._element,Lt).defaultPrevented)return;for(const e of t)e.hide();const e=this._getDimension();this._element.classList.remove(Pt),this._element.classList.add(Mt),this._element.style[e]=0,this._addAriaAndCollapsedClass(this._triggerArray,!0),this._isTransitioning=!0;const i=`scroll${e[0].toUpperCase()+e.slice(1)}`;this._queueCallback((()=>{this._isTransitioning=!1,this._element.classList.remove(Mt),this._element.classList.add(Pt,Nt),this._element.style[e]="",N.trigger(this._element,St)}),this._element,!0),this._element.style[e]=`${this._element[i]}px`}hide(){if(this._isTransitioning||!this._isShown())return;if(N.trigger(this._element,Dt).defaultPrevented)return;const t=this._getDimension();this._element.style[t]=`${this._element.getBoundingClientRect()[t]}px`,d(this._element),this._element.classList.add(Mt),this._element.classList.remove(Pt,Nt);for(const t of this._triggerArray){const e=z.getElementFromSelector(t);e&&!this._isShown(e)&&this._addAriaAndCollapsedClass([t],!1)}this._isTransitioning=!0,this._element.style[t]="",this._queueCallback((()=>{this._isTransitioning=!1,this._element.classList.remove(Mt),this._element.classList.add(Pt),N.trigger(this._element,$t)}),this._element,!0)}_isShown(t=this._element){return t.classList.contains(Nt)}_configAfterMerge(t){return t.toggle=Boolean(t.toggle),t.parent=r(t.parent),t}_getDimension(){return this._element.classList.contains("collapse-horizontal")?"width":"height"}_initializeChildren(){if(!this._config.parent)return;const t=this._getFirstLevelChildren(Ft);for(const e of t){const t=z.getElementFromSelector(e);t&&this._addAriaAndCollapsedClass([e],this._isShown(t))}}_getFirstLevelChildren(t){const e=z.find(jt,this._config.parent);return z.find(t,this._config.parent).filter((t=>!e.includes(t)))}_addAriaAndCollapsedClass(t,e){if(t.length)for(const i of t)i.classList.toggle("collapsed",!e),i.setAttribute("aria-expanded",e)}static jQueryInterface(t){const e={};return"string"==typeof t&&/show|hide/.test(t)&&(e.toggle=!1),this.each((function(){const i=Bt.getOrCreateInstance(this,e);if("string"==typeof t){if(void 0===i[t])throw new TypeError(`No method named "${t}"`);i[t]()}}))}}N.on(document,It,Ft,(function(t){("A"===t.target.tagName||t.delegateTarget&&"A"===t.delegateTarget.tagName)&&t.preventDefault();for(const t of z.getMultipleElementsFromSelector(this))Bt.getOrCreateInstance(t,{toggle:!1}).toggle()})),m(Bt);var zt="top",Rt="bottom",qt="right",Vt="left",Kt="auto",Qt=[zt,Rt,qt,Vt],Xt="start",Yt="end",Ut="clippingParents",Gt="viewport",Jt="popper",Zt="reference",te=Qt.reduce((function(t,e){return t.concat([e+"-"+Xt,e+"-"+Yt])}),[]),ee=[].concat(Qt,[Kt]).reduce((function(t,e){return t.concat([e,e+"-"+Xt,e+"-"+Yt])}),[]),ie="beforeRead",ne="read",se="afterRead",oe="beforeMain",re="main",ae="afterMain",le="beforeWrite",ce="write",he="afterWrite",de=[ie,ne,se,oe,re,ae,le,ce,he];function ue(t){return t?(t.nodeName||"").toLowerCase():null}function fe(t){if(null==t)return window;if("[object Window]"!==t.toString()){var e=t.ownerDocument;return e&&e.defaultView||window}return t}function pe(t){return t instanceof fe(t).Element||t instanceof Element}function me(t){return t instanceof fe(t).HTMLElement||t instanceof HTMLElement}function ge(t){return"undefined"!=typeof ShadowRoot&&(t instanceof fe(t).ShadowRoot||t instanceof ShadowRoot)}const _e={name:"applyStyles",enabled:!0,phase:"write",fn:function(t){var e=t.state;Object.keys(e.elements).forEach((function(t){var i=e.styles[t]||{},n=e.attributes[t]||{},s=e.elements[t];me(s)&&ue(s)&&(Object.assign(s.style,i),Object.keys(n).forEach((function(t){var e=n[t];!1===e?s.removeAttribute(t):s.setAttribute(t,!0===e?"":e)})))}))},effect:function(t){var e=t.state,i={popper:{position:e.options.strategy,left:"0",top:"0",margin:"0"},arrow:{position:"absolute"},reference:{}};return Object.assign(e.elements.popper.style,i.popper),e.styles=i,e.elements.arrow&&Object.assign(e.elements.arrow.style,i.arrow),function(){Object.keys(e.elements).forEach((function(t){var n=e.elements[t],s=e.attributes[t]||{},o=Object.keys(e.styles.hasOwnProperty(t)?e.styles[t]:i[t]).reduce((function(t,e){return t[e]="",t}),{});me(n)&&ue(n)&&(Object.assign(n.style,o),Object.keys(s).forEach((function(t){n.removeAttribute(t)})))}))}},requires:["computeStyles"]};function be(t){return t.split("-")[0]}var ve=Math.max,ye=Math.min,we=Math.round;function Ae(){var t=navigator.userAgentData;return null!=t&&t.brands&&Array.isArray(t.brands)?t.brands.map((function(t){return t.brand+"/"+t.version})).join(" "):navigator.userAgent}function Ee(){return!/^((?!chrome|android).)*safari/i.test(Ae())}function Te(t,e,i){void 0===e&&(e=!1),void 0===i&&(i=!1);var n=t.getBoundingClientRect(),s=1,o=1;e&&me(t)&&(s=t.offsetWidth>0&&we(n.width)/t.offsetWidth||1,o=t.offsetHeight>0&&we(n.height)/t.offsetHeight||1);var r=(pe(t)?fe(t):window).visualViewport,a=!Ee()&&i,l=(n.left+(a&&r?r.offsetLeft:0))/s,c=(n.top+(a&&r?r.offsetTop:0))/o,h=n.width/s,d=n.height/o;return{width:h,height:d,top:c,right:l+h,bottom:c+d,left:l,x:l,y:c}}function Ce(t){var e=Te(t),i=t.offsetWidth,n=t.offsetHeight;return Math.abs(e.width-i)<=1&&(i=e.width),Math.abs(e.height-n)<=1&&(n=e.height),{x:t.offsetLeft,y:t.offsetTop,width:i,height:n}}function Oe(t,e){var i=e.getRootNode&&e.getRootNode();if(t.contains(e))return!0;if(i&&ge(i)){var n=e;do{if(n&&t.isSameNode(n))return!0;n=n.parentNode||n.host}while(n)}return!1}function xe(t){return fe(t).getComputedStyle(t)}function ke(t){return["table","td","th"].indexOf(ue(t))>=0}function Le(t){return((pe(t)?t.ownerDocument:t.document)||window.document).documentElement}function Se(t){return"html"===ue(t)?t:t.assignedSlot||t.parentNode||(ge(t)?t.host:null)||Le(t)}function De(t){return me(t)&&"fixed"!==xe(t).position?t.offsetParent:null}function $e(t){for(var e=fe(t),i=De(t);i&&ke(i)&&"static"===xe(i).position;)i=De(i);return i&&("html"===ue(i)||"body"===ue(i)&&"static"===xe(i).position)?e:i||function(t){var e=/firefox/i.test(Ae());if(/Trident/i.test(Ae())&&me(t)&&"fixed"===xe(t).position)return null;var i=Se(t);for(ge(i)&&(i=i.host);me(i)&&["html","body"].indexOf(ue(i))<0;){var n=xe(i);if("none"!==n.transform||"none"!==n.perspective||"paint"===n.contain||-1!==["transform","perspective"].indexOf(n.willChange)||e&&"filter"===n.willChange||e&&n.filter&&"none"!==n.filter)return i;i=i.parentNode}return null}(t)||e}function Ie(t){return["top","bottom"].indexOf(t)>=0?"x":"y"}function Ne(t,e,i){return ve(t,ye(e,i))}function Pe(t){return Object.assign({},{top:0,right:0,bottom:0,left:0},t)}function Me(t,e){return e.reduce((function(e,i){return e[i]=t,e}),{})}const je={name:"arrow",enabled:!0,phase:"main",fn:function(t){var e,i=t.state,n=t.name,s=t.options,o=i.elements.arrow,r=i.modifiersData.popperOffsets,a=be(i.placement),l=Ie(a),c=[Vt,qt].indexOf(a)>=0?"height":"width";if(o&&r){var h=function(t,e){return Pe("number"!=typeof(t="function"==typeof t?t(Object.assign({},e.rects,{placement:e.placement})):t)?t:Me(t,Qt))}(s.padding,i),d=Ce(o),u="y"===l?zt:Vt,f="y"===l?Rt:qt,p=i.rects.reference[c]+i.rects.reference[l]-r[l]-i.rects.popper[c],m=r[l]-i.rects.reference[l],g=$e(o),_=g?"y"===l?g.clientHeight||0:g.clientWidth||0:0,b=p/2-m/2,v=h[u],y=_-d[c]-h[f],w=_/2-d[c]/2+b,A=Ne(v,w,y),E=l;i.modifiersData[n]=((e={})[E]=A,e.centerOffset=A-w,e)}},effect:function(t){var e=t.state,i=t.options.element,n=void 0===i?"[data-popper-arrow]":i;null!=n&&("string"!=typeof n||(n=e.elements.popper.querySelector(n)))&&Oe(e.elements.popper,n)&&(e.elements.arrow=n)},requires:["popperOffsets"],requiresIfExists:["preventOverflow"]};function Fe(t){return t.split("-")[1]}var He={top:"auto",right:"auto",bottom:"auto",left:"auto"};function We(t){var e,i=t.popper,n=t.popperRect,s=t.placement,o=t.variation,r=t.offsets,a=t.position,l=t.gpuAcceleration,c=t.adaptive,h=t.roundOffsets,d=t.isFixed,u=r.x,f=void 0===u?0:u,p=r.y,m=void 0===p?0:p,g="function"==typeof h?h({x:f,y:m}):{x:f,y:m};f=g.x,m=g.y;var _=r.hasOwnProperty("x"),b=r.hasOwnProperty("y"),v=Vt,y=zt,w=window;if(c){var A=$e(i),E="clientHeight",T="clientWidth";A===fe(i)&&"static"!==xe(A=Le(i)).position&&"absolute"===a&&(E="scrollHeight",T="scrollWidth"),(s===zt||(s===Vt||s===qt)&&o===Yt)&&(y=Rt,m-=(d&&A===w&&w.visualViewport?w.visualViewport.height:A[E])-n.height,m*=l?1:-1),s!==Vt&&(s!==zt&&s!==Rt||o!==Yt)||(v=qt,f-=(d&&A===w&&w.visualViewport?w.visualViewport.width:A[T])-n.width,f*=l?1:-1)}var C,O=Object.assign({position:a},c&&He),x=!0===h?function(t,e){var i=t.x,n=t.y,s=e.devicePixelRatio||1;return{x:we(i*s)/s||0,y:we(n*s)/s||0}}({x:f,y:m},fe(i)):{x:f,y:m};return f=x.x,m=x.y,l?Object.assign({},O,((C={})[y]=b?"0":"",C[v]=_?"0":"",C.transform=(w.devicePixelRatio||1)<=1?"translate("+f+"px, "+m+"px)":"translate3d("+f+"px, "+m+"px, 0)",C)):Object.assign({},O,((e={})[y]=b?m+"px":"",e[v]=_?f+"px":"",e.transform="",e))}const Be={name:"computeStyles",enabled:!0,phase:"beforeWrite",fn:function(t){var e=t.state,i=t.options,n=i.gpuAcceleration,s=void 0===n||n,o=i.adaptive,r=void 0===o||o,a=i.roundOffsets,l=void 0===a||a,c={placement:be(e.placement),variation:Fe(e.placement),popper:e.elements.popper,popperRect:e.rects.popper,gpuAcceleration:s,isFixed:"fixed"===e.options.strategy};null!=e.modifiersData.popperOffsets&&(e.styles.popper=Object.assign({},e.styles.popper,We(Object.assign({},c,{offsets:e.modifiersData.popperOffsets,position:e.options.strategy,adaptive:r,roundOffsets:l})))),null!=e.modifiersData.arrow&&(e.styles.arrow=Object.assign({},e.styles.arrow,We(Object.assign({},c,{offsets:e.modifiersData.arrow,position:"absolute",adaptive:!1,roundOffsets:l})))),e.attributes.popper=Object.assign({},e.attributes.popper,{"data-popper-placement":e.placement})},data:{}};var ze={passive:!0};const Re={name:"eventListeners",enabled:!0,phase:"write",fn:function(){},effect:function(t){var e=t.state,i=t.instance,n=t.options,s=n.scroll,o=void 0===s||s,r=n.resize,a=void 0===r||r,l=fe(e.elements.popper),c=[].concat(e.scrollParents.reference,e.scrollParents.popper);return o&&c.forEach((function(t){t.addEventListener("scroll",i.update,ze)})),a&&l.addEventListener("resize",i.update,ze),function(){o&&c.forEach((function(t){t.removeEventListener("scroll",i.update,ze)})),a&&l.removeEventListener("resize",i.update,ze)}},data:{}};var qe={left:"right",right:"left",bottom:"top",top:"bottom"};function Ve(t){return t.replace(/left|right|bottom|top/g,(function(t){return qe[t]}))}var Ke={start:"end",end:"start"};function Qe(t){return t.replace(/start|end/g,(function(t){return Ke[t]}))}function Xe(t){var e=fe(t);return{scrollLeft:e.pageXOffset,scrollTop:e.pageYOffset}}function Ye(t){return Te(Le(t)).left+Xe(t).scrollLeft}function Ue(t){var e=xe(t),i=e.overflow,n=e.overflowX,s=e.overflowY;return/auto|scroll|overlay|hidden/.test(i+s+n)}function Ge(t){return["html","body","#document"].indexOf(ue(t))>=0?t.ownerDocument.body:me(t)&&Ue(t)?t:Ge(Se(t))}function Je(t,e){var i;void 0===e&&(e=[]);var n=Ge(t),s=n===(null==(i=t.ownerDocument)?void 0:i.body),o=fe(n),r=s?[o].concat(o.visualViewport||[],Ue(n)?n:[]):n,a=e.concat(r);return s?a:a.concat(Je(Se(r)))}function Ze(t){return Object.assign({},t,{left:t.x,top:t.y,right:t.x+t.width,bottom:t.y+t.height})}function ti(t,e,i){return e===Gt?Ze(function(t,e){var i=fe(t),n=Le(t),s=i.visualViewport,o=n.clientWidth,r=n.clientHeight,a=0,l=0;if(s){o=s.width,r=s.height;var c=Ee();(c||!c&&"fixed"===e)&&(a=s.offsetLeft,l=s.offsetTop)}return{width:o,height:r,x:a+Ye(t),y:l}}(t,i)):pe(e)?function(t,e){var i=Te(t,!1,"fixed"===e);return i.top=i.top+t.clientTop,i.left=i.left+t.clientLeft,i.bottom=i.top+t.clientHeight,i.right=i.left+t.clientWidth,i.width=t.clientWidth,i.height=t.clientHeight,i.x=i.left,i.y=i.top,i}(e,i):Ze(function(t){var e,i=Le(t),n=Xe(t),s=null==(e=t.ownerDocument)?void 0:e.body,o=ve(i.scrollWidth,i.clientWidth,s?s.scrollWidth:0,s?s.clientWidth:0),r=ve(i.scrollHeight,i.clientHeight,s?s.scrollHeight:0,s?s.clientHeight:0),a=-n.scrollLeft+Ye(t),l=-n.scrollTop;return"rtl"===xe(s||i).direction&&(a+=ve(i.clientWidth,s?s.clientWidth:0)-o),{width:o,height:r,x:a,y:l}}(Le(t)))}function ei(t){var e,i=t.reference,n=t.element,s=t.placement,o=s?be(s):null,r=s?Fe(s):null,a=i.x+i.width/2-n.width/2,l=i.y+i.height/2-n.height/2;switch(o){case zt:e={x:a,y:i.y-n.height};break;case Rt:e={x:a,y:i.y+i.height};break;case qt:e={x:i.x+i.width,y:l};break;case Vt:e={x:i.x-n.width,y:l};break;default:e={x:i.x,y:i.y}}var c=o?Ie(o):null;if(null!=c){var h="y"===c?"height":"width";switch(r){case Xt:e[c]=e[c]-(i[h]/2-n[h]/2);break;case Yt:e[c]=e[c]+(i[h]/2-n[h]/2)}}return e}function ii(t,e){void 0===e&&(e={});var i=e,n=i.placement,s=void 0===n?t.placement:n,o=i.strategy,r=void 0===o?t.strategy:o,a=i.boundary,l=void 0===a?Ut:a,c=i.rootBoundary,h=void 0===c?Gt:c,d=i.elementContext,u=void 0===d?Jt:d,f=i.altBoundary,p=void 0!==f&&f,m=i.padding,g=void 0===m?0:m,_=Pe("number"!=typeof g?g:Me(g,Qt)),b=u===Jt?Zt:Jt,v=t.rects.popper,y=t.elements[p?b:u],w=function(t,e,i,n){var s="clippingParents"===e?function(t){var e=Je(Se(t)),i=["absolute","fixed"].indexOf(xe(t).position)>=0&&me(t)?$e(t):t;return pe(i)?e.filter((function(t){return pe(t)&&Oe(t,i)&&"body"!==ue(t)})):[]}(t):[].concat(e),o=[].concat(s,[i]),r=o[0],a=o.reduce((function(e,i){var s=ti(t,i,n);return e.top=ve(s.top,e.top),e.right=ye(s.right,e.right),e.bottom=ye(s.bottom,e.bottom),e.left=ve(s.left,e.left),e}),ti(t,r,n));return a.width=a.right-a.left,a.height=a.bottom-a.top,a.x=a.left,a.y=a.top,a}(pe(y)?y:y.contextElement||Le(t.elements.popper),l,h,r),A=Te(t.elements.reference),E=ei({reference:A,element:v,strategy:"absolute",placement:s}),T=Ze(Object.assign({},v,E)),C=u===Jt?T:A,O={top:w.top-C.top+_.top,bottom:C.bottom-w.bottom+_.bottom,left:w.left-C.left+_.left,right:C.right-w.right+_.right},x=t.modifiersData.offset;if(u===Jt&&x){var k=x[s];Object.keys(O).forEach((function(t){var e=[qt,Rt].indexOf(t)>=0?1:-1,i=[zt,Rt].indexOf(t)>=0?"y":"x";O[t]+=k[i]*e}))}return O}function ni(t,e){void 0===e&&(e={});var i=e,n=i.placement,s=i.boundary,o=i.rootBoundary,r=i.padding,a=i.flipVariations,l=i.allowedAutoPlacements,c=void 0===l?ee:l,h=Fe(n),d=h?a?te:te.filter((function(t){return Fe(t)===h})):Qt,u=d.filter((function(t){return c.indexOf(t)>=0}));0===u.length&&(u=d);var f=u.reduce((function(e,i){return e[i]=ii(t,{placement:i,boundary:s,rootBoundary:o,padding:r})[be(i)],e}),{});return Object.keys(f).sort((function(t,e){return f[t]-f[e]}))}const si={name:"flip",enabled:!0,phase:"main",fn:function(t){var e=t.state,i=t.options,n=t.name;if(!e.modifiersData[n]._skip){for(var s=i.mainAxis,o=void 0===s||s,r=i.altAxis,a=void 0===r||r,l=i.fallbackPlacements,c=i.padding,h=i.boundary,d=i.rootBoundary,u=i.altBoundary,f=i.flipVariations,p=void 0===f||f,m=i.allowedAutoPlacements,g=e.options.placement,_=be(g),b=l||(_!==g&&p?function(t){if(be(t)===Kt)return[];var e=Ve(t);return[Qe(t),e,Qe(e)]}(g):[Ve(g)]),v=[g].concat(b).reduce((function(t,i){return t.concat(be(i)===Kt?ni(e,{placement:i,boundary:h,rootBoundary:d,padding:c,flipVariations:p,allowedAutoPlacements:m}):i)}),[]),y=e.rects.reference,w=e.rects.popper,A=new Map,E=!0,T=v[0],C=0;C=0,S=L?"width":"height",D=ii(e,{placement:O,boundary:h,rootBoundary:d,altBoundary:u,padding:c}),$=L?k?qt:Vt:k?Rt:zt;y[S]>w[S]&&($=Ve($));var I=Ve($),N=[];if(o&&N.push(D[x]<=0),a&&N.push(D[$]<=0,D[I]<=0),N.every((function(t){return t}))){T=O,E=!1;break}A.set(O,N)}if(E)for(var P=function(t){var e=v.find((function(e){var i=A.get(e);if(i)return i.slice(0,t).every((function(t){return t}))}));if(e)return T=e,"break"},M=p?3:1;M>0&&"break"!==P(M);M--);e.placement!==T&&(e.modifiersData[n]._skip=!0,e.placement=T,e.reset=!0)}},requiresIfExists:["offset"],data:{_skip:!1}};function oi(t,e,i){return void 0===i&&(i={x:0,y:0}),{top:t.top-e.height-i.y,right:t.right-e.width+i.x,bottom:t.bottom-e.height+i.y,left:t.left-e.width-i.x}}function ri(t){return[zt,qt,Rt,Vt].some((function(e){return t[e]>=0}))}const ai={name:"hide",enabled:!0,phase:"main",requiresIfExists:["preventOverflow"],fn:function(t){var e=t.state,i=t.name,n=e.rects.reference,s=e.rects.popper,o=e.modifiersData.preventOverflow,r=ii(e,{elementContext:"reference"}),a=ii(e,{altBoundary:!0}),l=oi(r,n),c=oi(a,s,o),h=ri(l),d=ri(c);e.modifiersData[i]={referenceClippingOffsets:l,popperEscapeOffsets:c,isReferenceHidden:h,hasPopperEscaped:d},e.attributes.popper=Object.assign({},e.attributes.popper,{"data-popper-reference-hidden":h,"data-popper-escaped":d})}},li={name:"offset",enabled:!0,phase:"main",requires:["popperOffsets"],fn:function(t){var e=t.state,i=t.options,n=t.name,s=i.offset,o=void 0===s?[0,0]:s,r=ee.reduce((function(t,i){return t[i]=function(t,e,i){var n=be(t),s=[Vt,zt].indexOf(n)>=0?-1:1,o="function"==typeof i?i(Object.assign({},e,{placement:t})):i,r=o[0],a=o[1];return r=r||0,a=(a||0)*s,[Vt,qt].indexOf(n)>=0?{x:a,y:r}:{x:r,y:a}}(i,e.rects,o),t}),{}),a=r[e.placement],l=a.x,c=a.y;null!=e.modifiersData.popperOffsets&&(e.modifiersData.popperOffsets.x+=l,e.modifiersData.popperOffsets.y+=c),e.modifiersData[n]=r}},ci={name:"popperOffsets",enabled:!0,phase:"read",fn:function(t){var e=t.state,i=t.name;e.modifiersData[i]=ei({reference:e.rects.reference,element:e.rects.popper,strategy:"absolute",placement:e.placement})},data:{}},hi={name:"preventOverflow",enabled:!0,phase:"main",fn:function(t){var e=t.state,i=t.options,n=t.name,s=i.mainAxis,o=void 0===s||s,r=i.altAxis,a=void 0!==r&&r,l=i.boundary,c=i.rootBoundary,h=i.altBoundary,d=i.padding,u=i.tether,f=void 0===u||u,p=i.tetherOffset,m=void 0===p?0:p,g=ii(e,{boundary:l,rootBoundary:c,padding:d,altBoundary:h}),_=be(e.placement),b=Fe(e.placement),v=!b,y=Ie(_),w="x"===y?"y":"x",A=e.modifiersData.popperOffsets,E=e.rects.reference,T=e.rects.popper,C="function"==typeof m?m(Object.assign({},e.rects,{placement:e.placement})):m,O="number"==typeof C?{mainAxis:C,altAxis:C}:Object.assign({mainAxis:0,altAxis:0},C),x=e.modifiersData.offset?e.modifiersData.offset[e.placement]:null,k={x:0,y:0};if(A){if(o){var L,S="y"===y?zt:Vt,D="y"===y?Rt:qt,$="y"===y?"height":"width",I=A[y],N=I+g[S],P=I-g[D],M=f?-T[$]/2:0,j=b===Xt?E[$]:T[$],F=b===Xt?-T[$]:-E[$],H=e.elements.arrow,W=f&&H?Ce(H):{width:0,height:0},B=e.modifiersData["arrow#persistent"]?e.modifiersData["arrow#persistent"].padding:{top:0,right:0,bottom:0,left:0},z=B[S],R=B[D],q=Ne(0,E[$],W[$]),V=v?E[$]/2-M-q-z-O.mainAxis:j-q-z-O.mainAxis,K=v?-E[$]/2+M+q+R+O.mainAxis:F+q+R+O.mainAxis,Q=e.elements.arrow&&$e(e.elements.arrow),X=Q?"y"===y?Q.clientTop||0:Q.clientLeft||0:0,Y=null!=(L=null==x?void 0:x[y])?L:0,U=I+K-Y,G=Ne(f?ye(N,I+V-Y-X):N,I,f?ve(P,U):P);A[y]=G,k[y]=G-I}if(a){var J,Z="x"===y?zt:Vt,tt="x"===y?Rt:qt,et=A[w],it="y"===w?"height":"width",nt=et+g[Z],st=et-g[tt],ot=-1!==[zt,Vt].indexOf(_),rt=null!=(J=null==x?void 0:x[w])?J:0,at=ot?nt:et-E[it]-T[it]-rt+O.altAxis,lt=ot?et+E[it]+T[it]-rt-O.altAxis:st,ct=f&&ot?function(t,e,i){var n=Ne(t,e,i);return n>i?i:n}(at,et,lt):Ne(f?at:nt,et,f?lt:st);A[w]=ct,k[w]=ct-et}e.modifiersData[n]=k}},requiresIfExists:["offset"]};function di(t,e,i){void 0===i&&(i=!1);var n,s,o=me(e),r=me(e)&&function(t){var e=t.getBoundingClientRect(),i=we(e.width)/t.offsetWidth||1,n=we(e.height)/t.offsetHeight||1;return 1!==i||1!==n}(e),a=Le(e),l=Te(t,r,i),c={scrollLeft:0,scrollTop:0},h={x:0,y:0};return(o||!o&&!i)&&(("body"!==ue(e)||Ue(a))&&(c=(n=e)!==fe(n)&&me(n)?{scrollLeft:(s=n).scrollLeft,scrollTop:s.scrollTop}:Xe(n)),me(e)?((h=Te(e,!0)).x+=e.clientLeft,h.y+=e.clientTop):a&&(h.x=Ye(a))),{x:l.left+c.scrollLeft-h.x,y:l.top+c.scrollTop-h.y,width:l.width,height:l.height}}function ui(t){var e=new Map,i=new Set,n=[];function s(t){i.add(t.name),[].concat(t.requires||[],t.requiresIfExists||[]).forEach((function(t){if(!i.has(t)){var n=e.get(t);n&&s(n)}})),n.push(t)}return t.forEach((function(t){e.set(t.name,t)})),t.forEach((function(t){i.has(t.name)||s(t)})),n}var fi={placement:"bottom",modifiers:[],strategy:"absolute"};function pi(){for(var t=arguments.length,e=new Array(t),i=0;iNumber.parseInt(t,10))):"function"==typeof t?e=>t(e,this._element):t}_getPopperConfig(){const t={placement:this._getPlacement(),modifiers:[{name:"preventOverflow",options:{boundary:this._config.boundary}},{name:"offset",options:{offset:this._getOffset()}}]};return(this._inNavbar||"static"===this._config.display)&&(F.setDataAttribute(this._menu,"popper","static"),t.modifiers=[{name:"applyStyles",enabled:!1}]),{...t,...g(this._config.popperConfig,[t])}}_selectMenuItem({key:t,target:e}){const i=z.find(".dropdown-menu .dropdown-item:not(.disabled):not(:disabled)",this._menu).filter((t=>a(t)));i.length&&b(i,e,t===Ti,!i.includes(e)).focus()}static jQueryInterface(t){return this.each((function(){const e=qi.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}static clearMenus(t){if(2===t.button||"keyup"===t.type&&"Tab"!==t.key)return;const e=z.find(Ni);for(const i of e){const e=qi.getInstance(i);if(!e||!1===e._config.autoClose)continue;const n=t.composedPath(),s=n.includes(e._menu);if(n.includes(e._element)||"inside"===e._config.autoClose&&!s||"outside"===e._config.autoClose&&s)continue;if(e._menu.contains(t.target)&&("keyup"===t.type&&"Tab"===t.key||/input|select|option|textarea|form/i.test(t.target.tagName)))continue;const o={relatedTarget:e._element};"click"===t.type&&(o.clickEvent=t),e._completeHide(o)}}static dataApiKeydownHandler(t){const e=/input|textarea/i.test(t.target.tagName),i="Escape"===t.key,n=[Ei,Ti].includes(t.key);if(!n&&!i)return;if(e&&!i)return;t.preventDefault();const s=this.matches(Ii)?this:z.prev(this,Ii)[0]||z.next(this,Ii)[0]||z.findOne(Ii,t.delegateTarget.parentNode),o=qi.getOrCreateInstance(s);if(n)return t.stopPropagation(),o.show(),void o._selectMenuItem(t);o._isShown()&&(t.stopPropagation(),o.hide(),s.focus())}}N.on(document,Si,Ii,qi.dataApiKeydownHandler),N.on(document,Si,Pi,qi.dataApiKeydownHandler),N.on(document,Li,qi.clearMenus),N.on(document,Di,qi.clearMenus),N.on(document,Li,Ii,(function(t){t.preventDefault(),qi.getOrCreateInstance(this).toggle()})),m(qi);const Vi="backdrop",Ki="show",Qi=`mousedown.bs.${Vi}`,Xi={className:"modal-backdrop",clickCallback:null,isAnimated:!1,isVisible:!0,rootElement:"body"},Yi={className:"string",clickCallback:"(function|null)",isAnimated:"boolean",isVisible:"boolean",rootElement:"(element|string)"};class Ui extends H{constructor(t){super(),this._config=this._getConfig(t),this._isAppended=!1,this._element=null}static get Default(){return Xi}static get DefaultType(){return Yi}static get NAME(){return Vi}show(t){if(!this._config.isVisible)return void g(t);this._append();const e=this._getElement();this._config.isAnimated&&d(e),e.classList.add(Ki),this._emulateAnimation((()=>{g(t)}))}hide(t){this._config.isVisible?(this._getElement().classList.remove(Ki),this._emulateAnimation((()=>{this.dispose(),g(t)}))):g(t)}dispose(){this._isAppended&&(N.off(this._element,Qi),this._element.remove(),this._isAppended=!1)}_getElement(){if(!this._element){const t=document.createElement("div");t.className=this._config.className,this._config.isAnimated&&t.classList.add("fade"),this._element=t}return this._element}_configAfterMerge(t){return t.rootElement=r(t.rootElement),t}_append(){if(this._isAppended)return;const t=this._getElement();this._config.rootElement.append(t),N.on(t,Qi,(()=>{g(this._config.clickCallback)})),this._isAppended=!0}_emulateAnimation(t){_(t,this._getElement(),this._config.isAnimated)}}const Gi=".bs.focustrap",Ji=`focusin${Gi}`,Zi=`keydown.tab${Gi}`,tn="backward",en={autofocus:!0,trapElement:null},nn={autofocus:"boolean",trapElement:"element"};class sn extends H{constructor(t){super(),this._config=this._getConfig(t),this._isActive=!1,this._lastTabNavDirection=null}static get Default(){return en}static get DefaultType(){return nn}static get NAME(){return"focustrap"}activate(){this._isActive||(this._config.autofocus&&this._config.trapElement.focus(),N.off(document,Gi),N.on(document,Ji,(t=>this._handleFocusin(t))),N.on(document,Zi,(t=>this._handleKeydown(t))),this._isActive=!0)}deactivate(){this._isActive&&(this._isActive=!1,N.off(document,Gi))}_handleFocusin(t){const{trapElement:e}=this._config;if(t.target===document||t.target===e||e.contains(t.target))return;const i=z.focusableChildren(e);0===i.length?e.focus():this._lastTabNavDirection===tn?i[i.length-1].focus():i[0].focus()}_handleKeydown(t){"Tab"===t.key&&(this._lastTabNavDirection=t.shiftKey?tn:"forward")}}const on=".fixed-top, .fixed-bottom, .is-fixed, .sticky-top",rn=".sticky-top",an="padding-right",ln="margin-right";class cn{constructor(){this._element=document.body}getWidth(){const t=document.documentElement.clientWidth;return Math.abs(window.innerWidth-t)}hide(){const t=this.getWidth();this._disableOverFlow(),this._setElementAttributes(this._element,an,(e=>e+t)),this._setElementAttributes(on,an,(e=>e+t)),this._setElementAttributes(rn,ln,(e=>e-t))}reset(){this._resetElementAttributes(this._element,"overflow"),this._resetElementAttributes(this._element,an),this._resetElementAttributes(on,an),this._resetElementAttributes(rn,ln)}isOverflowing(){return this.getWidth()>0}_disableOverFlow(){this._saveInitialAttribute(this._element,"overflow"),this._element.style.overflow="hidden"}_setElementAttributes(t,e,i){const n=this.getWidth();this._applyManipulationCallback(t,(t=>{if(t!==this._element&&window.innerWidth>t.clientWidth+n)return;this._saveInitialAttribute(t,e);const s=window.getComputedStyle(t).getPropertyValue(e);t.style.setProperty(e,`${i(Number.parseFloat(s))}px`)}))}_saveInitialAttribute(t,e){const i=t.style.getPropertyValue(e);i&&F.setDataAttribute(t,e,i)}_resetElementAttributes(t,e){this._applyManipulationCallback(t,(t=>{const i=F.getDataAttribute(t,e);null!==i?(F.removeDataAttribute(t,e),t.style.setProperty(e,i)):t.style.removeProperty(e)}))}_applyManipulationCallback(t,e){if(o(t))e(t);else for(const i of z.find(t,this._element))e(i)}}const hn=".bs.modal",dn=`hide${hn}`,un=`hidePrevented${hn}`,fn=`hidden${hn}`,pn=`show${hn}`,mn=`shown${hn}`,gn=`resize${hn}`,_n=`click.dismiss${hn}`,bn=`mousedown.dismiss${hn}`,vn=`keydown.dismiss${hn}`,yn=`click${hn}.data-api`,wn="modal-open",An="show",En="modal-static",Tn={backdrop:!0,focus:!0,keyboard:!0},Cn={backdrop:"(boolean|string)",focus:"boolean",keyboard:"boolean"};class On extends W{constructor(t,e){super(t,e),this._dialog=z.findOne(".modal-dialog",this._element),this._backdrop=this._initializeBackDrop(),this._focustrap=this._initializeFocusTrap(),this._isShown=!1,this._isTransitioning=!1,this._scrollBar=new cn,this._addEventListeners()}static get Default(){return Tn}static get DefaultType(){return Cn}static get NAME(){return"modal"}toggle(t){return this._isShown?this.hide():this.show(t)}show(t){this._isShown||this._isTransitioning||N.trigger(this._element,pn,{relatedTarget:t}).defaultPrevented||(this._isShown=!0,this._isTransitioning=!0,this._scrollBar.hide(),document.body.classList.add(wn),this._adjustDialog(),this._backdrop.show((()=>this._showElement(t))))}hide(){this._isShown&&!this._isTransitioning&&(N.trigger(this._element,dn).defaultPrevented||(this._isShown=!1,this._isTransitioning=!0,this._focustrap.deactivate(),this._element.classList.remove(An),this._queueCallback((()=>this._hideModal()),this._element,this._isAnimated())))}dispose(){N.off(window,hn),N.off(this._dialog,hn),this._backdrop.dispose(),this._focustrap.deactivate(),super.dispose()}handleUpdate(){this._adjustDialog()}_initializeBackDrop(){return new Ui({isVisible:Boolean(this._config.backdrop),isAnimated:this._isAnimated()})}_initializeFocusTrap(){return new sn({trapElement:this._element})}_showElement(t){document.body.contains(this._element)||document.body.append(this._element),this._element.style.display="block",this._element.removeAttribute("aria-hidden"),this._element.setAttribute("aria-modal",!0),this._element.setAttribute("role","dialog"),this._element.scrollTop=0;const e=z.findOne(".modal-body",this._dialog);e&&(e.scrollTop=0),d(this._element),this._element.classList.add(An),this._queueCallback((()=>{this._config.focus&&this._focustrap.activate(),this._isTransitioning=!1,N.trigger(this._element,mn,{relatedTarget:t})}),this._dialog,this._isAnimated())}_addEventListeners(){N.on(this._element,vn,(t=>{"Escape"===t.key&&(this._config.keyboard?this.hide():this._triggerBackdropTransition())})),N.on(window,gn,(()=>{this._isShown&&!this._isTransitioning&&this._adjustDialog()})),N.on(this._element,bn,(t=>{N.one(this._element,_n,(e=>{this._element===t.target&&this._element===e.target&&("static"!==this._config.backdrop?this._config.backdrop&&this.hide():this._triggerBackdropTransition())}))}))}_hideModal(){this._element.style.display="none",this._element.setAttribute("aria-hidden",!0),this._element.removeAttribute("aria-modal"),this._element.removeAttribute("role"),this._isTransitioning=!1,this._backdrop.hide((()=>{document.body.classList.remove(wn),this._resetAdjustments(),this._scrollBar.reset(),N.trigger(this._element,fn)}))}_isAnimated(){return this._element.classList.contains("fade")}_triggerBackdropTransition(){if(N.trigger(this._element,un).defaultPrevented)return;const t=this._element.scrollHeight>document.documentElement.clientHeight,e=this._element.style.overflowY;"hidden"===e||this._element.classList.contains(En)||(t||(this._element.style.overflowY="hidden"),this._element.classList.add(En),this._queueCallback((()=>{this._element.classList.remove(En),this._queueCallback((()=>{this._element.style.overflowY=e}),this._dialog)}),this._dialog),this._element.focus())}_adjustDialog(){const t=this._element.scrollHeight>document.documentElement.clientHeight,e=this._scrollBar.getWidth(),i=e>0;if(i&&!t){const t=p()?"paddingLeft":"paddingRight";this._element.style[t]=`${e}px`}if(!i&&t){const t=p()?"paddingRight":"paddingLeft";this._element.style[t]=`${e}px`}}_resetAdjustments(){this._element.style.paddingLeft="",this._element.style.paddingRight=""}static jQueryInterface(t,e){return this.each((function(){const i=On.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===i[t])throw new TypeError(`No method named "${t}"`);i[t](e)}}))}}N.on(document,yn,'[data-bs-toggle="modal"]',(function(t){const e=z.getElementFromSelector(this);["A","AREA"].includes(this.tagName)&&t.preventDefault(),N.one(e,pn,(t=>{t.defaultPrevented||N.one(e,fn,(()=>{a(this)&&this.focus()}))}));const i=z.findOne(".modal.show");i&&On.getInstance(i).hide(),On.getOrCreateInstance(e).toggle(this)})),R(On),m(On);const xn=".bs.offcanvas",kn=".data-api",Ln=`load${xn}${kn}`,Sn="show",Dn="showing",$n="hiding",In=".offcanvas.show",Nn=`show${xn}`,Pn=`shown${xn}`,Mn=`hide${xn}`,jn=`hidePrevented${xn}`,Fn=`hidden${xn}`,Hn=`resize${xn}`,Wn=`click${xn}${kn}`,Bn=`keydown.dismiss${xn}`,zn={backdrop:!0,keyboard:!0,scroll:!1},Rn={backdrop:"(boolean|string)",keyboard:"boolean",scroll:"boolean"};class qn extends W{constructor(t,e){super(t,e),this._isShown=!1,this._backdrop=this._initializeBackDrop(),this._focustrap=this._initializeFocusTrap(),this._addEventListeners()}static get Default(){return zn}static get DefaultType(){return Rn}static get NAME(){return"offcanvas"}toggle(t){return this._isShown?this.hide():this.show(t)}show(t){this._isShown||N.trigger(this._element,Nn,{relatedTarget:t}).defaultPrevented||(this._isShown=!0,this._backdrop.show(),this._config.scroll||(new cn).hide(),this._element.setAttribute("aria-modal",!0),this._element.setAttribute("role","dialog"),this._element.classList.add(Dn),this._queueCallback((()=>{this._config.scroll&&!this._config.backdrop||this._focustrap.activate(),this._element.classList.add(Sn),this._element.classList.remove(Dn),N.trigger(this._element,Pn,{relatedTarget:t})}),this._element,!0))}hide(){this._isShown&&(N.trigger(this._element,Mn).defaultPrevented||(this._focustrap.deactivate(),this._element.blur(),this._isShown=!1,this._element.classList.add($n),this._backdrop.hide(),this._queueCallback((()=>{this._element.classList.remove(Sn,$n),this._element.removeAttribute("aria-modal"),this._element.removeAttribute("role"),this._config.scroll||(new cn).reset(),N.trigger(this._element,Fn)}),this._element,!0)))}dispose(){this._backdrop.dispose(),this._focustrap.deactivate(),super.dispose()}_initializeBackDrop(){const t=Boolean(this._config.backdrop);return new Ui({className:"offcanvas-backdrop",isVisible:t,isAnimated:!0,rootElement:this._element.parentNode,clickCallback:t?()=>{"static"!==this._config.backdrop?this.hide():N.trigger(this._element,jn)}:null})}_initializeFocusTrap(){return new sn({trapElement:this._element})}_addEventListeners(){N.on(this._element,Bn,(t=>{"Escape"===t.key&&(this._config.keyboard?this.hide():N.trigger(this._element,jn))}))}static jQueryInterface(t){return this.each((function(){const e=qn.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}N.on(document,Wn,'[data-bs-toggle="offcanvas"]',(function(t){const e=z.getElementFromSelector(this);if(["A","AREA"].includes(this.tagName)&&t.preventDefault(),l(this))return;N.one(e,Fn,(()=>{a(this)&&this.focus()}));const i=z.findOne(In);i&&i!==e&&qn.getInstance(i).hide(),qn.getOrCreateInstance(e).toggle(this)})),N.on(window,Ln,(()=>{for(const t of z.find(In))qn.getOrCreateInstance(t).show()})),N.on(window,Hn,(()=>{for(const t of z.find("[aria-modal][class*=show][class*=offcanvas-]"))"fixed"!==getComputedStyle(t).position&&qn.getOrCreateInstance(t).hide()})),R(qn),m(qn);const Vn={"*":["class","dir","id","lang","role",/^aria-[\w-]*$/i],a:["target","href","title","rel"],area:[],b:[],br:[],col:[],code:[],div:[],em:[],hr:[],h1:[],h2:[],h3:[],h4:[],h5:[],h6:[],i:[],img:["src","srcset","alt","title","width","height"],li:[],ol:[],p:[],pre:[],s:[],small:[],span:[],sub:[],sup:[],strong:[],u:[],ul:[]},Kn=new Set(["background","cite","href","itemtype","longdesc","poster","src","xlink:href"]),Qn=/^(?!javascript:)(?:[a-z0-9+.-]+:|[^&:/?#]*(?:[/?#]|$))/i,Xn=(t,e)=>{const i=t.nodeName.toLowerCase();return e.includes(i)?!Kn.has(i)||Boolean(Qn.test(t.nodeValue)):e.filter((t=>t instanceof RegExp)).some((t=>t.test(i)))},Yn={allowList:Vn,content:{},extraClass:"",html:!1,sanitize:!0,sanitizeFn:null,template:"

"},Un={allowList:"object",content:"object",extraClass:"(string|function)",html:"boolean",sanitize:"boolean",sanitizeFn:"(null|function)",template:"string"},Gn={entry:"(string|element|function|null)",selector:"(string|element)"};class Jn extends H{constructor(t){super(),this._config=this._getConfig(t)}static get Default(){return Yn}static get DefaultType(){return Un}static get NAME(){return"TemplateFactory"}getContent(){return Object.values(this._config.content).map((t=>this._resolvePossibleFunction(t))).filter(Boolean)}hasContent(){return this.getContent().length>0}changeContent(t){return this._checkContent(t),this._config.content={...this._config.content,...t},this}toHtml(){const t=document.createElement("div");t.innerHTML=this._maybeSanitize(this._config.template);for(const[e,i]of Object.entries(this._config.content))this._setContent(t,i,e);const e=t.children[0],i=this._resolvePossibleFunction(this._config.extraClass);return i&&e.classList.add(...i.split(" ")),e}_typeCheckConfig(t){super._typeCheckConfig(t),this._checkContent(t.content)}_checkContent(t){for(const[e,i]of Object.entries(t))super._typeCheckConfig({selector:e,entry:i},Gn)}_setContent(t,e,i){const n=z.findOne(i,t);n&&((e=this._resolvePossibleFunction(e))?o(e)?this._putElementInTemplate(r(e),n):this._config.html?n.innerHTML=this._maybeSanitize(e):n.textContent=e:n.remove())}_maybeSanitize(t){return this._config.sanitize?function(t,e,i){if(!t.length)return t;if(i&&"function"==typeof i)return i(t);const n=(new window.DOMParser).parseFromString(t,"text/html"),s=[].concat(...n.body.querySelectorAll("*"));for(const t of s){const i=t.nodeName.toLowerCase();if(!Object.keys(e).includes(i)){t.remove();continue}const n=[].concat(...t.attributes),s=[].concat(e["*"]||[],e[i]||[]);for(const e of n)Xn(e,s)||t.removeAttribute(e.nodeName)}return n.body.innerHTML}(t,this._config.allowList,this._config.sanitizeFn):t}_resolvePossibleFunction(t){return g(t,[this])}_putElementInTemplate(t,e){if(this._config.html)return e.innerHTML="",void e.append(t);e.textContent=t.textContent}}const Zn=new Set(["sanitize","allowList","sanitizeFn"]),ts="fade",es="show",is=".modal",ns="hide.bs.modal",ss="hover",os="focus",rs={AUTO:"auto",TOP:"top",RIGHT:p()?"left":"right",BOTTOM:"bottom",LEFT:p()?"right":"left"},as={allowList:Vn,animation:!0,boundary:"clippingParents",container:!1,customClass:"",delay:0,fallbackPlacements:["top","right","bottom","left"],html:!1,offset:[0,6],placement:"top",popperConfig:null,sanitize:!0,sanitizeFn:null,selector:!1,template:'',title:"",trigger:"hover focus"},ls={allowList:"object",animation:"boolean",boundary:"(string|element)",container:"(string|element|boolean)",customClass:"(string|function)",delay:"(number|object)",fallbackPlacements:"array",html:"boolean",offset:"(array|string|function)",placement:"(string|function)",popperConfig:"(null|object|function)",sanitize:"boolean",sanitizeFn:"(null|function)",selector:"(string|boolean)",template:"string",title:"(string|element|function)",trigger:"string"};class cs extends W{constructor(t,e){if(void 0===vi)throw new TypeError("Bootstrap's tooltips require Popper (https://popper.js.org)");super(t,e),this._isEnabled=!0,this._timeout=0,this._isHovered=null,this._activeTrigger={},this._popper=null,this._templateFactory=null,this._newContent=null,this.tip=null,this._setListeners(),this._config.selector||this._fixTitle()}static get Default(){return as}static get DefaultType(){return ls}static get NAME(){return"tooltip"}enable(){this._isEnabled=!0}disable(){this._isEnabled=!1}toggleEnabled(){this._isEnabled=!this._isEnabled}toggle(){this._isEnabled&&(this._activeTrigger.click=!this._activeTrigger.click,this._isShown()?this._leave():this._enter())}dispose(){clearTimeout(this._timeout),N.off(this._element.closest(is),ns,this._hideModalHandler),this._element.getAttribute("data-bs-original-title")&&this._element.setAttribute("title",this._element.getAttribute("data-bs-original-title")),this._disposePopper(),super.dispose()}show(){if("none"===this._element.style.display)throw new Error("Please use show on visible elements");if(!this._isWithContent()||!this._isEnabled)return;const t=N.trigger(this._element,this.constructor.eventName("show")),e=(c(this._element)||this._element.ownerDocument.documentElement).contains(this._element);if(t.defaultPrevented||!e)return;this._disposePopper();const i=this._getTipElement();this._element.setAttribute("aria-describedby",i.getAttribute("id"));const{container:n}=this._config;if(this._element.ownerDocument.documentElement.contains(this.tip)||(n.append(i),N.trigger(this._element,this.constructor.eventName("inserted"))),this._popper=this._createPopper(i),i.classList.add(es),"ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))N.on(t,"mouseover",h);this._queueCallback((()=>{N.trigger(this._element,this.constructor.eventName("shown")),!1===this._isHovered&&this._leave(),this._isHovered=!1}),this.tip,this._isAnimated())}hide(){if(this._isShown()&&!N.trigger(this._element,this.constructor.eventName("hide")).defaultPrevented){if(this._getTipElement().classList.remove(es),"ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))N.off(t,"mouseover",h);this._activeTrigger.click=!1,this._activeTrigger[os]=!1,this._activeTrigger[ss]=!1,this._isHovered=null,this._queueCallback((()=>{this._isWithActiveTrigger()||(this._isHovered||this._disposePopper(),this._element.removeAttribute("aria-describedby"),N.trigger(this._element,this.constructor.eventName("hidden")))}),this.tip,this._isAnimated())}}update(){this._popper&&this._popper.update()}_isWithContent(){return Boolean(this._getTitle())}_getTipElement(){return this.tip||(this.tip=this._createTipElement(this._newContent||this._getContentForTemplate())),this.tip}_createTipElement(t){const e=this._getTemplateFactory(t).toHtml();if(!e)return null;e.classList.remove(ts,es),e.classList.add(`bs-${this.constructor.NAME}-auto`);const i=(t=>{do{t+=Math.floor(1e6*Math.random())}while(document.getElementById(t));return t})(this.constructor.NAME).toString();return e.setAttribute("id",i),this._isAnimated()&&e.classList.add(ts),e}setContent(t){this._newContent=t,this._isShown()&&(this._disposePopper(),this.show())}_getTemplateFactory(t){return this._templateFactory?this._templateFactory.changeContent(t):this._templateFactory=new Jn({...this._config,content:t,extraClass:this._resolvePossibleFunction(this._config.customClass)}),this._templateFactory}_getContentForTemplate(){return{".tooltip-inner":this._getTitle()}}_getTitle(){return this._resolvePossibleFunction(this._config.title)||this._element.getAttribute("data-bs-original-title")}_initializeOnDelegatedTarget(t){return this.constructor.getOrCreateInstance(t.delegateTarget,this._getDelegateConfig())}_isAnimated(){return this._config.animation||this.tip&&this.tip.classList.contains(ts)}_isShown(){return this.tip&&this.tip.classList.contains(es)}_createPopper(t){const e=g(this._config.placement,[this,t,this._element]),i=rs[e.toUpperCase()];return bi(this._element,t,this._getPopperConfig(i))}_getOffset(){const{offset:t}=this._config;return"string"==typeof t?t.split(",").map((t=>Number.parseInt(t,10))):"function"==typeof t?e=>t(e,this._element):t}_resolvePossibleFunction(t){return g(t,[this._element])}_getPopperConfig(t){const e={placement:t,modifiers:[{name:"flip",options:{fallbackPlacements:this._config.fallbackPlacements}},{name:"offset",options:{offset:this._getOffset()}},{name:"preventOverflow",options:{boundary:this._config.boundary}},{name:"arrow",options:{element:`.${this.constructor.NAME}-arrow`}},{name:"preSetPlacement",enabled:!0,phase:"beforeMain",fn:t=>{this._getTipElement().setAttribute("data-popper-placement",t.state.placement)}}]};return{...e,...g(this._config.popperConfig,[e])}}_setListeners(){const t=this._config.trigger.split(" ");for(const e of t)if("click"===e)N.on(this._element,this.constructor.eventName("click"),this._config.selector,(t=>{this._initializeOnDelegatedTarget(t).toggle()}));else if("manual"!==e){const t=e===ss?this.constructor.eventName("mouseenter"):this.constructor.eventName("focusin"),i=e===ss?this.constructor.eventName("mouseleave"):this.constructor.eventName("focusout");N.on(this._element,t,this._config.selector,(t=>{const e=this._initializeOnDelegatedTarget(t);e._activeTrigger["focusin"===t.type?os:ss]=!0,e._enter()})),N.on(this._element,i,this._config.selector,(t=>{const e=this._initializeOnDelegatedTarget(t);e._activeTrigger["focusout"===t.type?os:ss]=e._element.contains(t.relatedTarget),e._leave()}))}this._hideModalHandler=()=>{this._element&&this.hide()},N.on(this._element.closest(is),ns,this._hideModalHandler)}_fixTitle(){const t=this._element.getAttribute("title");t&&(this._element.getAttribute("aria-label")||this._element.textContent.trim()||this._element.setAttribute("aria-label",t),this._element.setAttribute("data-bs-original-title",t),this._element.removeAttribute("title"))}_enter(){this._isShown()||this._isHovered?this._isHovered=!0:(this._isHovered=!0,this._setTimeout((()=>{this._isHovered&&this.show()}),this._config.delay.show))}_leave(){this._isWithActiveTrigger()||(this._isHovered=!1,this._setTimeout((()=>{this._isHovered||this.hide()}),this._config.delay.hide))}_setTimeout(t,e){clearTimeout(this._timeout),this._timeout=setTimeout(t,e)}_isWithActiveTrigger(){return Object.values(this._activeTrigger).includes(!0)}_getConfig(t){const e=F.getDataAttributes(this._element);for(const t of Object.keys(e))Zn.has(t)&&delete e[t];return t={...e,..."object"==typeof t&&t?t:{}},t=this._mergeConfigObj(t),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}_configAfterMerge(t){return t.container=!1===t.container?document.body:r(t.container),"number"==typeof t.delay&&(t.delay={show:t.delay,hide:t.delay}),"number"==typeof t.title&&(t.title=t.title.toString()),"number"==typeof t.content&&(t.content=t.content.toString()),t}_getDelegateConfig(){const t={};for(const[e,i]of Object.entries(this._config))this.constructor.Default[e]!==i&&(t[e]=i);return t.selector=!1,t.trigger="manual",t}_disposePopper(){this._popper&&(this._popper.destroy(),this._popper=null),this.tip&&(this.tip.remove(),this.tip=null)}static jQueryInterface(t){return this.each((function(){const e=cs.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}}m(cs);const hs={...cs.Default,content:"",offset:[0,8],placement:"right",template:'',trigger:"click"},ds={...cs.DefaultType,content:"(null|string|element|function)"};class us extends cs{static get Default(){return hs}static get DefaultType(){return ds}static get NAME(){return"popover"}_isWithContent(){return this._getTitle()||this._getContent()}_getContentForTemplate(){return{".popover-header":this._getTitle(),".popover-body":this._getContent()}}_getContent(){return this._resolvePossibleFunction(this._config.content)}static jQueryInterface(t){return this.each((function(){const e=us.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}}m(us);const fs=".bs.scrollspy",ps=`activate${fs}`,ms=`click${fs}`,gs=`load${fs}.data-api`,_s="active",bs="[href]",vs=".nav-link",ys=`${vs}, .nav-item > ${vs}, .list-group-item`,ws={offset:null,rootMargin:"0px 0px -25%",smoothScroll:!1,target:null,threshold:[.1,.5,1]},As={offset:"(number|null)",rootMargin:"string",smoothScroll:"boolean",target:"element",threshold:"array"};class Es extends W{constructor(t,e){super(t,e),this._targetLinks=new Map,this._observableSections=new Map,this._rootElement="visible"===getComputedStyle(this._element).overflowY?null:this._element,this._activeTarget=null,this._observer=null,this._previousScrollData={visibleEntryTop:0,parentScrollTop:0},this.refresh()}static get Default(){return ws}static get DefaultType(){return As}static get NAME(){return"scrollspy"}refresh(){this._initializeTargetsAndObservables(),this._maybeEnableSmoothScroll(),this._observer?this._observer.disconnect():this._observer=this._getNewObserver();for(const t of this._observableSections.values())this._observer.observe(t)}dispose(){this._observer.disconnect(),super.dispose()}_configAfterMerge(t){return t.target=r(t.target)||document.body,t.rootMargin=t.offset?`${t.offset}px 0px -30%`:t.rootMargin,"string"==typeof t.threshold&&(t.threshold=t.threshold.split(",").map((t=>Number.parseFloat(t)))),t}_maybeEnableSmoothScroll(){this._config.smoothScroll&&(N.off(this._config.target,ms),N.on(this._config.target,ms,bs,(t=>{const e=this._observableSections.get(t.target.hash);if(e){t.preventDefault();const i=this._rootElement||window,n=e.offsetTop-this._element.offsetTop;if(i.scrollTo)return void i.scrollTo({top:n,behavior:"smooth"});i.scrollTop=n}})))}_getNewObserver(){const t={root:this._rootElement,threshold:this._config.threshold,rootMargin:this._config.rootMargin};return new IntersectionObserver((t=>this._observerCallback(t)),t)}_observerCallback(t){const e=t=>this._targetLinks.get(`#${t.target.id}`),i=t=>{this._previousScrollData.visibleEntryTop=t.target.offsetTop,this._process(e(t))},n=(this._rootElement||document.documentElement).scrollTop,s=n>=this._previousScrollData.parentScrollTop;this._previousScrollData.parentScrollTop=n;for(const o of t){if(!o.isIntersecting){this._activeTarget=null,this._clearActiveClass(e(o));continue}const t=o.target.offsetTop>=this._previousScrollData.visibleEntryTop;if(s&&t){if(i(o),!n)return}else s||t||i(o)}}_initializeTargetsAndObservables(){this._targetLinks=new Map,this._observableSections=new Map;const t=z.find(bs,this._config.target);for(const e of t){if(!e.hash||l(e))continue;const t=z.findOne(decodeURI(e.hash),this._element);a(t)&&(this._targetLinks.set(decodeURI(e.hash),e),this._observableSections.set(e.hash,t))}}_process(t){this._activeTarget!==t&&(this._clearActiveClass(this._config.target),this._activeTarget=t,t.classList.add(_s),this._activateParents(t),N.trigger(this._element,ps,{relatedTarget:t}))}_activateParents(t){if(t.classList.contains("dropdown-item"))z.findOne(".dropdown-toggle",t.closest(".dropdown")).classList.add(_s);else for(const e of z.parents(t,".nav, .list-group"))for(const t of z.prev(e,ys))t.classList.add(_s)}_clearActiveClass(t){t.classList.remove(_s);const e=z.find(`${bs}.${_s}`,t);for(const t of e)t.classList.remove(_s)}static jQueryInterface(t){return this.each((function(){const e=Es.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}))}}N.on(window,gs,(()=>{for(const t of z.find('[data-bs-spy="scroll"]'))Es.getOrCreateInstance(t)})),m(Es);const Ts=".bs.tab",Cs=`hide${Ts}`,Os=`hidden${Ts}`,xs=`show${Ts}`,ks=`shown${Ts}`,Ls=`click${Ts}`,Ss=`keydown${Ts}`,Ds=`load${Ts}`,$s="ArrowLeft",Is="ArrowRight",Ns="ArrowUp",Ps="ArrowDown",Ms="Home",js="End",Fs="active",Hs="fade",Ws="show",Bs=":not(.dropdown-toggle)",zs='[data-bs-toggle="tab"], [data-bs-toggle="pill"], [data-bs-toggle="list"]',Rs=`.nav-link${Bs}, .list-group-item${Bs}, [role="tab"]${Bs}, ${zs}`,qs=`.${Fs}[data-bs-toggle="tab"], .${Fs}[data-bs-toggle="pill"], .${Fs}[data-bs-toggle="list"]`;class Vs extends W{constructor(t){super(t),this._parent=this._element.closest('.list-group, .nav, [role="tablist"]'),this._parent&&(this._setInitialAttributes(this._parent,this._getChildren()),N.on(this._element,Ss,(t=>this._keydown(t))))}static get NAME(){return"tab"}show(){const t=this._element;if(this._elemIsActive(t))return;const e=this._getActiveElem(),i=e?N.trigger(e,Cs,{relatedTarget:t}):null;N.trigger(t,xs,{relatedTarget:e}).defaultPrevented||i&&i.defaultPrevented||(this._deactivate(e,t),this._activate(t,e))}_activate(t,e){t&&(t.classList.add(Fs),this._activate(z.getElementFromSelector(t)),this._queueCallback((()=>{"tab"===t.getAttribute("role")?(t.removeAttribute("tabindex"),t.setAttribute("aria-selected",!0),this._toggleDropDown(t,!0),N.trigger(t,ks,{relatedTarget:e})):t.classList.add(Ws)}),t,t.classList.contains(Hs)))}_deactivate(t,e){t&&(t.classList.remove(Fs),t.blur(),this._deactivate(z.getElementFromSelector(t)),this._queueCallback((()=>{"tab"===t.getAttribute("role")?(t.setAttribute("aria-selected",!1),t.setAttribute("tabindex","-1"),this._toggleDropDown(t,!1),N.trigger(t,Os,{relatedTarget:e})):t.classList.remove(Ws)}),t,t.classList.contains(Hs)))}_keydown(t){if(![$s,Is,Ns,Ps,Ms,js].includes(t.key))return;t.stopPropagation(),t.preventDefault();const e=this._getChildren().filter((t=>!l(t)));let i;if([Ms,js].includes(t.key))i=e[t.key===Ms?0:e.length-1];else{const n=[Is,Ps].includes(t.key);i=b(e,t.target,n,!0)}i&&(i.focus({preventScroll:!0}),Vs.getOrCreateInstance(i).show())}_getChildren(){return z.find(Rs,this._parent)}_getActiveElem(){return this._getChildren().find((t=>this._elemIsActive(t)))||null}_setInitialAttributes(t,e){this._setAttributeIfNotExists(t,"role","tablist");for(const t of e)this._setInitialAttributesOnChild(t)}_setInitialAttributesOnChild(t){t=this._getInnerElement(t);const e=this._elemIsActive(t),i=this._getOuterElement(t);t.setAttribute("aria-selected",e),i!==t&&this._setAttributeIfNotExists(i,"role","presentation"),e||t.setAttribute("tabindex","-1"),this._setAttributeIfNotExists(t,"role","tab"),this._setInitialAttributesOnTargetPanel(t)}_setInitialAttributesOnTargetPanel(t){const e=z.getElementFromSelector(t);e&&(this._setAttributeIfNotExists(e,"role","tabpanel"),t.id&&this._setAttributeIfNotExists(e,"aria-labelledby",`${t.id}`))}_toggleDropDown(t,e){const i=this._getOuterElement(t);if(!i.classList.contains("dropdown"))return;const n=(t,n)=>{const s=z.findOne(t,i);s&&s.classList.toggle(n,e)};n(".dropdown-toggle",Fs),n(".dropdown-menu",Ws),i.setAttribute("aria-expanded",e)}_setAttributeIfNotExists(t,e,i){t.hasAttribute(e)||t.setAttribute(e,i)}_elemIsActive(t){return t.classList.contains(Fs)}_getInnerElement(t){return t.matches(Rs)?t:z.findOne(Rs,t)}_getOuterElement(t){return t.closest(".nav-item, .list-group-item")||t}static jQueryInterface(t){return this.each((function(){const e=Vs.getOrCreateInstance(this);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}))}}N.on(document,Ls,zs,(function(t){["A","AREA"].includes(this.tagName)&&t.preventDefault(),l(this)||Vs.getOrCreateInstance(this).show()})),N.on(window,Ds,(()=>{for(const t of z.find(qs))Vs.getOrCreateInstance(t)})),m(Vs);const Ks=".bs.toast",Qs=`mouseover${Ks}`,Xs=`mouseout${Ks}`,Ys=`focusin${Ks}`,Us=`focusout${Ks}`,Gs=`hide${Ks}`,Js=`hidden${Ks}`,Zs=`show${Ks}`,to=`shown${Ks}`,eo="hide",io="show",no="showing",so={animation:"boolean",autohide:"boolean",delay:"number"},oo={animation:!0,autohide:!0,delay:5e3};class ro extends W{constructor(t,e){super(t,e),this._timeout=null,this._hasMouseInteraction=!1,this._hasKeyboardInteraction=!1,this._setListeners()}static get Default(){return oo}static get DefaultType(){return so}static get NAME(){return"toast"}show(){N.trigger(this._element,Zs).defaultPrevented||(this._clearTimeout(),this._config.animation&&this._element.classList.add("fade"),this._element.classList.remove(eo),d(this._element),this._element.classList.add(io,no),this._queueCallback((()=>{this._element.classList.remove(no),N.trigger(this._element,to),this._maybeScheduleHide()}),this._element,this._config.animation))}hide(){this.isShown()&&(N.trigger(this._element,Gs).defaultPrevented||(this._element.classList.add(no),this._queueCallback((()=>{this._element.classList.add(eo),this._element.classList.remove(no,io),N.trigger(this._element,Js)}),this._element,this._config.animation)))}dispose(){this._clearTimeout(),this.isShown()&&this._element.classList.remove(io),super.dispose()}isShown(){return this._element.classList.contains(io)}_maybeScheduleHide(){this._config.autohide&&(this._hasMouseInteraction||this._hasKeyboardInteraction||(this._timeout=setTimeout((()=>{this.hide()}),this._config.delay)))}_onInteraction(t,e){switch(t.type){case"mouseover":case"mouseout":this._hasMouseInteraction=e;break;case"focusin":case"focusout":this._hasKeyboardInteraction=e}if(e)return void this._clearTimeout();const i=t.relatedTarget;this._element===i||this._element.contains(i)||this._maybeScheduleHide()}_setListeners(){N.on(this._element,Qs,(t=>this._onInteraction(t,!0))),N.on(this._element,Xs,(t=>this._onInteraction(t,!1))),N.on(this._element,Ys,(t=>this._onInteraction(t,!0))),N.on(this._element,Us,(t=>this._onInteraction(t,!1)))}_clearTimeout(){clearTimeout(this._timeout),this._timeout=null}static jQueryInterface(t){return this.each((function(){const e=ro.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}return R(ro),m(ro),{Alert:Q,Button:Y,Carousel:xt,Collapse:Bt,Dropdown:qi,Modal:On,Offcanvas:qn,Popover:us,ScrollSpy:Es,Tab:Vs,Toast:ro,Tooltip:cs}})); +//# sourceMappingURL=bootstrap.bundle.min.js.map \ No newline at end of file diff --git a/site_libs/clipboard/clipboard.min.js b/site_libs/clipboard/clipboard.min.js new file mode 100644 index 00000000..1103f811 --- /dev/null +++ b/site_libs/clipboard/clipboard.min.js @@ -0,0 +1,7 @@ +/*! + * clipboard.js v2.0.11 + * https://clipboardjs.com/ + * + * Licensed MIT © Zeno Rocha + */ +!function(t,e){"object"==typeof exports&&"object"==typeof module?module.exports=e():"function"==typeof define&&define.amd?define([],e):"object"==typeof exports?exports.ClipboardJS=e():t.ClipboardJS=e()}(this,function(){return n={686:function(t,e,n){"use strict";n.d(e,{default:function(){return b}});var e=n(279),i=n.n(e),e=n(370),u=n.n(e),e=n(817),r=n.n(e);function c(t){try{return document.execCommand(t)}catch(t){return}}var a=function(t){t=r()(t);return c("cut"),t};function o(t,e){var n,o,t=(n=t,o="rtl"===document.documentElement.getAttribute("dir"),(t=document.createElement("textarea")).style.fontSize="12pt",t.style.border="0",t.style.padding="0",t.style.margin="0",t.style.position="absolute",t.style[o?"right":"left"]="-9999px",o=window.pageYOffset||document.documentElement.scrollTop,t.style.top="".concat(o,"px"),t.setAttribute("readonly",""),t.value=n,t);return e.container.appendChild(t),e=r()(t),c("copy"),t.remove(),e}var f=function(t){var e=1.anchorjs-link,.anchorjs-link:focus{opacity:1}",A.sheet.cssRules.length),A.sheet.insertRule("[data-anchorjs-icon]::after{content:attr(data-anchorjs-icon)}",A.sheet.cssRules.length),A.sheet.insertRule('@font-face{font-family:anchorjs-icons;src:url(data:n/a;base64,AAEAAAALAIAAAwAwT1MvMg8yG2cAAAE4AAAAYGNtYXDp3gC3AAABpAAAAExnYXNwAAAAEAAAA9wAAAAIZ2x5ZlQCcfwAAAH4AAABCGhlYWQHFvHyAAAAvAAAADZoaGVhBnACFwAAAPQAAAAkaG10eASAADEAAAGYAAAADGxvY2EACACEAAAB8AAAAAhtYXhwAAYAVwAAARgAAAAgbmFtZQGOH9cAAAMAAAAAunBvc3QAAwAAAAADvAAAACAAAQAAAAEAAHzE2p9fDzz1AAkEAAAAAADRecUWAAAAANQA6R8AAAAAAoACwAAAAAgAAgAAAAAAAAABAAADwP/AAAACgAAA/9MCrQABAAAAAAAAAAAAAAAAAAAAAwABAAAAAwBVAAIAAAAAAAIAAAAAAAAAAAAAAAAAAAAAAAMCQAGQAAUAAAKZAswAAACPApkCzAAAAesAMwEJAAAAAAAAAAAAAAAAAAAAARAAAAAAAAAAAAAAAAAAAAAAQAAg//0DwP/AAEADwABAAAAAAQAAAAAAAAAAAAAAIAAAAAAAAAIAAAACgAAxAAAAAwAAAAMAAAAcAAEAAwAAABwAAwABAAAAHAAEADAAAAAIAAgAAgAAACDpy//9//8AAAAg6cv//f///+EWNwADAAEAAAAAAAAAAAAAAAAACACEAAEAAAAAAAAAAAAAAAAxAAACAAQARAKAAsAAKwBUAAABIiYnJjQ3NzY2MzIWFxYUBwcGIicmNDc3NjQnJiYjIgYHBwYUFxYUBwYGIwciJicmNDc3NjIXFhQHBwYUFxYWMzI2Nzc2NCcmNDc2MhcWFAcHBgYjARQGDAUtLXoWOR8fORYtLTgKGwoKCjgaGg0gEhIgDXoaGgkJBQwHdR85Fi0tOAobCgoKOBoaDSASEiANehoaCQkKGwotLXoWOR8BMwUFLYEuehYXFxYugC44CQkKGwo4GkoaDQ0NDXoaShoKGwoFBe8XFi6ALjgJCQobCjgaShoNDQ0NehpKGgobCgoKLYEuehYXAAAADACWAAEAAAAAAAEACAAAAAEAAAAAAAIAAwAIAAEAAAAAAAMACAAAAAEAAAAAAAQACAAAAAEAAAAAAAUAAQALAAEAAAAAAAYACAAAAAMAAQQJAAEAEAAMAAMAAQQJAAIABgAcAAMAAQQJAAMAEAAMAAMAAQQJAAQAEAAMAAMAAQQJAAUAAgAiAAMAAQQJAAYAEAAMYW5jaG9yanM0MDBAAGEAbgBjAGgAbwByAGoAcwA0ADAAMABAAAAAAwAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABAAH//wAP) format("truetype")}',A.sheet.cssRules.length)),h=document.querySelectorAll("[id]"),t=[].map.call(h,function(A){return A.id}),i=0;i\]./()*\\\n\t\b\v\u00A0]/g,"-").replace(/-{2,}/g,"-").substring(0,this.options.truncate).replace(/^-+|-+$/gm,"").toLowerCase()},this.hasAnchorJSLink=function(A){var e=A.firstChild&&-1<(" "+A.firstChild.className+" ").indexOf(" anchorjs-link "),A=A.lastChild&&-1<(" "+A.lastChild.className+" ").indexOf(" anchorjs-link ");return e||A||!1}}}); +// @license-end \ No newline at end of file diff --git a/site_libs/quarto-html/popper.min.js b/site_libs/quarto-html/popper.min.js new file mode 100644 index 00000000..e3726d72 --- /dev/null +++ b/site_libs/quarto-html/popper.min.js @@ -0,0 +1,6 @@ +/** + * @popperjs/core v2.11.7 - MIT License + */ + +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).Popper={})}(this,(function(e){"use strict";function t(e){if(null==e)return window;if("[object Window]"!==e.toString()){var t=e.ownerDocument;return t&&t.defaultView||window}return e}function n(e){return e instanceof t(e).Element||e instanceof Element}function r(e){return e instanceof t(e).HTMLElement||e instanceof HTMLElement}function o(e){return"undefined"!=typeof ShadowRoot&&(e instanceof t(e).ShadowRoot||e instanceof ShadowRoot)}var i=Math.max,a=Math.min,s=Math.round;function f(){var e=navigator.userAgentData;return null!=e&&e.brands&&Array.isArray(e.brands)?e.brands.map((function(e){return e.brand+"/"+e.version})).join(" "):navigator.userAgent}function c(){return!/^((?!chrome|android).)*safari/i.test(f())}function p(e,o,i){void 0===o&&(o=!1),void 0===i&&(i=!1);var a=e.getBoundingClientRect(),f=1,p=1;o&&r(e)&&(f=e.offsetWidth>0&&s(a.width)/e.offsetWidth||1,p=e.offsetHeight>0&&s(a.height)/e.offsetHeight||1);var u=(n(e)?t(e):window).visualViewport,l=!c()&&i,d=(a.left+(l&&u?u.offsetLeft:0))/f,h=(a.top+(l&&u?u.offsetTop:0))/p,m=a.width/f,v=a.height/p;return{width:m,height:v,top:h,right:d+m,bottom:h+v,left:d,x:d,y:h}}function u(e){var n=t(e);return{scrollLeft:n.pageXOffset,scrollTop:n.pageYOffset}}function l(e){return e?(e.nodeName||"").toLowerCase():null}function d(e){return((n(e)?e.ownerDocument:e.document)||window.document).documentElement}function h(e){return p(d(e)).left+u(e).scrollLeft}function m(e){return t(e).getComputedStyle(e)}function v(e){var t=m(e),n=t.overflow,r=t.overflowX,o=t.overflowY;return/auto|scroll|overlay|hidden/.test(n+o+r)}function y(e,n,o){void 0===o&&(o=!1);var i,a,f=r(n),c=r(n)&&function(e){var t=e.getBoundingClientRect(),n=s(t.width)/e.offsetWidth||1,r=s(t.height)/e.offsetHeight||1;return 1!==n||1!==r}(n),m=d(n),y=p(e,c,o),g={scrollLeft:0,scrollTop:0},b={x:0,y:0};return(f||!f&&!o)&&(("body"!==l(n)||v(m))&&(g=(i=n)!==t(i)&&r(i)?{scrollLeft:(a=i).scrollLeft,scrollTop:a.scrollTop}:u(i)),r(n)?((b=p(n,!0)).x+=n.clientLeft,b.y+=n.clientTop):m&&(b.x=h(m))),{x:y.left+g.scrollLeft-b.x,y:y.top+g.scrollTop-b.y,width:y.width,height:y.height}}function g(e){var t=p(e),n=e.offsetWidth,r=e.offsetHeight;return Math.abs(t.width-n)<=1&&(n=t.width),Math.abs(t.height-r)<=1&&(r=t.height),{x:e.offsetLeft,y:e.offsetTop,width:n,height:r}}function b(e){return"html"===l(e)?e:e.assignedSlot||e.parentNode||(o(e)?e.host:null)||d(e)}function x(e){return["html","body","#document"].indexOf(l(e))>=0?e.ownerDocument.body:r(e)&&v(e)?e:x(b(e))}function w(e,n){var r;void 0===n&&(n=[]);var o=x(e),i=o===(null==(r=e.ownerDocument)?void 0:r.body),a=t(o),s=i?[a].concat(a.visualViewport||[],v(o)?o:[]):o,f=n.concat(s);return i?f:f.concat(w(b(s)))}function O(e){return["table","td","th"].indexOf(l(e))>=0}function j(e){return r(e)&&"fixed"!==m(e).position?e.offsetParent:null}function E(e){for(var n=t(e),i=j(e);i&&O(i)&&"static"===m(i).position;)i=j(i);return i&&("html"===l(i)||"body"===l(i)&&"static"===m(i).position)?n:i||function(e){var t=/firefox/i.test(f());if(/Trident/i.test(f())&&r(e)&&"fixed"===m(e).position)return null;var n=b(e);for(o(n)&&(n=n.host);r(n)&&["html","body"].indexOf(l(n))<0;){var i=m(n);if("none"!==i.transform||"none"!==i.perspective||"paint"===i.contain||-1!==["transform","perspective"].indexOf(i.willChange)||t&&"filter"===i.willChange||t&&i.filter&&"none"!==i.filter)return n;n=n.parentNode}return null}(e)||n}var D="top",A="bottom",L="right",P="left",M="auto",k=[D,A,L,P],W="start",B="end",H="viewport",T="popper",R=k.reduce((function(e,t){return e.concat([t+"-"+W,t+"-"+B])}),[]),S=[].concat(k,[M]).reduce((function(e,t){return e.concat([t,t+"-"+W,t+"-"+B])}),[]),V=["beforeRead","read","afterRead","beforeMain","main","afterMain","beforeWrite","write","afterWrite"];function q(e){var t=new Map,n=new Set,r=[];function o(e){n.add(e.name),[].concat(e.requires||[],e.requiresIfExists||[]).forEach((function(e){if(!n.has(e)){var r=t.get(e);r&&o(r)}})),r.push(e)}return e.forEach((function(e){t.set(e.name,e)})),e.forEach((function(e){n.has(e.name)||o(e)})),r}function C(e){return e.split("-")[0]}function N(e,t){var n=t.getRootNode&&t.getRootNode();if(e.contains(t))return!0;if(n&&o(n)){var r=t;do{if(r&&e.isSameNode(r))return!0;r=r.parentNode||r.host}while(r)}return!1}function I(e){return Object.assign({},e,{left:e.x,top:e.y,right:e.x+e.width,bottom:e.y+e.height})}function _(e,r,o){return r===H?I(function(e,n){var r=t(e),o=d(e),i=r.visualViewport,a=o.clientWidth,s=o.clientHeight,f=0,p=0;if(i){a=i.width,s=i.height;var u=c();(u||!u&&"fixed"===n)&&(f=i.offsetLeft,p=i.offsetTop)}return{width:a,height:s,x:f+h(e),y:p}}(e,o)):n(r)?function(e,t){var n=p(e,!1,"fixed"===t);return n.top=n.top+e.clientTop,n.left=n.left+e.clientLeft,n.bottom=n.top+e.clientHeight,n.right=n.left+e.clientWidth,n.width=e.clientWidth,n.height=e.clientHeight,n.x=n.left,n.y=n.top,n}(r,o):I(function(e){var t,n=d(e),r=u(e),o=null==(t=e.ownerDocument)?void 0:t.body,a=i(n.scrollWidth,n.clientWidth,o?o.scrollWidth:0,o?o.clientWidth:0),s=i(n.scrollHeight,n.clientHeight,o?o.scrollHeight:0,o?o.clientHeight:0),f=-r.scrollLeft+h(e),c=-r.scrollTop;return"rtl"===m(o||n).direction&&(f+=i(n.clientWidth,o?o.clientWidth:0)-a),{width:a,height:s,x:f,y:c}}(d(e)))}function F(e,t,o,s){var f="clippingParents"===t?function(e){var t=w(b(e)),o=["absolute","fixed"].indexOf(m(e).position)>=0&&r(e)?E(e):e;return n(o)?t.filter((function(e){return n(e)&&N(e,o)&&"body"!==l(e)})):[]}(e):[].concat(t),c=[].concat(f,[o]),p=c[0],u=c.reduce((function(t,n){var r=_(e,n,s);return t.top=i(r.top,t.top),t.right=a(r.right,t.right),t.bottom=a(r.bottom,t.bottom),t.left=i(r.left,t.left),t}),_(e,p,s));return u.width=u.right-u.left,u.height=u.bottom-u.top,u.x=u.left,u.y=u.top,u}function U(e){return e.split("-")[1]}function z(e){return["top","bottom"].indexOf(e)>=0?"x":"y"}function X(e){var t,n=e.reference,r=e.element,o=e.placement,i=o?C(o):null,a=o?U(o):null,s=n.x+n.width/2-r.width/2,f=n.y+n.height/2-r.height/2;switch(i){case D:t={x:s,y:n.y-r.height};break;case A:t={x:s,y:n.y+n.height};break;case L:t={x:n.x+n.width,y:f};break;case P:t={x:n.x-r.width,y:f};break;default:t={x:n.x,y:n.y}}var c=i?z(i):null;if(null!=c){var p="y"===c?"height":"width";switch(a){case W:t[c]=t[c]-(n[p]/2-r[p]/2);break;case B:t[c]=t[c]+(n[p]/2-r[p]/2)}}return t}function Y(e){return Object.assign({},{top:0,right:0,bottom:0,left:0},e)}function G(e,t){return t.reduce((function(t,n){return t[n]=e,t}),{})}function J(e,t){void 0===t&&(t={});var r=t,o=r.placement,i=void 0===o?e.placement:o,a=r.strategy,s=void 0===a?e.strategy:a,f=r.boundary,c=void 0===f?"clippingParents":f,u=r.rootBoundary,l=void 0===u?H:u,h=r.elementContext,m=void 0===h?T:h,v=r.altBoundary,y=void 0!==v&&v,g=r.padding,b=void 0===g?0:g,x=Y("number"!=typeof b?b:G(b,k)),w=m===T?"reference":T,O=e.rects.popper,j=e.elements[y?w:m],E=F(n(j)?j:j.contextElement||d(e.elements.popper),c,l,s),P=p(e.elements.reference),M=X({reference:P,element:O,strategy:"absolute",placement:i}),W=I(Object.assign({},O,M)),B=m===T?W:P,R={top:E.top-B.top+x.top,bottom:B.bottom-E.bottom+x.bottom,left:E.left-B.left+x.left,right:B.right-E.right+x.right},S=e.modifiersData.offset;if(m===T&&S){var V=S[i];Object.keys(R).forEach((function(e){var t=[L,A].indexOf(e)>=0?1:-1,n=[D,A].indexOf(e)>=0?"y":"x";R[e]+=V[n]*t}))}return R}var K={placement:"bottom",modifiers:[],strategy:"absolute"};function Q(){for(var e=arguments.length,t=new Array(e),n=0;n=0?-1:1,i="function"==typeof n?n(Object.assign({},t,{placement:e})):n,a=i[0],s=i[1];return a=a||0,s=(s||0)*o,[P,L].indexOf(r)>=0?{x:s,y:a}:{x:a,y:s}}(n,t.rects,i),e}),{}),s=a[t.placement],f=s.x,c=s.y;null!=t.modifiersData.popperOffsets&&(t.modifiersData.popperOffsets.x+=f,t.modifiersData.popperOffsets.y+=c),t.modifiersData[r]=a}},se={left:"right",right:"left",bottom:"top",top:"bottom"};function fe(e){return e.replace(/left|right|bottom|top/g,(function(e){return se[e]}))}var ce={start:"end",end:"start"};function pe(e){return e.replace(/start|end/g,(function(e){return ce[e]}))}function ue(e,t){void 0===t&&(t={});var n=t,r=n.placement,o=n.boundary,i=n.rootBoundary,a=n.padding,s=n.flipVariations,f=n.allowedAutoPlacements,c=void 0===f?S:f,p=U(r),u=p?s?R:R.filter((function(e){return U(e)===p})):k,l=u.filter((function(e){return c.indexOf(e)>=0}));0===l.length&&(l=u);var d=l.reduce((function(t,n){return t[n]=J(e,{placement:n,boundary:o,rootBoundary:i,padding:a})[C(n)],t}),{});return Object.keys(d).sort((function(e,t){return d[e]-d[t]}))}var le={name:"flip",enabled:!0,phase:"main",fn:function(e){var t=e.state,n=e.options,r=e.name;if(!t.modifiersData[r]._skip){for(var o=n.mainAxis,i=void 0===o||o,a=n.altAxis,s=void 0===a||a,f=n.fallbackPlacements,c=n.padding,p=n.boundary,u=n.rootBoundary,l=n.altBoundary,d=n.flipVariations,h=void 0===d||d,m=n.allowedAutoPlacements,v=t.options.placement,y=C(v),g=f||(y===v||!h?[fe(v)]:function(e){if(C(e)===M)return[];var t=fe(e);return[pe(e),t,pe(t)]}(v)),b=[v].concat(g).reduce((function(e,n){return e.concat(C(n)===M?ue(t,{placement:n,boundary:p,rootBoundary:u,padding:c,flipVariations:h,allowedAutoPlacements:m}):n)}),[]),x=t.rects.reference,w=t.rects.popper,O=new Map,j=!0,E=b[0],k=0;k=0,S=R?"width":"height",V=J(t,{placement:B,boundary:p,rootBoundary:u,altBoundary:l,padding:c}),q=R?T?L:P:T?A:D;x[S]>w[S]&&(q=fe(q));var N=fe(q),I=[];if(i&&I.push(V[H]<=0),s&&I.push(V[q]<=0,V[N]<=0),I.every((function(e){return e}))){E=B,j=!1;break}O.set(B,I)}if(j)for(var _=function(e){var t=b.find((function(t){var n=O.get(t);if(n)return n.slice(0,e).every((function(e){return e}))}));if(t)return E=t,"break"},F=h?3:1;F>0;F--){if("break"===_(F))break}t.placement!==E&&(t.modifiersData[r]._skip=!0,t.placement=E,t.reset=!0)}},requiresIfExists:["offset"],data:{_skip:!1}};function de(e,t,n){return i(e,a(t,n))}var he={name:"preventOverflow",enabled:!0,phase:"main",fn:function(e){var t=e.state,n=e.options,r=e.name,o=n.mainAxis,s=void 0===o||o,f=n.altAxis,c=void 0!==f&&f,p=n.boundary,u=n.rootBoundary,l=n.altBoundary,d=n.padding,h=n.tether,m=void 0===h||h,v=n.tetherOffset,y=void 0===v?0:v,b=J(t,{boundary:p,rootBoundary:u,padding:d,altBoundary:l}),x=C(t.placement),w=U(t.placement),O=!w,j=z(x),M="x"===j?"y":"x",k=t.modifiersData.popperOffsets,B=t.rects.reference,H=t.rects.popper,T="function"==typeof y?y(Object.assign({},t.rects,{placement:t.placement})):y,R="number"==typeof T?{mainAxis:T,altAxis:T}:Object.assign({mainAxis:0,altAxis:0},T),S=t.modifiersData.offset?t.modifiersData.offset[t.placement]:null,V={x:0,y:0};if(k){if(s){var q,N="y"===j?D:P,I="y"===j?A:L,_="y"===j?"height":"width",F=k[j],X=F+b[N],Y=F-b[I],G=m?-H[_]/2:0,K=w===W?B[_]:H[_],Q=w===W?-H[_]:-B[_],Z=t.elements.arrow,$=m&&Z?g(Z):{width:0,height:0},ee=t.modifiersData["arrow#persistent"]?t.modifiersData["arrow#persistent"].padding:{top:0,right:0,bottom:0,left:0},te=ee[N],ne=ee[I],re=de(0,B[_],$[_]),oe=O?B[_]/2-G-re-te-R.mainAxis:K-re-te-R.mainAxis,ie=O?-B[_]/2+G+re+ne+R.mainAxis:Q+re+ne+R.mainAxis,ae=t.elements.arrow&&E(t.elements.arrow),se=ae?"y"===j?ae.clientTop||0:ae.clientLeft||0:0,fe=null!=(q=null==S?void 0:S[j])?q:0,ce=F+ie-fe,pe=de(m?a(X,F+oe-fe-se):X,F,m?i(Y,ce):Y);k[j]=pe,V[j]=pe-F}if(c){var ue,le="x"===j?D:P,he="x"===j?A:L,me=k[M],ve="y"===M?"height":"width",ye=me+b[le],ge=me-b[he],be=-1!==[D,P].indexOf(x),xe=null!=(ue=null==S?void 0:S[M])?ue:0,we=be?ye:me-B[ve]-H[ve]-xe+R.altAxis,Oe=be?me+B[ve]+H[ve]-xe-R.altAxis:ge,je=m&&be?function(e,t,n){var r=de(e,t,n);return r>n?n:r}(we,me,Oe):de(m?we:ye,me,m?Oe:ge);k[M]=je,V[M]=je-me}t.modifiersData[r]=V}},requiresIfExists:["offset"]};var me={name:"arrow",enabled:!0,phase:"main",fn:function(e){var t,n=e.state,r=e.name,o=e.options,i=n.elements.arrow,a=n.modifiersData.popperOffsets,s=C(n.placement),f=z(s),c=[P,L].indexOf(s)>=0?"height":"width";if(i&&a){var p=function(e,t){return Y("number"!=typeof(e="function"==typeof e?e(Object.assign({},t.rects,{placement:t.placement})):e)?e:G(e,k))}(o.padding,n),u=g(i),l="y"===f?D:P,d="y"===f?A:L,h=n.rects.reference[c]+n.rects.reference[f]-a[f]-n.rects.popper[c],m=a[f]-n.rects.reference[f],v=E(i),y=v?"y"===f?v.clientHeight||0:v.clientWidth||0:0,b=h/2-m/2,x=p[l],w=y-u[c]-p[d],O=y/2-u[c]/2+b,j=de(x,O,w),M=f;n.modifiersData[r]=((t={})[M]=j,t.centerOffset=j-O,t)}},effect:function(e){var t=e.state,n=e.options.element,r=void 0===n?"[data-popper-arrow]":n;null!=r&&("string"!=typeof r||(r=t.elements.popper.querySelector(r)))&&N(t.elements.popper,r)&&(t.elements.arrow=r)},requires:["popperOffsets"],requiresIfExists:["preventOverflow"]};function ve(e,t,n){return void 0===n&&(n={x:0,y:0}),{top:e.top-t.height-n.y,right:e.right-t.width+n.x,bottom:e.bottom-t.height+n.y,left:e.left-t.width-n.x}}function ye(e){return[D,L,A,P].some((function(t){return e[t]>=0}))}var ge={name:"hide",enabled:!0,phase:"main",requiresIfExists:["preventOverflow"],fn:function(e){var t=e.state,n=e.name,r=t.rects.reference,o=t.rects.popper,i=t.modifiersData.preventOverflow,a=J(t,{elementContext:"reference"}),s=J(t,{altBoundary:!0}),f=ve(a,r),c=ve(s,o,i),p=ye(f),u=ye(c);t.modifiersData[n]={referenceClippingOffsets:f,popperEscapeOffsets:c,isReferenceHidden:p,hasPopperEscaped:u},t.attributes.popper=Object.assign({},t.attributes.popper,{"data-popper-reference-hidden":p,"data-popper-escaped":u})}},be=Z({defaultModifiers:[ee,te,oe,ie]}),xe=[ee,te,oe,ie,ae,le,he,me,ge],we=Z({defaultModifiers:xe});e.applyStyles=ie,e.arrow=me,e.computeStyles=oe,e.createPopper=we,e.createPopperLite=be,e.defaultModifiers=xe,e.detectOverflow=J,e.eventListeners=ee,e.flip=le,e.hide=ge,e.offset=ae,e.popperGenerator=Z,e.popperOffsets=te,e.preventOverflow=he,Object.defineProperty(e,"__esModule",{value:!0})})); + diff --git a/site_libs/quarto-html/quarto-syntax-highlighting-549806ee2085284f45b00abea8c6df48.css b/site_libs/quarto-html/quarto-syntax-highlighting-549806ee2085284f45b00abea8c6df48.css new file mode 100644 index 00000000..80e34e41 --- /dev/null +++ b/site_libs/quarto-html/quarto-syntax-highlighting-549806ee2085284f45b00abea8c6df48.css @@ -0,0 +1,205 @@ +/* quarto syntax highlight colors */ +:root { + --quarto-hl-ot-color: #003B4F; + --quarto-hl-at-color: #657422; + --quarto-hl-ss-color: #20794D; + --quarto-hl-an-color: #5E5E5E; + --quarto-hl-fu-color: #4758AB; + --quarto-hl-st-color: #20794D; + --quarto-hl-cf-color: #003B4F; + --quarto-hl-op-color: #5E5E5E; + --quarto-hl-er-color: #AD0000; + --quarto-hl-bn-color: #AD0000; + --quarto-hl-al-color: #AD0000; + --quarto-hl-va-color: #111111; + --quarto-hl-bu-color: inherit; + --quarto-hl-ex-color: inherit; + --quarto-hl-pp-color: #AD0000; + --quarto-hl-in-color: #5E5E5E; + --quarto-hl-vs-color: #20794D; + --quarto-hl-wa-color: #5E5E5E; + --quarto-hl-do-color: #5E5E5E; + --quarto-hl-im-color: #00769E; + --quarto-hl-ch-color: #20794D; + --quarto-hl-dt-color: #AD0000; + --quarto-hl-fl-color: #AD0000; + --quarto-hl-co-color: #5E5E5E; + --quarto-hl-cv-color: #5E5E5E; + --quarto-hl-cn-color: #8f5902; + --quarto-hl-sc-color: #5E5E5E; + --quarto-hl-dv-color: #AD0000; + --quarto-hl-kw-color: #003B4F; +} + +/* other quarto variables */ +:root { + --quarto-font-monospace: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; +} + +pre > code.sourceCode > span { + color: #003B4F; +} + +code span { + color: #003B4F; +} + +code.sourceCode > span { + color: #003B4F; +} + +div.sourceCode, +div.sourceCode pre.sourceCode { + color: #003B4F; +} + +code span.ot { + color: #003B4F; + font-style: inherit; +} + +code span.at { + color: #657422; + font-style: inherit; +} + +code span.ss { + color: #20794D; + font-style: inherit; +} + +code span.an { + color: #5E5E5E; + font-style: inherit; +} + +code span.fu { + color: #4758AB; + font-style: inherit; +} + +code span.st { + color: #20794D; + font-style: inherit; +} + +code span.cf { + color: #003B4F; + font-weight: bold; + font-style: inherit; +} + +code span.op { + color: #5E5E5E; + font-style: inherit; +} + +code span.er { + color: #AD0000; + font-style: inherit; +} + +code span.bn { + color: #AD0000; + font-style: inherit; +} + +code span.al { + color: #AD0000; + font-style: inherit; +} + +code span.va { + color: #111111; + font-style: inherit; +} + +code span.bu { + font-style: inherit; +} + +code span.ex { + font-style: inherit; +} + +code span.pp { + color: #AD0000; + font-style: inherit; +} + +code span.in { + color: #5E5E5E; + font-style: inherit; +} + +code span.vs { + color: #20794D; + font-style: inherit; +} + +code span.wa { + color: #5E5E5E; + font-style: italic; +} + +code span.do { + color: #5E5E5E; + font-style: italic; +} + +code span.im { + color: #00769E; + font-style: inherit; +} + +code span.ch { + color: #20794D; + font-style: inherit; +} + +code span.dt { + color: #AD0000; + font-style: inherit; +} + +code span.fl { + color: #AD0000; + font-style: inherit; +} + +code span.co { + color: #5E5E5E; + font-style: inherit; +} + +code span.cv { + color: #5E5E5E; + font-style: italic; +} + +code span.cn { + color: #8f5902; + font-style: inherit; +} + +code span.sc { + color: #5E5E5E; + font-style: inherit; +} + +code span.dv { + color: #AD0000; + font-style: inherit; +} + +code span.kw { + color: #003B4F; + font-weight: bold; + font-style: inherit; +} + +.prevent-inlining { + content: " { + // Find any conflicting margin elements and add margins to the + // top to prevent overlap + const marginChildren = window.document.querySelectorAll( + ".column-margin.column-container > *, .margin-caption, .aside" + ); + + let lastBottom = 0; + for (const marginChild of marginChildren) { + if (marginChild.offsetParent !== null) { + // clear the top margin so we recompute it + marginChild.style.marginTop = null; + const top = marginChild.getBoundingClientRect().top + window.scrollY; + if (top < lastBottom) { + const marginChildStyle = window.getComputedStyle(marginChild); + const marginBottom = parseFloat(marginChildStyle["marginBottom"]); + const margin = lastBottom - top + marginBottom; + marginChild.style.marginTop = `${margin}px`; + } + const styles = window.getComputedStyle(marginChild); + const marginTop = parseFloat(styles["marginTop"]); + lastBottom = top + marginChild.getBoundingClientRect().height + marginTop; + } + } +}; + +window.document.addEventListener("DOMContentLoaded", function (_event) { + // Recompute the position of margin elements anytime the body size changes + if (window.ResizeObserver) { + const resizeObserver = new window.ResizeObserver( + throttle(() => { + layoutMarginEls(); + if ( + window.document.body.getBoundingClientRect().width < 990 && + isReaderMode() + ) { + quartoToggleReader(); + } + }, 50) + ); + resizeObserver.observe(window.document.body); + } + + const tocEl = window.document.querySelector('nav.toc-active[role="doc-toc"]'); + const sidebarEl = window.document.getElementById("quarto-sidebar"); + const leftTocEl = window.document.getElementById("quarto-sidebar-toc-left"); + const marginSidebarEl = window.document.getElementById( + "quarto-margin-sidebar" + ); + // function to determine whether the element has a previous sibling that is active + const prevSiblingIsActiveLink = (el) => { + const sibling = el.previousElementSibling; + if (sibling && sibling.tagName === "A") { + return sibling.classList.contains("active"); + } else { + return false; + } + }; + + // fire slideEnter for bootstrap tab activations (for htmlwidget resize behavior) + function fireSlideEnter(e) { + const event = window.document.createEvent("Event"); + event.initEvent("slideenter", true, true); + window.document.dispatchEvent(event); + } + const tabs = window.document.querySelectorAll('a[data-bs-toggle="tab"]'); + tabs.forEach((tab) => { + tab.addEventListener("shown.bs.tab", fireSlideEnter); + }); + + // fire slideEnter for tabby tab activations (for htmlwidget resize behavior) + document.addEventListener("tabby", fireSlideEnter, false); + + // Track scrolling and mark TOC links as active + // get table of contents and sidebar (bail if we don't have at least one) + const tocLinks = tocEl + ? [...tocEl.querySelectorAll("a[data-scroll-target]")] + : []; + const makeActive = (link) => tocLinks[link].classList.add("active"); + const removeActive = (link) => tocLinks[link].classList.remove("active"); + const removeAllActive = () => + [...Array(tocLinks.length).keys()].forEach((link) => removeActive(link)); + + // activate the anchor for a section associated with this TOC entry + tocLinks.forEach((link) => { + link.addEventListener("click", () => { + if (link.href.indexOf("#") !== -1) { + const anchor = link.href.split("#")[1]; + const heading = window.document.querySelector( + `[data-anchor-id="${anchor}"]` + ); + if (heading) { + // Add the class + heading.classList.add("reveal-anchorjs-link"); + + // function to show the anchor + const handleMouseout = () => { + heading.classList.remove("reveal-anchorjs-link"); + heading.removeEventListener("mouseout", handleMouseout); + }; + + // add a function to clear the anchor when the user mouses out of it + heading.addEventListener("mouseout", handleMouseout); + } + } + }); + }); + + const sections = tocLinks.map((link) => { + const target = link.getAttribute("data-scroll-target"); + if (target.startsWith("#")) { + return window.document.getElementById(decodeURI(`${target.slice(1)}`)); + } else { + return window.document.querySelector(decodeURI(`${target}`)); + } + }); + + const sectionMargin = 200; + let currentActive = 0; + // track whether we've initialized state the first time + let init = false; + + const updateActiveLink = () => { + // The index from bottom to top (e.g. reversed list) + let sectionIndex = -1; + if ( + window.innerHeight + window.pageYOffset >= + window.document.body.offsetHeight + ) { + // This is the no-scroll case where last section should be the active one + sectionIndex = 0; + } else { + // This finds the last section visible on screen that should be made active + sectionIndex = [...sections].reverse().findIndex((section) => { + if (section) { + return window.pageYOffset >= section.offsetTop - sectionMargin; + } else { + return false; + } + }); + } + if (sectionIndex > -1) { + const current = sections.length - sectionIndex - 1; + if (current !== currentActive) { + removeAllActive(); + currentActive = current; + makeActive(current); + if (init) { + window.dispatchEvent(sectionChanged); + } + init = true; + } + } + }; + + const inHiddenRegion = (top, bottom, hiddenRegions) => { + for (const region of hiddenRegions) { + if (top <= region.bottom && bottom >= region.top) { + return true; + } + } + return false; + }; + + const categorySelector = "header.quarto-title-block .quarto-category"; + const activateCategories = (href) => { + // Find any categories + // Surround them with a link pointing back to: + // #category=Authoring + try { + const categoryEls = window.document.querySelectorAll(categorySelector); + for (const categoryEl of categoryEls) { + const categoryText = categoryEl.textContent; + if (categoryText) { + const link = `${href}#category=${encodeURIComponent(categoryText)}`; + const linkEl = window.document.createElement("a"); + linkEl.setAttribute("href", link); + for (const child of categoryEl.childNodes) { + linkEl.append(child); + } + categoryEl.appendChild(linkEl); + } + } + } catch { + // Ignore errors + } + }; + function hasTitleCategories() { + return window.document.querySelector(categorySelector) !== null; + } + + function offsetRelativeUrl(url) { + const offset = getMeta("quarto:offset"); + return offset ? offset + url : url; + } + + function offsetAbsoluteUrl(url) { + const offset = getMeta("quarto:offset"); + const baseUrl = new URL(offset, window.location); + + const projRelativeUrl = url.replace(baseUrl, ""); + if (projRelativeUrl.startsWith("/")) { + return projRelativeUrl; + } else { + return "/" + projRelativeUrl; + } + } + + // read a meta tag value + function getMeta(metaName) { + const metas = window.document.getElementsByTagName("meta"); + for (let i = 0; i < metas.length; i++) { + if (metas[i].getAttribute("name") === metaName) { + return metas[i].getAttribute("content"); + } + } + return ""; + } + + async function findAndActivateCategories() { + // Categories search with listing only use path without query + const currentPagePath = offsetAbsoluteUrl( + window.location.origin + window.location.pathname + ); + const response = await fetch(offsetRelativeUrl("listings.json")); + if (response.status == 200) { + return response.json().then(function (listingPaths) { + const listingHrefs = []; + for (const listingPath of listingPaths) { + const pathWithoutLeadingSlash = listingPath.listing.substring(1); + for (const item of listingPath.items) { + if ( + item === currentPagePath || + item === currentPagePath + "index.html" + ) { + // Resolve this path against the offset to be sure + // we already are using the correct path to the listing + // (this adjusts the listing urls to be rooted against + // whatever root the page is actually running against) + const relative = offsetRelativeUrl(pathWithoutLeadingSlash); + const baseUrl = window.location; + const resolvedPath = new URL(relative, baseUrl); + listingHrefs.push(resolvedPath.pathname); + break; + } + } + } + + // Look up the tree for a nearby linting and use that if we find one + const nearestListing = findNearestParentListing( + offsetAbsoluteUrl(window.location.pathname), + listingHrefs + ); + if (nearestListing) { + activateCategories(nearestListing); + } else { + // See if the referrer is a listing page for this item + const referredRelativePath = offsetAbsoluteUrl(document.referrer); + const referrerListing = listingHrefs.find((listingHref) => { + const isListingReferrer = + listingHref === referredRelativePath || + listingHref === referredRelativePath + "index.html"; + return isListingReferrer; + }); + + if (referrerListing) { + // Try to use the referrer if possible + activateCategories(referrerListing); + } else if (listingHrefs.length > 0) { + // Otherwise, just fall back to the first listing + activateCategories(listingHrefs[0]); + } + } + }); + } + } + if (hasTitleCategories()) { + findAndActivateCategories(); + } + + const findNearestParentListing = (href, listingHrefs) => { + if (!href || !listingHrefs) { + return undefined; + } + // Look up the tree for a nearby linting and use that if we find one + const relativeParts = href.substring(1).split("/"); + while (relativeParts.length > 0) { + const path = relativeParts.join("/"); + for (const listingHref of listingHrefs) { + if (listingHref.startsWith(path)) { + return listingHref; + } + } + relativeParts.pop(); + } + + return undefined; + }; + + const manageSidebarVisiblity = (el, placeholderDescriptor) => { + let isVisible = true; + let elRect; + + return (hiddenRegions) => { + if (el === null) { + return; + } + + // Find the last element of the TOC + const lastChildEl = el.lastElementChild; + + if (lastChildEl) { + // Converts the sidebar to a menu + const convertToMenu = () => { + for (const child of el.children) { + child.style.opacity = 0; + child.style.overflow = "hidden"; + child.style.pointerEvents = "none"; + } + + nexttick(() => { + const toggleContainer = window.document.createElement("div"); + toggleContainer.style.width = "100%"; + toggleContainer.classList.add("zindex-over-content"); + toggleContainer.classList.add("quarto-sidebar-toggle"); + toggleContainer.classList.add("headroom-target"); // Marks this to be managed by headeroom + toggleContainer.id = placeholderDescriptor.id; + toggleContainer.style.position = "fixed"; + + const toggleIcon = window.document.createElement("i"); + toggleIcon.classList.add("quarto-sidebar-toggle-icon"); + toggleIcon.classList.add("bi"); + toggleIcon.classList.add("bi-caret-down-fill"); + + const toggleTitle = window.document.createElement("div"); + const titleEl = window.document.body.querySelector( + placeholderDescriptor.titleSelector + ); + if (titleEl) { + toggleTitle.append( + titleEl.textContent || titleEl.innerText, + toggleIcon + ); + } + toggleTitle.classList.add("zindex-over-content"); + toggleTitle.classList.add("quarto-sidebar-toggle-title"); + toggleContainer.append(toggleTitle); + + const toggleContents = window.document.createElement("div"); + toggleContents.classList = el.classList; + toggleContents.classList.add("zindex-over-content"); + toggleContents.classList.add("quarto-sidebar-toggle-contents"); + for (const child of el.children) { + if (child.id === "toc-title") { + continue; + } + + const clone = child.cloneNode(true); + clone.style.opacity = 1; + clone.style.pointerEvents = null; + clone.style.display = null; + toggleContents.append(clone); + } + toggleContents.style.height = "0px"; + const positionToggle = () => { + // position the element (top left of parent, same width as parent) + if (!elRect) { + elRect = el.getBoundingClientRect(); + } + toggleContainer.style.left = `${elRect.left}px`; + toggleContainer.style.top = `${elRect.top}px`; + toggleContainer.style.width = `${elRect.width}px`; + }; + positionToggle(); + + toggleContainer.append(toggleContents); + el.parentElement.prepend(toggleContainer); + + // Process clicks + let tocShowing = false; + // Allow the caller to control whether this is dismissed + // when it is clicked (e.g. sidebar navigation supports + // opening and closing the nav tree, so don't dismiss on click) + const clickEl = placeholderDescriptor.dismissOnClick + ? toggleContainer + : toggleTitle; + + const closeToggle = () => { + if (tocShowing) { + toggleContainer.classList.remove("expanded"); + toggleContents.style.height = "0px"; + tocShowing = false; + } + }; + + // Get rid of any expanded toggle if the user scrolls + window.document.addEventListener( + "scroll", + throttle(() => { + closeToggle(); + }, 50) + ); + + // Handle positioning of the toggle + window.addEventListener( + "resize", + throttle(() => { + elRect = undefined; + positionToggle(); + }, 50) + ); + + window.addEventListener("quarto-hrChanged", () => { + elRect = undefined; + }); + + // Process the click + clickEl.onclick = () => { + if (!tocShowing) { + toggleContainer.classList.add("expanded"); + toggleContents.style.height = null; + tocShowing = true; + } else { + closeToggle(); + } + }; + }); + }; + + // Converts a sidebar from a menu back to a sidebar + const convertToSidebar = () => { + for (const child of el.children) { + child.style.opacity = 1; + child.style.overflow = null; + child.style.pointerEvents = null; + } + + const placeholderEl = window.document.getElementById( + placeholderDescriptor.id + ); + if (placeholderEl) { + placeholderEl.remove(); + } + + el.classList.remove("rollup"); + }; + + if (isReaderMode()) { + convertToMenu(); + isVisible = false; + } else { + // Find the top and bottom o the element that is being managed + const elTop = el.offsetTop; + const elBottom = + elTop + lastChildEl.offsetTop + lastChildEl.offsetHeight; + + if (!isVisible) { + // If the element is current not visible reveal if there are + // no conflicts with overlay regions + if (!inHiddenRegion(elTop, elBottom, hiddenRegions)) { + convertToSidebar(); + isVisible = true; + } + } else { + // If the element is visible, hide it if it conflicts with overlay regions + // and insert a placeholder toggle (or if we're in reader mode) + if (inHiddenRegion(elTop, elBottom, hiddenRegions)) { + convertToMenu(); + isVisible = false; + } + } + } + } + }; + }; + + const tabEls = document.querySelectorAll('a[data-bs-toggle="tab"]'); + for (const tabEl of tabEls) { + const id = tabEl.getAttribute("data-bs-target"); + if (id) { + const columnEl = document.querySelector( + `${id} .column-margin, .tabset-margin-content` + ); + if (columnEl) + tabEl.addEventListener("shown.bs.tab", function (event) { + const el = event.srcElement; + if (el) { + const visibleCls = `${el.id}-margin-content`; + // walk up until we find a parent tabset + let panelTabsetEl = el.parentElement; + while (panelTabsetEl) { + if (panelTabsetEl.classList.contains("panel-tabset")) { + break; + } + panelTabsetEl = panelTabsetEl.parentElement; + } + + if (panelTabsetEl) { + const prevSib = panelTabsetEl.previousElementSibling; + if ( + prevSib && + prevSib.classList.contains("tabset-margin-container") + ) { + const childNodes = prevSib.querySelectorAll( + ".tabset-margin-content" + ); + for (const childEl of childNodes) { + if (childEl.classList.contains(visibleCls)) { + childEl.classList.remove("collapse"); + } else { + childEl.classList.add("collapse"); + } + } + } + } + } + + layoutMarginEls(); + }); + } + } + + // Manage the visibility of the toc and the sidebar + const marginScrollVisibility = manageSidebarVisiblity(marginSidebarEl, { + id: "quarto-toc-toggle", + titleSelector: "#toc-title", + dismissOnClick: true, + }); + const sidebarScrollVisiblity = manageSidebarVisiblity(sidebarEl, { + id: "quarto-sidebarnav-toggle", + titleSelector: ".title", + dismissOnClick: false, + }); + let tocLeftScrollVisibility; + if (leftTocEl) { + tocLeftScrollVisibility = manageSidebarVisiblity(leftTocEl, { + id: "quarto-lefttoc-toggle", + titleSelector: "#toc-title", + dismissOnClick: true, + }); + } + + // Find the first element that uses formatting in special columns + const conflictingEls = window.document.body.querySelectorAll( + '[class^="column-"], [class*=" column-"], aside, [class*="margin-caption"], [class*=" margin-caption"], [class*="margin-ref"], [class*=" margin-ref"]' + ); + + // Filter all the possibly conflicting elements into ones + // the do conflict on the left or ride side + const arrConflictingEls = Array.from(conflictingEls); + const leftSideConflictEls = arrConflictingEls.filter((el) => { + if (el.tagName === "ASIDE") { + return false; + } + return Array.from(el.classList).find((className) => { + return ( + className !== "column-body" && + className.startsWith("column-") && + !className.endsWith("right") && + !className.endsWith("container") && + className !== "column-margin" + ); + }); + }); + const rightSideConflictEls = arrConflictingEls.filter((el) => { + if (el.tagName === "ASIDE") { + return true; + } + + const hasMarginCaption = Array.from(el.classList).find((className) => { + return className == "margin-caption"; + }); + if (hasMarginCaption) { + return true; + } + + return Array.from(el.classList).find((className) => { + return ( + className !== "column-body" && + !className.endsWith("container") && + className.startsWith("column-") && + !className.endsWith("left") + ); + }); + }); + + const kOverlapPaddingSize = 10; + function toRegions(els) { + return els.map((el) => { + const boundRect = el.getBoundingClientRect(); + const top = + boundRect.top + + document.documentElement.scrollTop - + kOverlapPaddingSize; + return { + top, + bottom: top + el.scrollHeight + 2 * kOverlapPaddingSize, + }; + }); + } + + let hasObserved = false; + const visibleItemObserver = (els) => { + let visibleElements = [...els]; + const intersectionObserver = new IntersectionObserver( + (entries, _observer) => { + entries.forEach((entry) => { + if (entry.isIntersecting) { + if (visibleElements.indexOf(entry.target) === -1) { + visibleElements.push(entry.target); + } + } else { + visibleElements = visibleElements.filter((visibleEntry) => { + return visibleEntry !== entry; + }); + } + }); + + if (!hasObserved) { + hideOverlappedSidebars(); + } + hasObserved = true; + }, + {} + ); + els.forEach((el) => { + intersectionObserver.observe(el); + }); + + return { + getVisibleEntries: () => { + return visibleElements; + }, + }; + }; + + const rightElementObserver = visibleItemObserver(rightSideConflictEls); + const leftElementObserver = visibleItemObserver(leftSideConflictEls); + + const hideOverlappedSidebars = () => { + marginScrollVisibility(toRegions(rightElementObserver.getVisibleEntries())); + sidebarScrollVisiblity(toRegions(leftElementObserver.getVisibleEntries())); + if (tocLeftScrollVisibility) { + tocLeftScrollVisibility( + toRegions(leftElementObserver.getVisibleEntries()) + ); + } + }; + + window.quartoToggleReader = () => { + // Applies a slow class (or removes it) + // to update the transition speed + const slowTransition = (slow) => { + const manageTransition = (id, slow) => { + const el = document.getElementById(id); + if (el) { + if (slow) { + el.classList.add("slow"); + } else { + el.classList.remove("slow"); + } + } + }; + + manageTransition("TOC", slow); + manageTransition("quarto-sidebar", slow); + }; + const readerMode = !isReaderMode(); + setReaderModeValue(readerMode); + + // If we're entering reader mode, slow the transition + if (readerMode) { + slowTransition(readerMode); + } + highlightReaderToggle(readerMode); + hideOverlappedSidebars(); + + // If we're exiting reader mode, restore the non-slow transition + if (!readerMode) { + slowTransition(!readerMode); + } + }; + + const highlightReaderToggle = (readerMode) => { + const els = document.querySelectorAll(".quarto-reader-toggle"); + if (els) { + els.forEach((el) => { + if (readerMode) { + el.classList.add("reader"); + } else { + el.classList.remove("reader"); + } + }); + } + }; + + const setReaderModeValue = (val) => { + if (window.location.protocol !== "file:") { + window.localStorage.setItem("quarto-reader-mode", val); + } else { + localReaderMode = val; + } + }; + + const isReaderMode = () => { + if (window.location.protocol !== "file:") { + return window.localStorage.getItem("quarto-reader-mode") === "true"; + } else { + return localReaderMode; + } + }; + let localReaderMode = null; + + const tocOpenDepthStr = tocEl?.getAttribute("data-toc-expanded"); + const tocOpenDepth = tocOpenDepthStr ? Number(tocOpenDepthStr) : 1; + + // Walk the TOC and collapse/expand nodes + // Nodes are expanded if: + // - they are top level + // - they have children that are 'active' links + // - they are directly below an link that is 'active' + const walk = (el, depth) => { + // Tick depth when we enter a UL + if (el.tagName === "UL") { + depth = depth + 1; + } + + // It this is active link + let isActiveNode = false; + if (el.tagName === "A" && el.classList.contains("active")) { + isActiveNode = true; + } + + // See if there is an active child to this element + let hasActiveChild = false; + for (child of el.children) { + hasActiveChild = walk(child, depth) || hasActiveChild; + } + + // Process the collapse state if this is an UL + if (el.tagName === "UL") { + if (tocOpenDepth === -1 && depth > 1) { + // toc-expand: false + el.classList.add("collapse"); + } else if ( + depth <= tocOpenDepth || + hasActiveChild || + prevSiblingIsActiveLink(el) + ) { + el.classList.remove("collapse"); + } else { + el.classList.add("collapse"); + } + + // untick depth when we leave a UL + depth = depth - 1; + } + return hasActiveChild || isActiveNode; + }; + + // walk the TOC and expand / collapse any items that should be shown + if (tocEl) { + updateActiveLink(); + walk(tocEl, 0); + } + + // Throttle the scroll event and walk peridiocally + window.document.addEventListener( + "scroll", + throttle(() => { + if (tocEl) { + updateActiveLink(); + walk(tocEl, 0); + } + if (!isReaderMode()) { + hideOverlappedSidebars(); + } + }, 5) + ); + window.addEventListener( + "resize", + throttle(() => { + if (tocEl) { + updateActiveLink(); + walk(tocEl, 0); + } + if (!isReaderMode()) { + hideOverlappedSidebars(); + } + }, 10) + ); + hideOverlappedSidebars(); + highlightReaderToggle(isReaderMode()); +}); + +// grouped tabsets +window.addEventListener("pageshow", (_event) => { + function getTabSettings() { + const data = localStorage.getItem("quarto-persistent-tabsets-data"); + if (!data) { + localStorage.setItem("quarto-persistent-tabsets-data", "{}"); + return {}; + } + if (data) { + return JSON.parse(data); + } + } + + function setTabSettings(data) { + localStorage.setItem( + "quarto-persistent-tabsets-data", + JSON.stringify(data) + ); + } + + function setTabState(groupName, groupValue) { + const data = getTabSettings(); + data[groupName] = groupValue; + setTabSettings(data); + } + + function toggleTab(tab, active) { + const tabPanelId = tab.getAttribute("aria-controls"); + const tabPanel = document.getElementById(tabPanelId); + if (active) { + tab.classList.add("active"); + tabPanel.classList.add("active"); + } else { + tab.classList.remove("active"); + tabPanel.classList.remove("active"); + } + } + + function toggleAll(selectedGroup, selectorsToSync) { + for (const [thisGroup, tabs] of Object.entries(selectorsToSync)) { + const active = selectedGroup === thisGroup; + for (const tab of tabs) { + toggleTab(tab, active); + } + } + } + + function findSelectorsToSyncByLanguage() { + const result = {}; + const tabs = Array.from( + document.querySelectorAll(`div[data-group] a[id^='tabset-']`) + ); + for (const item of tabs) { + const div = item.parentElement.parentElement.parentElement; + const group = div.getAttribute("data-group"); + if (!result[group]) { + result[group] = {}; + } + const selectorsToSync = result[group]; + const value = item.innerHTML; + if (!selectorsToSync[value]) { + selectorsToSync[value] = []; + } + selectorsToSync[value].push(item); + } + return result; + } + + function setupSelectorSync() { + const selectorsToSync = findSelectorsToSyncByLanguage(); + Object.entries(selectorsToSync).forEach(([group, tabSetsByValue]) => { + Object.entries(tabSetsByValue).forEach(([value, items]) => { + items.forEach((item) => { + item.addEventListener("click", (_event) => { + setTabState(group, value); + toggleAll(value, selectorsToSync[group]); + }); + }); + }); + }); + return selectorsToSync; + } + + const selectorsToSync = setupSelectorSync(); + for (const [group, selectedName] of Object.entries(getTabSettings())) { + const selectors = selectorsToSync[group]; + // it's possible that stale state gives us empty selections, so we explicitly check here. + if (selectors) { + toggleAll(selectedName, selectors); + } + } +}); + +function throttle(func, wait) { + let waiting = false; + return function () { + if (!waiting) { + func.apply(this, arguments); + waiting = true; + setTimeout(function () { + waiting = false; + }, wait); + } + }; +} + +function nexttick(func) { + return setTimeout(func, 0); +} diff --git a/site_libs/quarto-html/tippy.css b/site_libs/quarto-html/tippy.css new file mode 100644 index 00000000..e6ae635c --- /dev/null +++ b/site_libs/quarto-html/tippy.css @@ -0,0 +1 @@ +.tippy-box[data-animation=fade][data-state=hidden]{opacity:0}[data-tippy-root]{max-width:calc(100vw - 10px)}.tippy-box{position:relative;background-color:#333;color:#fff;border-radius:4px;font-size:14px;line-height:1.4;white-space:normal;outline:0;transition-property:transform,visibility,opacity}.tippy-box[data-placement^=top]>.tippy-arrow{bottom:0}.tippy-box[data-placement^=top]>.tippy-arrow:before{bottom:-7px;left:0;border-width:8px 8px 0;border-top-color:initial;transform-origin:center top}.tippy-box[data-placement^=bottom]>.tippy-arrow{top:0}.tippy-box[data-placement^=bottom]>.tippy-arrow:before{top:-7px;left:0;border-width:0 8px 8px;border-bottom-color:initial;transform-origin:center bottom}.tippy-box[data-placement^=left]>.tippy-arrow{right:0}.tippy-box[data-placement^=left]>.tippy-arrow:before{border-width:8px 0 8px 8px;border-left-color:initial;right:-7px;transform-origin:center left}.tippy-box[data-placement^=right]>.tippy-arrow{left:0}.tippy-box[data-placement^=right]>.tippy-arrow:before{left:-7px;border-width:8px 8px 8px 0;border-right-color:initial;transform-origin:center right}.tippy-box[data-inertia][data-state=visible]{transition-timing-function:cubic-bezier(.54,1.5,.38,1.11)}.tippy-arrow{width:16px;height:16px;color:#333}.tippy-arrow:before{content:"";position:absolute;border-color:transparent;border-style:solid}.tippy-content{position:relative;padding:5px 9px;z-index:1} \ No newline at end of file diff --git a/site_libs/quarto-html/tippy.umd.min.js b/site_libs/quarto-html/tippy.umd.min.js new file mode 100644 index 00000000..ca292be3 --- /dev/null +++ b/site_libs/quarto-html/tippy.umd.min.js @@ -0,0 +1,2 @@ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t(require("@popperjs/core")):"function"==typeof define&&define.amd?define(["@popperjs/core"],t):(e=e||self).tippy=t(e.Popper)}(this,(function(e){"use strict";var t={passive:!0,capture:!0},n=function(){return document.body};function r(e,t,n){if(Array.isArray(e)){var r=e[t];return null==r?Array.isArray(n)?n[t]:n:r}return e}function o(e,t){var n={}.toString.call(e);return 0===n.indexOf("[object")&&n.indexOf(t+"]")>-1}function i(e,t){return"function"==typeof e?e.apply(void 0,t):e}function a(e,t){return 0===t?e:function(r){clearTimeout(n),n=setTimeout((function(){e(r)}),t)};var n}function s(e,t){var n=Object.assign({},e);return t.forEach((function(e){delete n[e]})),n}function u(e){return[].concat(e)}function c(e,t){-1===e.indexOf(t)&&e.push(t)}function p(e){return e.split("-")[0]}function f(e){return[].slice.call(e)}function l(e){return Object.keys(e).reduce((function(t,n){return void 0!==e[n]&&(t[n]=e[n]),t}),{})}function d(){return document.createElement("div")}function v(e){return["Element","Fragment"].some((function(t){return o(e,t)}))}function m(e){return o(e,"MouseEvent")}function g(e){return!(!e||!e._tippy||e._tippy.reference!==e)}function h(e){return v(e)?[e]:function(e){return o(e,"NodeList")}(e)?f(e):Array.isArray(e)?e:f(document.querySelectorAll(e))}function b(e,t){e.forEach((function(e){e&&(e.style.transitionDuration=t+"ms")}))}function y(e,t){e.forEach((function(e){e&&e.setAttribute("data-state",t)}))}function w(e){var t,n=u(e)[0];return null!=n&&null!=(t=n.ownerDocument)&&t.body?n.ownerDocument:document}function E(e,t,n){var r=t+"EventListener";["transitionend","webkitTransitionEnd"].forEach((function(t){e[r](t,n)}))}function O(e,t){for(var n=t;n;){var r;if(e.contains(n))return!0;n=null==n.getRootNode||null==(r=n.getRootNode())?void 0:r.host}return!1}var x={isTouch:!1},C=0;function T(){x.isTouch||(x.isTouch=!0,window.performance&&document.addEventListener("mousemove",A))}function A(){var e=performance.now();e-C<20&&(x.isTouch=!1,document.removeEventListener("mousemove",A)),C=e}function L(){var e=document.activeElement;if(g(e)){var t=e._tippy;e.blur&&!t.state.isVisible&&e.blur()}}var D=!!("undefined"!=typeof window&&"undefined"!=typeof document)&&!!window.msCrypto,R=Object.assign({appendTo:n,aria:{content:"auto",expanded:"auto"},delay:0,duration:[300,250],getReferenceClientRect:null,hideOnClick:!0,ignoreAttributes:!1,interactive:!1,interactiveBorder:2,interactiveDebounce:0,moveTransition:"",offset:[0,10],onAfterUpdate:function(){},onBeforeUpdate:function(){},onCreate:function(){},onDestroy:function(){},onHidden:function(){},onHide:function(){},onMount:function(){},onShow:function(){},onShown:function(){},onTrigger:function(){},onUntrigger:function(){},onClickOutside:function(){},placement:"top",plugins:[],popperOptions:{},render:null,showOnCreate:!1,touch:!0,trigger:"mouseenter focus",triggerTarget:null},{animateFill:!1,followCursor:!1,inlinePositioning:!1,sticky:!1},{allowHTML:!1,animation:"fade",arrow:!0,content:"",inertia:!1,maxWidth:350,role:"tooltip",theme:"",zIndex:9999}),k=Object.keys(R);function P(e){var t=(e.plugins||[]).reduce((function(t,n){var r,o=n.name,i=n.defaultValue;o&&(t[o]=void 0!==e[o]?e[o]:null!=(r=R[o])?r:i);return t}),{});return Object.assign({},e,t)}function j(e,t){var n=Object.assign({},t,{content:i(t.content,[e])},t.ignoreAttributes?{}:function(e,t){return(t?Object.keys(P(Object.assign({},R,{plugins:t}))):k).reduce((function(t,n){var r=(e.getAttribute("data-tippy-"+n)||"").trim();if(!r)return t;if("content"===n)t[n]=r;else try{t[n]=JSON.parse(r)}catch(e){t[n]=r}return t}),{})}(e,t.plugins));return n.aria=Object.assign({},R.aria,n.aria),n.aria={expanded:"auto"===n.aria.expanded?t.interactive:n.aria.expanded,content:"auto"===n.aria.content?t.interactive?null:"describedby":n.aria.content},n}function M(e,t){e.innerHTML=t}function V(e){var t=d();return!0===e?t.className="tippy-arrow":(t.className="tippy-svg-arrow",v(e)?t.appendChild(e):M(t,e)),t}function I(e,t){v(t.content)?(M(e,""),e.appendChild(t.content)):"function"!=typeof t.content&&(t.allowHTML?M(e,t.content):e.textContent=t.content)}function S(e){var t=e.firstElementChild,n=f(t.children);return{box:t,content:n.find((function(e){return e.classList.contains("tippy-content")})),arrow:n.find((function(e){return e.classList.contains("tippy-arrow")||e.classList.contains("tippy-svg-arrow")})),backdrop:n.find((function(e){return e.classList.contains("tippy-backdrop")}))}}function N(e){var t=d(),n=d();n.className="tippy-box",n.setAttribute("data-state","hidden"),n.setAttribute("tabindex","-1");var r=d();function o(n,r){var o=S(t),i=o.box,a=o.content,s=o.arrow;r.theme?i.setAttribute("data-theme",r.theme):i.removeAttribute("data-theme"),"string"==typeof r.animation?i.setAttribute("data-animation",r.animation):i.removeAttribute("data-animation"),r.inertia?i.setAttribute("data-inertia",""):i.removeAttribute("data-inertia"),i.style.maxWidth="number"==typeof r.maxWidth?r.maxWidth+"px":r.maxWidth,r.role?i.setAttribute("role",r.role):i.removeAttribute("role"),n.content===r.content&&n.allowHTML===r.allowHTML||I(a,e.props),r.arrow?s?n.arrow!==r.arrow&&(i.removeChild(s),i.appendChild(V(r.arrow))):i.appendChild(V(r.arrow)):s&&i.removeChild(s)}return r.className="tippy-content",r.setAttribute("data-state","hidden"),I(r,e.props),t.appendChild(n),n.appendChild(r),o(e.props,e.props),{popper:t,onUpdate:o}}N.$$tippy=!0;var B=1,H=[],U=[];function _(o,s){var v,g,h,C,T,A,L,k,M=j(o,Object.assign({},R,P(l(s)))),V=!1,I=!1,N=!1,_=!1,F=[],W=a(we,M.interactiveDebounce),X=B++,Y=(k=M.plugins).filter((function(e,t){return k.indexOf(e)===t})),$={id:X,reference:o,popper:d(),popperInstance:null,props:M,state:{isEnabled:!0,isVisible:!1,isDestroyed:!1,isMounted:!1,isShown:!1},plugins:Y,clearDelayTimeouts:function(){clearTimeout(v),clearTimeout(g),cancelAnimationFrame(h)},setProps:function(e){if($.state.isDestroyed)return;ae("onBeforeUpdate",[$,e]),be();var t=$.props,n=j(o,Object.assign({},t,l(e),{ignoreAttributes:!0}));$.props=n,he(),t.interactiveDebounce!==n.interactiveDebounce&&(ce(),W=a(we,n.interactiveDebounce));t.triggerTarget&&!n.triggerTarget?u(t.triggerTarget).forEach((function(e){e.removeAttribute("aria-expanded")})):n.triggerTarget&&o.removeAttribute("aria-expanded");ue(),ie(),J&&J(t,n);$.popperInstance&&(Ce(),Ae().forEach((function(e){requestAnimationFrame(e._tippy.popperInstance.forceUpdate)})));ae("onAfterUpdate",[$,e])},setContent:function(e){$.setProps({content:e})},show:function(){var e=$.state.isVisible,t=$.state.isDestroyed,o=!$.state.isEnabled,a=x.isTouch&&!$.props.touch,s=r($.props.duration,0,R.duration);if(e||t||o||a)return;if(te().hasAttribute("disabled"))return;if(ae("onShow",[$],!1),!1===$.props.onShow($))return;$.state.isVisible=!0,ee()&&(z.style.visibility="visible");ie(),de(),$.state.isMounted||(z.style.transition="none");if(ee()){var u=re(),p=u.box,f=u.content;b([p,f],0)}A=function(){var e;if($.state.isVisible&&!_){if(_=!0,z.offsetHeight,z.style.transition=$.props.moveTransition,ee()&&$.props.animation){var t=re(),n=t.box,r=t.content;b([n,r],s),y([n,r],"visible")}se(),ue(),c(U,$),null==(e=$.popperInstance)||e.forceUpdate(),ae("onMount",[$]),$.props.animation&&ee()&&function(e,t){me(e,t)}(s,(function(){$.state.isShown=!0,ae("onShown",[$])}))}},function(){var e,t=$.props.appendTo,r=te();e=$.props.interactive&&t===n||"parent"===t?r.parentNode:i(t,[r]);e.contains(z)||e.appendChild(z);$.state.isMounted=!0,Ce()}()},hide:function(){var e=!$.state.isVisible,t=$.state.isDestroyed,n=!$.state.isEnabled,o=r($.props.duration,1,R.duration);if(e||t||n)return;if(ae("onHide",[$],!1),!1===$.props.onHide($))return;$.state.isVisible=!1,$.state.isShown=!1,_=!1,V=!1,ee()&&(z.style.visibility="hidden");if(ce(),ve(),ie(!0),ee()){var i=re(),a=i.box,s=i.content;$.props.animation&&(b([a,s],o),y([a,s],"hidden"))}se(),ue(),$.props.animation?ee()&&function(e,t){me(e,(function(){!$.state.isVisible&&z.parentNode&&z.parentNode.contains(z)&&t()}))}(o,$.unmount):$.unmount()},hideWithInteractivity:function(e){ne().addEventListener("mousemove",W),c(H,W),W(e)},enable:function(){$.state.isEnabled=!0},disable:function(){$.hide(),$.state.isEnabled=!1},unmount:function(){$.state.isVisible&&$.hide();if(!$.state.isMounted)return;Te(),Ae().forEach((function(e){e._tippy.unmount()})),z.parentNode&&z.parentNode.removeChild(z);U=U.filter((function(e){return e!==$})),$.state.isMounted=!1,ae("onHidden",[$])},destroy:function(){if($.state.isDestroyed)return;$.clearDelayTimeouts(),$.unmount(),be(),delete o._tippy,$.state.isDestroyed=!0,ae("onDestroy",[$])}};if(!M.render)return $;var q=M.render($),z=q.popper,J=q.onUpdate;z.setAttribute("data-tippy-root",""),z.id="tippy-"+$.id,$.popper=z,o._tippy=$,z._tippy=$;var G=Y.map((function(e){return e.fn($)})),K=o.hasAttribute("aria-expanded");return he(),ue(),ie(),ae("onCreate",[$]),M.showOnCreate&&Le(),z.addEventListener("mouseenter",(function(){$.props.interactive&&$.state.isVisible&&$.clearDelayTimeouts()})),z.addEventListener("mouseleave",(function(){$.props.interactive&&$.props.trigger.indexOf("mouseenter")>=0&&ne().addEventListener("mousemove",W)})),$;function Q(){var e=$.props.touch;return Array.isArray(e)?e:[e,0]}function Z(){return"hold"===Q()[0]}function ee(){var e;return!(null==(e=$.props.render)||!e.$$tippy)}function te(){return L||o}function ne(){var e=te().parentNode;return e?w(e):document}function re(){return S(z)}function oe(e){return $.state.isMounted&&!$.state.isVisible||x.isTouch||C&&"focus"===C.type?0:r($.props.delay,e?0:1,R.delay)}function ie(e){void 0===e&&(e=!1),z.style.pointerEvents=$.props.interactive&&!e?"":"none",z.style.zIndex=""+$.props.zIndex}function ae(e,t,n){var r;(void 0===n&&(n=!0),G.forEach((function(n){n[e]&&n[e].apply(n,t)})),n)&&(r=$.props)[e].apply(r,t)}function se(){var e=$.props.aria;if(e.content){var t="aria-"+e.content,n=z.id;u($.props.triggerTarget||o).forEach((function(e){var r=e.getAttribute(t);if($.state.isVisible)e.setAttribute(t,r?r+" "+n:n);else{var o=r&&r.replace(n,"").trim();o?e.setAttribute(t,o):e.removeAttribute(t)}}))}}function ue(){!K&&$.props.aria.expanded&&u($.props.triggerTarget||o).forEach((function(e){$.props.interactive?e.setAttribute("aria-expanded",$.state.isVisible&&e===te()?"true":"false"):e.removeAttribute("aria-expanded")}))}function ce(){ne().removeEventListener("mousemove",W),H=H.filter((function(e){return e!==W}))}function pe(e){if(!x.isTouch||!N&&"mousedown"!==e.type){var t=e.composedPath&&e.composedPath()[0]||e.target;if(!$.props.interactive||!O(z,t)){if(u($.props.triggerTarget||o).some((function(e){return O(e,t)}))){if(x.isTouch)return;if($.state.isVisible&&$.props.trigger.indexOf("click")>=0)return}else ae("onClickOutside",[$,e]);!0===$.props.hideOnClick&&($.clearDelayTimeouts(),$.hide(),I=!0,setTimeout((function(){I=!1})),$.state.isMounted||ve())}}}function fe(){N=!0}function le(){N=!1}function de(){var e=ne();e.addEventListener("mousedown",pe,!0),e.addEventListener("touchend",pe,t),e.addEventListener("touchstart",le,t),e.addEventListener("touchmove",fe,t)}function ve(){var e=ne();e.removeEventListener("mousedown",pe,!0),e.removeEventListener("touchend",pe,t),e.removeEventListener("touchstart",le,t),e.removeEventListener("touchmove",fe,t)}function me(e,t){var n=re().box;function r(e){e.target===n&&(E(n,"remove",r),t())}if(0===e)return t();E(n,"remove",T),E(n,"add",r),T=r}function ge(e,t,n){void 0===n&&(n=!1),u($.props.triggerTarget||o).forEach((function(r){r.addEventListener(e,t,n),F.push({node:r,eventType:e,handler:t,options:n})}))}function he(){var e;Z()&&(ge("touchstart",ye,{passive:!0}),ge("touchend",Ee,{passive:!0})),(e=$.props.trigger,e.split(/\s+/).filter(Boolean)).forEach((function(e){if("manual"!==e)switch(ge(e,ye),e){case"mouseenter":ge("mouseleave",Ee);break;case"focus":ge(D?"focusout":"blur",Oe);break;case"focusin":ge("focusout",Oe)}}))}function be(){F.forEach((function(e){var t=e.node,n=e.eventType,r=e.handler,o=e.options;t.removeEventListener(n,r,o)})),F=[]}function ye(e){var t,n=!1;if($.state.isEnabled&&!xe(e)&&!I){var r="focus"===(null==(t=C)?void 0:t.type);C=e,L=e.currentTarget,ue(),!$.state.isVisible&&m(e)&&H.forEach((function(t){return t(e)})),"click"===e.type&&($.props.trigger.indexOf("mouseenter")<0||V)&&!1!==$.props.hideOnClick&&$.state.isVisible?n=!0:Le(e),"click"===e.type&&(V=!n),n&&!r&&De(e)}}function we(e){var t=e.target,n=te().contains(t)||z.contains(t);"mousemove"===e.type&&n||function(e,t){var n=t.clientX,r=t.clientY;return e.every((function(e){var t=e.popperRect,o=e.popperState,i=e.props.interactiveBorder,a=p(o.placement),s=o.modifiersData.offset;if(!s)return!0;var u="bottom"===a?s.top.y:0,c="top"===a?s.bottom.y:0,f="right"===a?s.left.x:0,l="left"===a?s.right.x:0,d=t.top-r+u>i,v=r-t.bottom-c>i,m=t.left-n+f>i,g=n-t.right-l>i;return d||v||m||g}))}(Ae().concat(z).map((function(e){var t,n=null==(t=e._tippy.popperInstance)?void 0:t.state;return n?{popperRect:e.getBoundingClientRect(),popperState:n,props:M}:null})).filter(Boolean),e)&&(ce(),De(e))}function Ee(e){xe(e)||$.props.trigger.indexOf("click")>=0&&V||($.props.interactive?$.hideWithInteractivity(e):De(e))}function Oe(e){$.props.trigger.indexOf("focusin")<0&&e.target!==te()||$.props.interactive&&e.relatedTarget&&z.contains(e.relatedTarget)||De(e)}function xe(e){return!!x.isTouch&&Z()!==e.type.indexOf("touch")>=0}function Ce(){Te();var t=$.props,n=t.popperOptions,r=t.placement,i=t.offset,a=t.getReferenceClientRect,s=t.moveTransition,u=ee()?S(z).arrow:null,c=a?{getBoundingClientRect:a,contextElement:a.contextElement||te()}:o,p=[{name:"offset",options:{offset:i}},{name:"preventOverflow",options:{padding:{top:2,bottom:2,left:5,right:5}}},{name:"flip",options:{padding:5}},{name:"computeStyles",options:{adaptive:!s}},{name:"$$tippy",enabled:!0,phase:"beforeWrite",requires:["computeStyles"],fn:function(e){var t=e.state;if(ee()){var n=re().box;["placement","reference-hidden","escaped"].forEach((function(e){"placement"===e?n.setAttribute("data-placement",t.placement):t.attributes.popper["data-popper-"+e]?n.setAttribute("data-"+e,""):n.removeAttribute("data-"+e)})),t.attributes.popper={}}}}];ee()&&u&&p.push({name:"arrow",options:{element:u,padding:3}}),p.push.apply(p,(null==n?void 0:n.modifiers)||[]),$.popperInstance=e.createPopper(c,z,Object.assign({},n,{placement:r,onFirstUpdate:A,modifiers:p}))}function Te(){$.popperInstance&&($.popperInstance.destroy(),$.popperInstance=null)}function Ae(){return f(z.querySelectorAll("[data-tippy-root]"))}function Le(e){$.clearDelayTimeouts(),e&&ae("onTrigger",[$,e]),de();var t=oe(!0),n=Q(),r=n[0],o=n[1];x.isTouch&&"hold"===r&&o&&(t=o),t?v=setTimeout((function(){$.show()}),t):$.show()}function De(e){if($.clearDelayTimeouts(),ae("onUntrigger",[$,e]),$.state.isVisible){if(!($.props.trigger.indexOf("mouseenter")>=0&&$.props.trigger.indexOf("click")>=0&&["mouseleave","mousemove"].indexOf(e.type)>=0&&V)){var t=oe(!1);t?g=setTimeout((function(){$.state.isVisible&&$.hide()}),t):h=requestAnimationFrame((function(){$.hide()}))}}else ve()}}function F(e,n){void 0===n&&(n={});var r=R.plugins.concat(n.plugins||[]);document.addEventListener("touchstart",T,t),window.addEventListener("blur",L);var o=Object.assign({},n,{plugins:r}),i=h(e).reduce((function(e,t){var n=t&&_(t,o);return n&&e.push(n),e}),[]);return v(e)?i[0]:i}F.defaultProps=R,F.setDefaultProps=function(e){Object.keys(e).forEach((function(t){R[t]=e[t]}))},F.currentInput=x;var W=Object.assign({},e.applyStyles,{effect:function(e){var t=e.state,n={popper:{position:t.options.strategy,left:"0",top:"0",margin:"0"},arrow:{position:"absolute"},reference:{}};Object.assign(t.elements.popper.style,n.popper),t.styles=n,t.elements.arrow&&Object.assign(t.elements.arrow.style,n.arrow)}}),X={mouseover:"mouseenter",focusin:"focus",click:"click"};var Y={name:"animateFill",defaultValue:!1,fn:function(e){var t;if(null==(t=e.props.render)||!t.$$tippy)return{};var n=S(e.popper),r=n.box,o=n.content,i=e.props.animateFill?function(){var e=d();return e.className="tippy-backdrop",y([e],"hidden"),e}():null;return{onCreate:function(){i&&(r.insertBefore(i,r.firstElementChild),r.setAttribute("data-animatefill",""),r.style.overflow="hidden",e.setProps({arrow:!1,animation:"shift-away"}))},onMount:function(){if(i){var e=r.style.transitionDuration,t=Number(e.replace("ms",""));o.style.transitionDelay=Math.round(t/10)+"ms",i.style.transitionDuration=e,y([i],"visible")}},onShow:function(){i&&(i.style.transitionDuration="0ms")},onHide:function(){i&&y([i],"hidden")}}}};var $={clientX:0,clientY:0},q=[];function z(e){var t=e.clientX,n=e.clientY;$={clientX:t,clientY:n}}var J={name:"followCursor",defaultValue:!1,fn:function(e){var t=e.reference,n=w(e.props.triggerTarget||t),r=!1,o=!1,i=!0,a=e.props;function s(){return"initial"===e.props.followCursor&&e.state.isVisible}function u(){n.addEventListener("mousemove",f)}function c(){n.removeEventListener("mousemove",f)}function p(){r=!0,e.setProps({getReferenceClientRect:null}),r=!1}function f(n){var r=!n.target||t.contains(n.target),o=e.props.followCursor,i=n.clientX,a=n.clientY,s=t.getBoundingClientRect(),u=i-s.left,c=a-s.top;!r&&e.props.interactive||e.setProps({getReferenceClientRect:function(){var e=t.getBoundingClientRect(),n=i,r=a;"initial"===o&&(n=e.left+u,r=e.top+c);var s="horizontal"===o?e.top:r,p="vertical"===o?e.right:n,f="horizontal"===o?e.bottom:r,l="vertical"===o?e.left:n;return{width:p-l,height:f-s,top:s,right:p,bottom:f,left:l}}})}function l(){e.props.followCursor&&(q.push({instance:e,doc:n}),function(e){e.addEventListener("mousemove",z)}(n))}function d(){0===(q=q.filter((function(t){return t.instance!==e}))).filter((function(e){return e.doc===n})).length&&function(e){e.removeEventListener("mousemove",z)}(n)}return{onCreate:l,onDestroy:d,onBeforeUpdate:function(){a=e.props},onAfterUpdate:function(t,n){var i=n.followCursor;r||void 0!==i&&a.followCursor!==i&&(d(),i?(l(),!e.state.isMounted||o||s()||u()):(c(),p()))},onMount:function(){e.props.followCursor&&!o&&(i&&(f($),i=!1),s()||u())},onTrigger:function(e,t){m(t)&&($={clientX:t.clientX,clientY:t.clientY}),o="focus"===t.type},onHidden:function(){e.props.followCursor&&(p(),c(),i=!0)}}}};var G={name:"inlinePositioning",defaultValue:!1,fn:function(e){var t,n=e.reference;var r=-1,o=!1,i=[],a={name:"tippyInlinePositioning",enabled:!0,phase:"afterWrite",fn:function(o){var a=o.state;e.props.inlinePositioning&&(-1!==i.indexOf(a.placement)&&(i=[]),t!==a.placement&&-1===i.indexOf(a.placement)&&(i.push(a.placement),e.setProps({getReferenceClientRect:function(){return function(e){return function(e,t,n,r){if(n.length<2||null===e)return t;if(2===n.length&&r>=0&&n[0].left>n[1].right)return n[r]||t;switch(e){case"top":case"bottom":var o=n[0],i=n[n.length-1],a="top"===e,s=o.top,u=i.bottom,c=a?o.left:i.left,p=a?o.right:i.right;return{top:s,bottom:u,left:c,right:p,width:p-c,height:u-s};case"left":case"right":var f=Math.min.apply(Math,n.map((function(e){return e.left}))),l=Math.max.apply(Math,n.map((function(e){return e.right}))),d=n.filter((function(t){return"left"===e?t.left===f:t.right===l})),v=d[0].top,m=d[d.length-1].bottom;return{top:v,bottom:m,left:f,right:l,width:l-f,height:m-v};default:return t}}(p(e),n.getBoundingClientRect(),f(n.getClientRects()),r)}(a.placement)}})),t=a.placement)}};function s(){var t;o||(t=function(e,t){var n;return{popperOptions:Object.assign({},e.popperOptions,{modifiers:[].concat(((null==(n=e.popperOptions)?void 0:n.modifiers)||[]).filter((function(e){return e.name!==t.name})),[t])})}}(e.props,a),o=!0,e.setProps(t),o=!1)}return{onCreate:s,onAfterUpdate:s,onTrigger:function(t,n){if(m(n)){var o=f(e.reference.getClientRects()),i=o.find((function(e){return e.left-2<=n.clientX&&e.right+2>=n.clientX&&e.top-2<=n.clientY&&e.bottom+2>=n.clientY})),a=o.indexOf(i);r=a>-1?a:r}},onHidden:function(){r=-1}}}};var K={name:"sticky",defaultValue:!1,fn:function(e){var t=e.reference,n=e.popper;function r(t){return!0===e.props.sticky||e.props.sticky===t}var o=null,i=null;function a(){var s=r("reference")?(e.popperInstance?e.popperInstance.state.elements.reference:t).getBoundingClientRect():null,u=r("popper")?n.getBoundingClientRect():null;(s&&Q(o,s)||u&&Q(i,u))&&e.popperInstance&&e.popperInstance.update(),o=s,i=u,e.state.isMounted&&requestAnimationFrame(a)}return{onMount:function(){e.props.sticky&&a()}}}};function Q(e,t){return!e||!t||(e.top!==t.top||e.right!==t.right||e.bottom!==t.bottom||e.left!==t.left)}return F.setDefaultProps({plugins:[Y,J,G,K],render:N}),F.createSingleton=function(e,t){var n;void 0===t&&(t={});var r,o=e,i=[],a=[],c=t.overrides,p=[],f=!1;function l(){a=o.map((function(e){return u(e.props.triggerTarget||e.reference)})).reduce((function(e,t){return e.concat(t)}),[])}function v(){i=o.map((function(e){return e.reference}))}function m(e){o.forEach((function(t){e?t.enable():t.disable()}))}function g(e){return o.map((function(t){var n=t.setProps;return t.setProps=function(o){n(o),t.reference===r&&e.setProps(o)},function(){t.setProps=n}}))}function h(e,t){var n=a.indexOf(t);if(t!==r){r=t;var s=(c||[]).concat("content").reduce((function(e,t){return e[t]=o[n].props[t],e}),{});e.setProps(Object.assign({},s,{getReferenceClientRect:"function"==typeof s.getReferenceClientRect?s.getReferenceClientRect:function(){var e;return null==(e=i[n])?void 0:e.getBoundingClientRect()}}))}}m(!1),v(),l();var b={fn:function(){return{onDestroy:function(){m(!0)},onHidden:function(){r=null},onClickOutside:function(e){e.props.showOnCreate&&!f&&(f=!0,r=null)},onShow:function(e){e.props.showOnCreate&&!f&&(f=!0,h(e,i[0]))},onTrigger:function(e,t){h(e,t.currentTarget)}}}},y=F(d(),Object.assign({},s(t,["overrides"]),{plugins:[b].concat(t.plugins||[]),triggerTarget:a,popperOptions:Object.assign({},t.popperOptions,{modifiers:[].concat((null==(n=t.popperOptions)?void 0:n.modifiers)||[],[W])})})),w=y.show;y.show=function(e){if(w(),!r&&null==e)return h(y,i[0]);if(!r||null!=e){if("number"==typeof e)return i[e]&&h(y,i[e]);if(o.indexOf(e)>=0){var t=e.reference;return h(y,t)}return i.indexOf(e)>=0?h(y,e):void 0}},y.showNext=function(){var e=i[0];if(!r)return y.show(0);var t=i.indexOf(r);y.show(i[t+1]||e)},y.showPrevious=function(){var e=i[i.length-1];if(!r)return y.show(e);var t=i.indexOf(r),n=i[t-1]||e;y.show(n)};var E=y.setProps;return y.setProps=function(e){c=e.overrides||c,E(e)},y.setInstances=function(e){m(!0),p.forEach((function(e){return e()})),o=e,m(!1),v(),l(),p=g(y),y.setProps({triggerTarget:a})},p=g(y),y},F.delegate=function(e,n){var r=[],o=[],i=!1,a=n.target,c=s(n,["target"]),p=Object.assign({},c,{trigger:"manual",touch:!1}),f=Object.assign({touch:R.touch},c,{showOnCreate:!0}),l=F(e,p);function d(e){if(e.target&&!i){var t=e.target.closest(a);if(t){var r=t.getAttribute("data-tippy-trigger")||n.trigger||R.trigger;if(!t._tippy&&!("touchstart"===e.type&&"boolean"==typeof f.touch||"touchstart"!==e.type&&r.indexOf(X[e.type])<0)){var s=F(t,f);s&&(o=o.concat(s))}}}}function v(e,t,n,o){void 0===o&&(o=!1),e.addEventListener(t,n,o),r.push({node:e,eventType:t,handler:n,options:o})}return u(l).forEach((function(e){var n=e.destroy,a=e.enable,s=e.disable;e.destroy=function(e){void 0===e&&(e=!0),e&&o.forEach((function(e){e.destroy()})),o=[],r.forEach((function(e){var t=e.node,n=e.eventType,r=e.handler,o=e.options;t.removeEventListener(n,r,o)})),r=[],n()},e.enable=function(){a(),o.forEach((function(e){return e.enable()})),i=!1},e.disable=function(){s(),o.forEach((function(e){return e.disable()})),i=!0},function(e){var n=e.reference;v(n,"touchstart",d,t),v(n,"mouseover",d),v(n,"focusin",d),v(n,"click",d)}(e)})),l},F.hideAll=function(e){var t=void 0===e?{}:e,n=t.exclude,r=t.duration;U.forEach((function(e){var t=!1;if(n&&(t=g(n)?e.reference===n:e.popper===n.popper),!t){var o=e.props.duration;e.setProps({duration:r}),e.hide(),e.state.isDestroyed||e.setProps({duration:o})}}))},F.roundArrow='',F})); + diff --git a/site_libs/quarto-listing/list.min.js b/site_libs/quarto-listing/list.min.js new file mode 100644 index 00000000..43dfd15a --- /dev/null +++ b/site_libs/quarto-listing/list.min.js @@ -0,0 +1,2 @@ +var List;List=function(){var t={"./src/add-async.js":function(t){t.exports=function(t){return function e(r,n,s){var i=r.splice(0,50);s=(s=s||[]).concat(t.add(i)),r.length>0?setTimeout((function(){e(r,n,s)}),1):(t.update(),n(s))}}},"./src/filter.js":function(t){t.exports=function(t){return t.handlers.filterStart=t.handlers.filterStart||[],t.handlers.filterComplete=t.handlers.filterComplete||[],function(e){if(t.trigger("filterStart"),t.i=1,t.reset.filter(),void 0===e)t.filtered=!1;else{t.filtered=!0;for(var r=t.items,n=0,s=r.length;nv.page,a=new g(t[s],void 0,n),v.items.push(a),r.push(a)}return v.update(),r}m(t.slice(0),e)}},this.show=function(t,e){return this.i=t,this.page=e,v.update(),v},this.remove=function(t,e,r){for(var n=0,s=0,i=v.items.length;s-1&&r.splice(n,1),v},this.trigger=function(t){for(var e=v.handlers[t].length;e--;)v.handlers[t][e](v);return v},this.reset={filter:function(){for(var t=v.items,e=t.length;e--;)t[e].filtered=!1;return v},search:function(){for(var t=v.items,e=t.length;e--;)t[e].found=!1;return v}},this.update=function(){var t=v.items,e=t.length;v.visibleItems=[],v.matchingItems=[],v.templater.clear();for(var r=0;r=v.i&&v.visibleItems.lengthe},innerWindow:function(t,e,r){return t>=e-r&&t<=e+r},dotted:function(t,e,r,n,s,i,a){return this.dottedLeft(t,e,r,n,s,i)||this.dottedRight(t,e,r,n,s,i,a)},dottedLeft:function(t,e,r,n,s,i){return e==r+1&&!this.innerWindow(e,s,i)&&!this.right(e,n)},dottedRight:function(t,e,r,n,s,i,a){return!t.items[a-1].values().dotted&&(e==n&&!this.innerWindow(e,s,i)&&!this.right(e,n))}};return function(e){var n=new i(t.listContainer.id,{listClass:e.paginationClass||"pagination",item:e.item||"",valueNames:["page","dotted"],searchClass:"pagination-search-that-is-not-supposed-to-exist",sortClass:"pagination-sort-that-is-not-supposed-to-exist"});s.bind(n.listContainer,"click",(function(e){var r=e.target||e.srcElement,n=t.utils.getAttribute(r,"data-page"),s=t.utils.getAttribute(r,"data-i");s&&t.show((s-1)*n+1,n)})),t.on("updated",(function(){r(n,e)})),r(n,e)}}},"./src/parse.js":function(t,e,r){t.exports=function(t){var e=r("./src/item.js")(t),n=function(r,n){for(var s=0,i=r.length;s0?setTimeout((function(){e(r,s)}),1):(t.update(),t.trigger("parseComplete"))};return t.handlers.parseComplete=t.handlers.parseComplete||[],function(){var e=function(t){for(var e=t.childNodes,r=[],n=0,s=e.length;n]/g.exec(t)){var e=document.createElement("tbody");return e.innerHTML=t,e.firstElementChild}if(-1!==t.indexOf("<")){var r=document.createElement("div");return r.innerHTML=t,r.firstElementChild}}},a=function(e,r,n){var s=void 0,i=function(e){for(var r=0,n=t.valueNames.length;r=1;)t.list.removeChild(t.list.firstChild)},function(){var r;if("function"!=typeof t.item){if(!(r="string"==typeof t.item?-1===t.item.indexOf("<")?document.getElementById(t.item):i(t.item):s()))throw new Error("The list needs to have at least one item on init otherwise you'll have to add a template.");r=n(r,t.valueNames),e=function(){return r.cloneNode(!0)}}else e=function(e){var r=t.item(e);return i(r)}}()};t.exports=function(t){return new e(t)}},"./src/utils/classes.js":function(t,e,r){var n=r("./src/utils/index-of.js"),s=/\s+/;Object.prototype.toString;function i(t){if(!t||!t.nodeType)throw new Error("A DOM element reference is required");this.el=t,this.list=t.classList}t.exports=function(t){return new i(t)},i.prototype.add=function(t){if(this.list)return this.list.add(t),this;var e=this.array();return~n(e,t)||e.push(t),this.el.className=e.join(" "),this},i.prototype.remove=function(t){if(this.list)return this.list.remove(t),this;var e=this.array(),r=n(e,t);return~r&&e.splice(r,1),this.el.className=e.join(" "),this},i.prototype.toggle=function(t,e){return this.list?(void 0!==e?e!==this.list.toggle(t,e)&&this.list.toggle(t):this.list.toggle(t),this):(void 0!==e?e?this.add(t):this.remove(t):this.has(t)?this.remove(t):this.add(t),this)},i.prototype.array=function(){var t=(this.el.getAttribute("class")||"").replace(/^\s+|\s+$/g,"").split(s);return""===t[0]&&t.shift(),t},i.prototype.has=i.prototype.contains=function(t){return this.list?this.list.contains(t):!!~n(this.array(),t)}},"./src/utils/events.js":function(t,e,r){var n=window.addEventListener?"addEventListener":"attachEvent",s=window.removeEventListener?"removeEventListener":"detachEvent",i="addEventListener"!==n?"on":"",a=r("./src/utils/to-array.js");e.bind=function(t,e,r,s){for(var o=0,l=(t=a(t)).length;o32)return!1;var a=n,o=function(){var t,r={};for(t=0;t=p;b--){var j=o[t.charAt(b-1)];if(C[b]=0===m?(C[b+1]<<1|1)&j:(C[b+1]<<1|1)&j|(v[b+1]|v[b])<<1|1|v[b+1],C[b]&d){var x=l(m,b-1);if(x<=u){if(u=x,!((c=b-1)>a))break;p=Math.max(1,2*a-c)}}}if(l(m+1,a)>u)break;v=C}return!(c<0)}},"./src/utils/get-attribute.js":function(t){t.exports=function(t,e){var r=t.getAttribute&&t.getAttribute(e)||null;if(!r)for(var n=t.attributes,s=n.length,i=0;i=48&&t<=57}function i(t,e){for(var i=(t+="").length,a=(e+="").length,o=0,l=0;o=i&&l=a?-1:l>=a&&o=i?1:i-a}i.caseInsensitive=i.i=function(t,e){return i((""+t).toLowerCase(),(""+e).toLowerCase())},Object.defineProperties(i,{alphabet:{get:function(){return e},set:function(t){r=[];var s=0;if(e=t)for(;s { + // category is URI encoded in EJS template for UTF-8 support + category = decodeURIComponent(atob(category)); + if (categoriesLoaded) { + activateCategory(category); + setCategoryHash(category); + } +}; + +window["quarto-listing-loaded"] = () => { + // Process any existing hash + const hash = getHash(); + + if (hash) { + // If there is a category, switch to that + if (hash.category) { + // category hash are URI encoded so we need to decode it before processing + // so that we can match it with the category element processed in JS + activateCategory(decodeURIComponent(hash.category)); + } + // Paginate a specific listing + const listingIds = Object.keys(window["quarto-listings"]); + for (const listingId of listingIds) { + const page = hash[getListingPageKey(listingId)]; + if (page) { + showPage(listingId, page); + } + } + } + + const listingIds = Object.keys(window["quarto-listings"]); + for (const listingId of listingIds) { + // The actual list + const list = window["quarto-listings"][listingId]; + + // Update the handlers for pagination events + refreshPaginationHandlers(listingId); + + // Render any visible items that need it + renderVisibleProgressiveImages(list); + + // Whenever the list is updated, we also need to + // attach handlers to the new pagination elements + // and refresh any newly visible items. + list.on("updated", function () { + renderVisibleProgressiveImages(list); + setTimeout(() => refreshPaginationHandlers(listingId)); + + // Show or hide the no matching message + toggleNoMatchingMessage(list); + }); + } +}; + +window.document.addEventListener("DOMContentLoaded", function (_event) { + // Attach click handlers to categories + const categoryEls = window.document.querySelectorAll( + ".quarto-listing-category .category" + ); + + for (const categoryEl of categoryEls) { + // category needs to support non ASCII characters + const category = decodeURIComponent( + atob(categoryEl.getAttribute("data-category")) + ); + categoryEl.onclick = () => { + activateCategory(category); + setCategoryHash(category); + }; + } + + // Attach a click handler to the category title + // (there should be only one, but since it is a class name, handle N) + const categoryTitleEls = window.document.querySelectorAll( + ".quarto-listing-category-title" + ); + for (const categoryTitleEl of categoryTitleEls) { + categoryTitleEl.onclick = () => { + activateCategory(""); + setCategoryHash(""); + }; + } + + categoriesLoaded = true; +}); + +function toggleNoMatchingMessage(list) { + const selector = `#${list.listContainer.id} .listing-no-matching`; + const noMatchingEl = window.document.querySelector(selector); + if (noMatchingEl) { + if (list.visibleItems.length === 0) { + noMatchingEl.classList.remove("d-none"); + } else { + if (!noMatchingEl.classList.contains("d-none")) { + noMatchingEl.classList.add("d-none"); + } + } + } +} + +function setCategoryHash(category) { + setHash({ category }); +} + +function setPageHash(listingId, page) { + const currentHash = getHash() || {}; + currentHash[getListingPageKey(listingId)] = page; + setHash(currentHash); +} + +function getListingPageKey(listingId) { + return `${listingId}-page`; +} + +function refreshPaginationHandlers(listingId) { + const listingEl = window.document.getElementById(listingId); + const paginationEls = listingEl.querySelectorAll( + ".pagination li.page-item:not(.disabled) .page.page-link" + ); + for (const paginationEl of paginationEls) { + paginationEl.onclick = (sender) => { + setPageHash(listingId, sender.target.getAttribute("data-i")); + showPage(listingId, sender.target.getAttribute("data-i")); + return false; + }; + } +} + +function renderVisibleProgressiveImages(list) { + // Run through the visible items and render any progressive images + for (const item of list.visibleItems) { + const itemEl = item.elm; + if (itemEl) { + const progressiveImgs = itemEl.querySelectorAll( + `img[${kProgressiveAttr}]` + ); + for (const progressiveImg of progressiveImgs) { + const srcValue = progressiveImg.getAttribute(kProgressiveAttr); + if (srcValue) { + progressiveImg.setAttribute("src", srcValue); + } + progressiveImg.removeAttribute(kProgressiveAttr); + } + } + } +} + +function getHash() { + // Hashes are of the form + // #name:value|name1:value1|name2:value2 + const currentUrl = new URL(window.location); + const hashRaw = currentUrl.hash ? currentUrl.hash.slice(1) : undefined; + return parseHash(hashRaw); +} + +const kAnd = "&"; +const kEquals = "="; + +function parseHash(hash) { + if (!hash) { + return undefined; + } + const hasValuesStrs = hash.split(kAnd); + const hashValues = hasValuesStrs + .map((hashValueStr) => { + const vals = hashValueStr.split(kEquals); + if (vals.length === 2) { + return { name: vals[0], value: vals[1] }; + } else { + return undefined; + } + }) + .filter((value) => { + return value !== undefined; + }); + + const hashObj = {}; + hashValues.forEach((hashValue) => { + hashObj[hashValue.name] = decodeURIComponent(hashValue.value); + }); + return hashObj; +} + +function makeHash(obj) { + return Object.keys(obj) + .map((key) => { + return `${key}${kEquals}${obj[key]}`; + }) + .join(kAnd); +} + +function setHash(obj) { + const hash = makeHash(obj); + window.history.pushState(null, null, `#${hash}`); +} + +function showPage(listingId, page) { + const list = window["quarto-listings"][listingId]; + if (list) { + list.show((page - 1) * list.page + 1, list.page); + } +} + +function activateCategory(category) { + // Deactivate existing categories + const activeEls = window.document.querySelectorAll( + ".quarto-listing-category .category.active" + ); + for (const activeEl of activeEls) { + activeEl.classList.remove("active"); + } + + // Activate this category + const categoryEl = window.document.querySelector( + `.quarto-listing-category .category[data-category='${btoa( + encodeURIComponent(category) + )}']` + ); + if (categoryEl) { + categoryEl.classList.add("active"); + } + + // Filter the listings to this category + filterListingCategory(category); +} + +function filterListingCategory(category) { + const listingIds = Object.keys(window["quarto-listings"]); + for (const listingId of listingIds) { + const list = window["quarto-listings"][listingId]; + if (list) { + if (category === "") { + // resets the filter + list.filter(); + } else { + // filter to this category + list.filter(function (item) { + const itemValues = item.values(); + if (itemValues.categories !== null) { + const categories = decodeURIComponent( + atob(itemValues.categories) + ).split(","); + return categories.includes(category); + } else { + return false; + } + }); + } + } + } +} diff --git a/site_libs/quarto-nav/headroom.min.js b/site_libs/quarto-nav/headroom.min.js new file mode 100644 index 00000000..b08f1dff --- /dev/null +++ b/site_libs/quarto-nav/headroom.min.js @@ -0,0 +1,7 @@ +/*! + * headroom.js v0.12.0 - Give your page some headroom. Hide your header until you need it + * Copyright (c) 2020 Nick Williams - http://wicky.nillia.ms/headroom.js + * License: MIT + */ + +!function(t,n){"object"==typeof exports&&"undefined"!=typeof module?module.exports=n():"function"==typeof define&&define.amd?define(n):(t=t||self).Headroom=n()}(this,function(){"use strict";function t(){return"undefined"!=typeof window}function d(t){return function(t){return t&&t.document&&function(t){return 9===t.nodeType}(t.document)}(t)?function(t){var n=t.document,o=n.body,s=n.documentElement;return{scrollHeight:function(){return Math.max(o.scrollHeight,s.scrollHeight,o.offsetHeight,s.offsetHeight,o.clientHeight,s.clientHeight)},height:function(){return t.innerHeight||s.clientHeight||o.clientHeight},scrollY:function(){return void 0!==t.pageYOffset?t.pageYOffset:(s||o.parentNode||o).scrollTop}}}(t):function(t){return{scrollHeight:function(){return Math.max(t.scrollHeight,t.offsetHeight,t.clientHeight)},height:function(){return Math.max(t.offsetHeight,t.clientHeight)},scrollY:function(){return t.scrollTop}}}(t)}function n(t,s,e){var n,o=function(){var n=!1;try{var t={get passive(){n=!0}};window.addEventListener("test",t,t),window.removeEventListener("test",t,t)}catch(t){n=!1}return n}(),i=!1,r=d(t),l=r.scrollY(),a={};function c(){var t=Math.round(r.scrollY()),n=r.height(),o=r.scrollHeight();a.scrollY=t,a.lastScrollY=l,a.direction=ls.tolerance[a.direction],e(a),l=t,i=!1}function h(){i||(i=!0,n=requestAnimationFrame(c))}var u=!!o&&{passive:!0,capture:!1};return t.addEventListener("scroll",h,u),c(),{destroy:function(){cancelAnimationFrame(n),t.removeEventListener("scroll",h,u)}}}function o(t){return t===Object(t)?t:{down:t,up:t}}function s(t,n){n=n||{},Object.assign(this,s.options,n),this.classes=Object.assign({},s.options.classes,n.classes),this.elem=t,this.tolerance=o(this.tolerance),this.offset=o(this.offset),this.initialised=!1,this.frozen=!1}return s.prototype={constructor:s,init:function(){return s.cutsTheMustard&&!this.initialised&&(this.addClass("initial"),this.initialised=!0,setTimeout(function(t){t.scrollTracker=n(t.scroller,{offset:t.offset,tolerance:t.tolerance},t.update.bind(t))},100,this)),this},destroy:function(){this.initialised=!1,Object.keys(this.classes).forEach(this.removeClass,this),this.scrollTracker.destroy()},unpin:function(){!this.hasClass("pinned")&&this.hasClass("unpinned")||(this.addClass("unpinned"),this.removeClass("pinned"),this.onUnpin&&this.onUnpin.call(this))},pin:function(){this.hasClass("unpinned")&&(this.addClass("pinned"),this.removeClass("unpinned"),this.onPin&&this.onPin.call(this))},freeze:function(){this.frozen=!0,this.addClass("frozen")},unfreeze:function(){this.frozen=!1,this.removeClass("frozen")},top:function(){this.hasClass("top")||(this.addClass("top"),this.removeClass("notTop"),this.onTop&&this.onTop.call(this))},notTop:function(){this.hasClass("notTop")||(this.addClass("notTop"),this.removeClass("top"),this.onNotTop&&this.onNotTop.call(this))},bottom:function(){this.hasClass("bottom")||(this.addClass("bottom"),this.removeClass("notBottom"),this.onBottom&&this.onBottom.call(this))},notBottom:function(){this.hasClass("notBottom")||(this.addClass("notBottom"),this.removeClass("bottom"),this.onNotBottom&&this.onNotBottom.call(this))},shouldUnpin:function(t){return"down"===t.direction&&!t.top&&t.toleranceExceeded},shouldPin:function(t){return"up"===t.direction&&t.toleranceExceeded||t.top},addClass:function(t){this.elem.classList.add.apply(this.elem.classList,this.classes[t].split(" "))},removeClass:function(t){this.elem.classList.remove.apply(this.elem.classList,this.classes[t].split(" "))},hasClass:function(t){return this.classes[t].split(" ").every(function(t){return this.classList.contains(t)},this.elem)},update:function(t){t.isOutOfBounds||!0!==this.frozen&&(t.top?this.top():this.notTop(),t.bottom?this.bottom():this.notBottom(),this.shouldUnpin(t)?this.unpin():this.shouldPin(t)&&this.pin())}},s.options={tolerance:{up:0,down:0},offset:0,scroller:t()?window:null,classes:{frozen:"headroom--frozen",pinned:"headroom--pinned",unpinned:"headroom--unpinned",top:"headroom--top",notTop:"headroom--not-top",bottom:"headroom--bottom",notBottom:"headroom--not-bottom",initial:"headroom"}},s.cutsTheMustard=!!(t()&&function(){}.bind&&"classList"in document.documentElement&&Object.assign&&Object.keys&&requestAnimationFrame),s}); diff --git a/site_libs/quarto-nav/quarto-nav.js b/site_libs/quarto-nav/quarto-nav.js new file mode 100644 index 00000000..38cc4305 --- /dev/null +++ b/site_libs/quarto-nav/quarto-nav.js @@ -0,0 +1,325 @@ +const headroomChanged = new CustomEvent("quarto-hrChanged", { + detail: {}, + bubbles: true, + cancelable: false, + composed: false, +}); + +const announceDismiss = () => { + const annEl = window.document.getElementById("quarto-announcement"); + if (annEl) { + annEl.remove(); + + const annId = annEl.getAttribute("data-announcement-id"); + window.localStorage.setItem(`quarto-announce-${annId}`, "true"); + } +}; + +const announceRegister = () => { + const annEl = window.document.getElementById("quarto-announcement"); + if (annEl) { + const annId = annEl.getAttribute("data-announcement-id"); + const isDismissed = + window.localStorage.getItem(`quarto-announce-${annId}`) || false; + if (isDismissed) { + announceDismiss(); + return; + } else { + annEl.classList.remove("hidden"); + } + + const actionEl = annEl.querySelector(".quarto-announcement-action"); + if (actionEl) { + actionEl.addEventListener("click", function (e) { + e.preventDefault(); + // Hide the bar immediately + announceDismiss(); + }); + } + } +}; + +window.document.addEventListener("DOMContentLoaded", function () { + let init = false; + + announceRegister(); + + // Manage the back to top button, if one is present. + let lastScrollTop = window.pageYOffset || document.documentElement.scrollTop; + const scrollDownBuffer = 5; + const scrollUpBuffer = 35; + const btn = document.getElementById("quarto-back-to-top"); + const hideBackToTop = () => { + btn.style.display = "none"; + }; + const showBackToTop = () => { + btn.style.display = "inline-block"; + }; + if (btn) { + window.document.addEventListener( + "scroll", + function () { + const currentScrollTop = + window.pageYOffset || document.documentElement.scrollTop; + + // Shows and hides the button 'intelligently' as the user scrolls + if (currentScrollTop - scrollDownBuffer > lastScrollTop) { + hideBackToTop(); + lastScrollTop = currentScrollTop <= 0 ? 0 : currentScrollTop; + } else if (currentScrollTop < lastScrollTop - scrollUpBuffer) { + showBackToTop(); + lastScrollTop = currentScrollTop <= 0 ? 0 : currentScrollTop; + } + + // Show the button at the bottom, hides it at the top + if (currentScrollTop <= 0) { + hideBackToTop(); + } else if ( + window.innerHeight + currentScrollTop >= + document.body.offsetHeight + ) { + showBackToTop(); + } + }, + false + ); + } + + function throttle(func, wait) { + var timeout; + return function () { + const context = this; + const args = arguments; + const later = function () { + clearTimeout(timeout); + timeout = null; + func.apply(context, args); + }; + + if (!timeout) { + timeout = setTimeout(later, wait); + } + }; + } + + function headerOffset() { + // Set an offset if there is are fixed top navbar + const headerEl = window.document.querySelector("header.fixed-top"); + if (headerEl) { + return headerEl.clientHeight; + } else { + return 0; + } + } + + function footerOffset() { + const footerEl = window.document.querySelector("footer.footer"); + if (footerEl) { + return footerEl.clientHeight; + } else { + return 0; + } + } + + function dashboardOffset() { + const dashboardNavEl = window.document.getElementById( + "quarto-dashboard-header" + ); + if (dashboardNavEl !== null) { + return dashboardNavEl.clientHeight; + } else { + return 0; + } + } + + function updateDocumentOffsetWithoutAnimation() { + updateDocumentOffset(false); + } + + function updateDocumentOffset(animated) { + // set body offset + const topOffset = headerOffset(); + const bodyOffset = topOffset + footerOffset() + dashboardOffset(); + const bodyEl = window.document.body; + bodyEl.setAttribute("data-bs-offset", topOffset); + bodyEl.style.paddingTop = topOffset + "px"; + + // deal with sidebar offsets + const sidebars = window.document.querySelectorAll( + ".sidebar, .headroom-target" + ); + sidebars.forEach((sidebar) => { + if (!animated) { + sidebar.classList.add("notransition"); + // Remove the no transition class after the animation has time to complete + setTimeout(function () { + sidebar.classList.remove("notransition"); + }, 201); + } + + if (window.Headroom && sidebar.classList.contains("sidebar-unpinned")) { + sidebar.style.top = "0"; + sidebar.style.maxHeight = "100vh"; + } else { + sidebar.style.top = topOffset + "px"; + sidebar.style.maxHeight = "calc(100vh - " + topOffset + "px)"; + } + }); + + // allow space for footer + const mainContainer = window.document.querySelector(".quarto-container"); + if (mainContainer) { + mainContainer.style.minHeight = "calc(100vh - " + bodyOffset + "px)"; + } + + // link offset + let linkStyle = window.document.querySelector("#quarto-target-style"); + if (!linkStyle) { + linkStyle = window.document.createElement("style"); + linkStyle.setAttribute("id", "quarto-target-style"); + window.document.head.appendChild(linkStyle); + } + while (linkStyle.firstChild) { + linkStyle.removeChild(linkStyle.firstChild); + } + if (topOffset > 0) { + linkStyle.appendChild( + window.document.createTextNode(` + section:target::before { + content: ""; + display: block; + height: ${topOffset}px; + margin: -${topOffset}px 0 0; + }`) + ); + } + if (init) { + window.dispatchEvent(headroomChanged); + } + init = true; + } + + // initialize headroom + var header = window.document.querySelector("#quarto-header"); + if (header && window.Headroom) { + const headroom = new window.Headroom(header, { + tolerance: 5, + onPin: function () { + const sidebars = window.document.querySelectorAll( + ".sidebar, .headroom-target" + ); + sidebars.forEach((sidebar) => { + sidebar.classList.remove("sidebar-unpinned"); + }); + updateDocumentOffset(); + }, + onUnpin: function () { + const sidebars = window.document.querySelectorAll( + ".sidebar, .headroom-target" + ); + sidebars.forEach((sidebar) => { + sidebar.classList.add("sidebar-unpinned"); + }); + updateDocumentOffset(); + }, + }); + headroom.init(); + + let frozen = false; + window.quartoToggleHeadroom = function () { + if (frozen) { + headroom.unfreeze(); + frozen = false; + } else { + headroom.freeze(); + frozen = true; + } + }; + } + + window.addEventListener( + "hashchange", + function (e) { + if ( + getComputedStyle(document.documentElement).scrollBehavior !== "smooth" + ) { + window.scrollTo(0, window.pageYOffset - headerOffset()); + } + }, + false + ); + + // Observe size changed for the header + const headerEl = window.document.querySelector("header.fixed-top"); + if (headerEl && window.ResizeObserver) { + const observer = new window.ResizeObserver(() => { + setTimeout(updateDocumentOffsetWithoutAnimation, 0); + }); + observer.observe(headerEl, { + attributes: true, + childList: true, + characterData: true, + }); + } else { + window.addEventListener( + "resize", + throttle(updateDocumentOffsetWithoutAnimation, 50) + ); + } + setTimeout(updateDocumentOffsetWithoutAnimation, 250); + + // fixup index.html links if we aren't on the filesystem + if (window.location.protocol !== "file:") { + const links = window.document.querySelectorAll("a"); + for (let i = 0; i < links.length; i++) { + if (links[i].href) { + links[i].dataset.originalHref = links[i].href; + links[i].href = links[i].href.replace(/\/index\.html/, "/"); + } + } + + // Fixup any sharing links that require urls + // Append url to any sharing urls + const sharingLinks = window.document.querySelectorAll( + "a.sidebar-tools-main-item, a.quarto-navigation-tool, a.quarto-navbar-tools, a.quarto-navbar-tools-item" + ); + for (let i = 0; i < sharingLinks.length; i++) { + const sharingLink = sharingLinks[i]; + const href = sharingLink.getAttribute("href"); + if (href) { + sharingLink.setAttribute( + "href", + href.replace("|url|", window.location.href) + ); + } + } + + // Scroll the active navigation item into view, if necessary + const navSidebar = window.document.querySelector("nav#quarto-sidebar"); + if (navSidebar) { + // Find the active item + const activeItem = navSidebar.querySelector("li.sidebar-item a.active"); + if (activeItem) { + // Wait for the scroll height and height to resolve by observing size changes on the + // nav element that is scrollable + const resizeObserver = new ResizeObserver((_entries) => { + // The bottom of the element + const elBottom = activeItem.offsetTop; + const viewBottom = navSidebar.scrollTop + navSidebar.clientHeight; + + // The element height and scroll height are the same, then we are still loading + if (viewBottom !== navSidebar.scrollHeight) { + // Determine if the item isn't visible and scroll to it + if (elBottom >= viewBottom) { + navSidebar.scrollTop = elBottom; + } + + // stop observing now since we've completed the scroll + resizeObserver.unobserve(navSidebar); + } + }); + resizeObserver.observe(navSidebar); + } + } + } +}); diff --git a/site_libs/quarto-search/autocomplete.umd.js b/site_libs/quarto-search/autocomplete.umd.js new file mode 100644 index 00000000..ae0063aa --- /dev/null +++ b/site_libs/quarto-search/autocomplete.umd.js @@ -0,0 +1,3 @@ +/*! @algolia/autocomplete-js 1.11.1 | MIT License | © Algolia, Inc. and contributors | https://github.com/algolia/autocomplete */ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self)["@algolia/autocomplete-js"]={})}(this,(function(e){"use strict";function t(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function n(e){for(var n=1;n=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}function a(e,t){return function(e){if(Array.isArray(e))return e}(e)||function(e,t){var n=null==e?null:"undefined"!=typeof Symbol&&e[Symbol.iterator]||e["@@iterator"];if(null!=n){var r,o,i,u,a=[],l=!0,c=!1;try{if(i=(n=n.call(e)).next,0===t){if(Object(n)!==n)return;l=!1}else for(;!(l=(r=i.call(n)).done)&&(a.push(r.value),a.length!==t);l=!0);}catch(e){c=!0,o=e}finally{try{if(!l&&null!=n.return&&(u=n.return(),Object(u)!==u))return}finally{if(c)throw o}}return a}}(e,t)||c(e,t)||function(){throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}()}function l(e){return function(e){if(Array.isArray(e))return s(e)}(e)||function(e){if("undefined"!=typeof Symbol&&null!=e[Symbol.iterator]||null!=e["@@iterator"])return Array.from(e)}(e)||c(e)||function(){throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}()}function c(e,t){if(e){if("string"==typeof e)return s(e,t);var n=Object.prototype.toString.call(e).slice(8,-1);return"Object"===n&&e.constructor&&(n=e.constructor.name),"Map"===n||"Set"===n?Array.from(e):"Arguments"===n||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)?s(e,t):void 0}}function s(e,t){(null==t||t>e.length)&&(t=e.length);for(var n=0,r=new Array(t);ne.length)&&(t=e.length);for(var n=0,r=new Array(t);ne.length)&&(t=e.length);for(var n=0,r=new Array(t);n=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}function x(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function N(e){for(var t=1;t1&&void 0!==arguments[1]?arguments[1]:20,n=[],r=0;r=3||2===n&&r>=4||1===n&&r>=10);function i(t,n,r){if(o&&void 0!==r){var i=r[0].__autocomplete_algoliaCredentials,u={"X-Algolia-Application-Id":i.appId,"X-Algolia-API-Key":i.apiKey};e.apply(void 0,[t].concat(D(n),[{headers:u}]))}else e.apply(void 0,[t].concat(D(n)))}return{init:function(t,n){e("init",{appId:t,apiKey:n})},setUserToken:function(t){e("setUserToken",t)},clickedObjectIDsAfterSearch:function(){for(var e=arguments.length,t=new Array(e),n=0;n0&&i("clickedObjectIDsAfterSearch",B(t),t[0].items)},clickedObjectIDs:function(){for(var e=arguments.length,t=new Array(e),n=0;n0&&i("clickedObjectIDs",B(t),t[0].items)},clickedFilters:function(){for(var t=arguments.length,n=new Array(t),r=0;r0&&e.apply(void 0,["clickedFilters"].concat(n))},convertedObjectIDsAfterSearch:function(){for(var e=arguments.length,t=new Array(e),n=0;n0&&i("convertedObjectIDsAfterSearch",B(t),t[0].items)},convertedObjectIDs:function(){for(var e=arguments.length,t=new Array(e),n=0;n0&&i("convertedObjectIDs",B(t),t[0].items)},convertedFilters:function(){for(var t=arguments.length,n=new Array(t),r=0;r0&&e.apply(void 0,["convertedFilters"].concat(n))},viewedObjectIDs:function(){for(var e=arguments.length,t=new Array(e),n=0;n0&&t.reduce((function(e,t){var n=t.items,r=k(t,A);return[].concat(D(e),D(q(N(N({},r),{},{objectIDs:(null==n?void 0:n.map((function(e){return e.objectID})))||r.objectIDs})).map((function(e){return{items:n,payload:e}}))))}),[]).forEach((function(e){var t=e.items;return i("viewedObjectIDs",[e.payload],t)}))},viewedFilters:function(){for(var t=arguments.length,n=new Array(t),r=0;r0&&e.apply(void 0,["viewedFilters"].concat(n))}}}function F(e){var t=e.items.reduce((function(e,t){var n;return e[t.__autocomplete_indexName]=(null!==(n=e[t.__autocomplete_indexName])&&void 0!==n?n:[]).concat(t),e}),{});return Object.keys(t).map((function(e){return{index:e,items:t[e],algoliaSource:["autocomplete"]}}))}function L(e){return e.objectID&&e.__autocomplete_indexName&&e.__autocomplete_queryID}function U(e){return U="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e},U(e)}function M(e){return function(e){if(Array.isArray(e))return H(e)}(e)||function(e){if("undefined"!=typeof Symbol&&null!=e[Symbol.iterator]||null!=e["@@iterator"])return Array.from(e)}(e)||function(e,t){if(!e)return;if("string"==typeof e)return H(e,t);var n=Object.prototype.toString.call(e).slice(8,-1);"Object"===n&&e.constructor&&(n=e.constructor.name);if("Map"===n||"Set"===n)return Array.from(e);if("Arguments"===n||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return H(e,t)}(e)||function(){throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}()}function H(e,t){(null==t||t>e.length)&&(t=e.length);for(var n=0,r=new Array(t);n0&&z({onItemsChange:r,items:n,insights:a,state:t}))}}),0);return{name:"aa.algoliaInsightsPlugin",subscribe:function(e){var t=e.setContext,n=e.onSelect,r=e.onActive;function l(e){t({algoliaInsightsPlugin:{__algoliaSearchParameters:W({clickAnalytics:!0},e?{userToken:e}:{}),insights:a}})}u("addAlgoliaAgent","insights-plugin"),l(),u("onUserTokenChange",l),u("getUserToken",null,(function(e,t){l(t)})),n((function(e){var t=e.item,n=e.state,r=e.event,i=e.source;L(t)&&o({state:n,event:r,insights:a,item:t,insightsEvents:[W({eventName:"Item Selected"},j({item:t,items:i.getItems().filter(L)}))]})})),r((function(e){var t=e.item,n=e.source,r=e.state,o=e.event;L(t)&&i({state:r,event:o,insights:a,item:t,insightsEvents:[W({eventName:"Item Active"},j({item:t,items:n.getItems().filter(L)}))]})}))},onStateChange:function(e){var t=e.state;c({state:t})},__autocomplete_pluginOptions:e}}function J(e,t){var n=t;return{then:function(t,r){return J(e.then(Y(t,n,e),Y(r,n,e)),n)},catch:function(t){return J(e.catch(Y(t,n,e)),n)},finally:function(t){return t&&n.onCancelList.push(t),J(e.finally(Y(t&&function(){return n.onCancelList=[],t()},n,e)),n)},cancel:function(){n.isCanceled=!0;var e=n.onCancelList;n.onCancelList=[],e.forEach((function(e){e()}))},isCanceled:function(){return!0===n.isCanceled}}}function X(e){return J(e,{isCanceled:!1,onCancelList:[]})}function Y(e,t,n){return e?function(n){return t.isCanceled?n:e(n)}:n}function Z(e,t,n,r){if(!n)return null;if(e<0&&(null===t||null!==r&&0===t))return n+e;var o=(null===t?-1:t)+e;return o<=-1||o>=n?null===r?null:0:o}function ee(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function te(e){for(var t=1;te.length)&&(t=e.length);for(var n=0,r=new Array(t);n0},reshape:function(e){return e.sources}},e),{},{id:null!==(n=e.id)&&void 0!==n?n:d(),plugins:o,initialState:he({activeItemId:null,query:"",completion:null,collections:[],isOpen:!1,status:"idle",context:{}},e.initialState),onStateChange:function(t){var n;null===(n=e.onStateChange)||void 0===n||n.call(e,t),o.forEach((function(e){var n;return null===(n=e.onStateChange)||void 0===n?void 0:n.call(e,t)}))},onSubmit:function(t){var n;null===(n=e.onSubmit)||void 0===n||n.call(e,t),o.forEach((function(e){var n;return null===(n=e.onSubmit)||void 0===n?void 0:n.call(e,t)}))},onReset:function(t){var n;null===(n=e.onReset)||void 0===n||n.call(e,t),o.forEach((function(e){var n;return null===(n=e.onReset)||void 0===n?void 0:n.call(e,t)}))},getSources:function(n){return Promise.all([].concat(ye(o.map((function(e){return e.getSources}))),[e.getSources]).filter(Boolean).map((function(e){return function(e,t){var n=[];return Promise.resolve(e(t)).then((function(e){return Promise.all(e.filter((function(e){return Boolean(e)})).map((function(e){if(e.sourceId,n.includes(e.sourceId))throw new Error("[Autocomplete] The `sourceId` ".concat(JSON.stringify(e.sourceId)," is not unique."));n.push(e.sourceId);var t={getItemInputValue:function(e){return e.state.query},getItemUrl:function(){},onSelect:function(e){(0,e.setIsOpen)(!1)},onActive:O,onResolve:O};Object.keys(t).forEach((function(e){t[e].__default=!0}));var r=te(te({},t),e);return Promise.resolve(r)})))}))}(e,n)}))).then((function(e){return m(e)})).then((function(e){return e.map((function(e){return he(he({},e),{},{onSelect:function(n){e.onSelect(n),t.forEach((function(e){var t;return null===(t=e.onSelect)||void 0===t?void 0:t.call(e,n)}))},onActive:function(n){e.onActive(n),t.forEach((function(e){var t;return null===(t=e.onActive)||void 0===t?void 0:t.call(e,n)}))},onResolve:function(n){e.onResolve(n),t.forEach((function(e){var t;return null===(t=e.onResolve)||void 0===t?void 0:t.call(e,n)}))}})}))}))},navigator:he({navigate:function(e){var t=e.itemUrl;r.location.assign(t)},navigateNewTab:function(e){var t=e.itemUrl,n=r.open(t,"_blank","noopener");null==n||n.focus()},navigateNewWindow:function(e){var t=e.itemUrl;r.open(t,"_blank","noopener")}},e.navigator)})}function Se(e){return Se="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e},Se(e)}function je(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function Pe(e){for(var t=1;te.length)&&(t=e.length);for(var n=0,r=new Array(t);n=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}var He,Ve,We,Ke=null,Qe=(He=-1,Ve=-1,We=void 0,function(e){var t=++He;return Promise.resolve(e).then((function(e){return We&&t=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}function et(e){return et="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e},et(e)}var tt=["props","refresh","store"],nt=["inputElement","formElement","panelElement"],rt=["inputElement"],ot=["inputElement","maxLength"],it=["source"],ut=["item","source"];function at(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function lt(e){for(var t=1;t=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}function ft(e){var t=e.props,n=e.refresh,r=e.store,o=st(e,tt);return{getEnvironmentProps:function(e){var n=e.inputElement,o=e.formElement,i=e.panelElement;function u(e){!r.getState().isOpen&&r.pendingRequests.isEmpty()||e.target===n||!1===[o,i].some((function(t){return n=t,r=e.target,n===r||n.contains(r);var n,r}))&&(r.dispatch("blur",null),t.debug||r.pendingRequests.cancelAll())}return lt({onTouchStart:u,onMouseDown:u,onTouchMove:function(e){!1!==r.getState().isOpen&&n===t.environment.document.activeElement&&e.target!==n&&n.blur()}},st(e,nt))},getRootProps:function(e){return lt({role:"combobox","aria-expanded":r.getState().isOpen,"aria-haspopup":"listbox","aria-owns":r.getState().isOpen?r.getState().collections.map((function(e){var n=e.source;return ie(t.id,"list",n)})).join(" "):void 0,"aria-labelledby":ie(t.id,"label")},e)},getFormProps:function(e){return e.inputElement,lt({action:"",noValidate:!0,role:"search",onSubmit:function(i){var u;i.preventDefault(),t.onSubmit(lt({event:i,refresh:n,state:r.getState()},o)),r.dispatch("submit",null),null===(u=e.inputElement)||void 0===u||u.blur()},onReset:function(i){var u;i.preventDefault(),t.onReset(lt({event:i,refresh:n,state:r.getState()},o)),r.dispatch("reset",null),null===(u=e.inputElement)||void 0===u||u.focus()}},st(e,rt))},getLabelProps:function(e){return lt({htmlFor:ie(t.id,"input"),id:ie(t.id,"label")},e)},getInputProps:function(e){var i;function u(e){(t.openOnFocus||Boolean(r.getState().query))&&$e(lt({event:e,props:t,query:r.getState().completion||r.getState().query,refresh:n,store:r},o)),r.dispatch("focus",null)}var a=e||{};a.inputElement;var l=a.maxLength,c=void 0===l?512:l,s=st(a,ot),f=oe(r.getState()),p=function(e){return Boolean(e&&e.match(ue))}((null===(i=t.environment.navigator)||void 0===i?void 0:i.userAgent)||""),m=t.enterKeyHint||(null!=f&&f.itemUrl&&!p?"go":"search");return lt({"aria-autocomplete":"both","aria-activedescendant":r.getState().isOpen&&null!==r.getState().activeItemId?ie(t.id,"item-".concat(r.getState().activeItemId),null==f?void 0:f.source):void 0,"aria-controls":r.getState().isOpen?r.getState().collections.map((function(e){var n=e.source;return ie(t.id,"list",n)})).join(" "):void 0,"aria-labelledby":ie(t.id,"label"),value:r.getState().completion||r.getState().query,id:ie(t.id,"input"),autoComplete:"off",autoCorrect:"off",autoCapitalize:"off",enterKeyHint:m,spellCheck:"false",autoFocus:t.autoFocus,placeholder:t.placeholder,maxLength:c,type:"search",onChange:function(e){$e(lt({event:e,props:t,query:e.currentTarget.value.slice(0,c),refresh:n,store:r},o))},onKeyDown:function(e){!function(e){var t=e.event,n=e.props,r=e.refresh,o=e.store,i=Ze(e,Ge);if("ArrowUp"===t.key||"ArrowDown"===t.key){var u=function(){var e=oe(o.getState()),t=n.environment.document.getElementById(ie(n.id,"item-".concat(o.getState().activeItemId),null==e?void 0:e.source));t&&(t.scrollIntoViewIfNeeded?t.scrollIntoViewIfNeeded(!1):t.scrollIntoView(!1))},a=function(){var e=oe(o.getState());if(null!==o.getState().activeItemId&&e){var n=e.item,u=e.itemInputValue,a=e.itemUrl,l=e.source;l.onActive(Xe({event:t,item:n,itemInputValue:u,itemUrl:a,refresh:r,source:l,state:o.getState()},i))}};t.preventDefault(),!1===o.getState().isOpen&&(n.openOnFocus||Boolean(o.getState().query))?$e(Xe({event:t,props:n,query:o.getState().query,refresh:r,store:o},i)).then((function(){o.dispatch(t.key,{nextActiveItemId:n.defaultActiveItemId}),a(),setTimeout(u,0)})):(o.dispatch(t.key,{}),a(),u())}else if("Escape"===t.key)t.preventDefault(),o.dispatch(t.key,null),o.pendingRequests.cancelAll();else if("Tab"===t.key)o.dispatch("blur",null),o.pendingRequests.cancelAll();else if("Enter"===t.key){if(null===o.getState().activeItemId||o.getState().collections.every((function(e){return 0===e.items.length})))return void(n.debug||o.pendingRequests.cancelAll());t.preventDefault();var l=oe(o.getState()),c=l.item,s=l.itemInputValue,f=l.itemUrl,p=l.source;if(t.metaKey||t.ctrlKey)void 0!==f&&(p.onSelect(Xe({event:t,item:c,itemInputValue:s,itemUrl:f,refresh:r,source:p,state:o.getState()},i)),n.navigator.navigateNewTab({itemUrl:f,item:c,state:o.getState()}));else if(t.shiftKey)void 0!==f&&(p.onSelect(Xe({event:t,item:c,itemInputValue:s,itemUrl:f,refresh:r,source:p,state:o.getState()},i)),n.navigator.navigateNewWindow({itemUrl:f,item:c,state:o.getState()}));else if(t.altKey);else{if(void 0!==f)return p.onSelect(Xe({event:t,item:c,itemInputValue:s,itemUrl:f,refresh:r,source:p,state:o.getState()},i)),void n.navigator.navigate({itemUrl:f,item:c,state:o.getState()});$e(Xe({event:t,nextState:{isOpen:!1},props:n,query:s,refresh:r,store:o},i)).then((function(){p.onSelect(Xe({event:t,item:c,itemInputValue:s,itemUrl:f,refresh:r,source:p,state:o.getState()},i))}))}}}(lt({event:e,props:t,refresh:n,store:r},o))},onFocus:u,onBlur:O,onClick:function(n){e.inputElement!==t.environment.document.activeElement||r.getState().isOpen||u(n)}},s)},getPanelProps:function(e){return lt({onMouseDown:function(e){e.preventDefault()},onMouseLeave:function(){r.dispatch("mouseleave",null)}},e)},getListProps:function(e){var n=e||{},r=n.source,o=st(n,it);return lt({role:"listbox","aria-labelledby":ie(t.id,"label"),id:ie(t.id,"list",r)},o)},getItemProps:function(e){var i=e.item,u=e.source,a=st(e,ut);return lt({id:ie(t.id,"item-".concat(i.__autocomplete_id),u),role:"option","aria-selected":r.getState().activeItemId===i.__autocomplete_id,onMouseMove:function(e){if(i.__autocomplete_id!==r.getState().activeItemId){r.dispatch("mousemove",i.__autocomplete_id);var t=oe(r.getState());if(null!==r.getState().activeItemId&&t){var u=t.item,a=t.itemInputValue,l=t.itemUrl,c=t.source;c.onActive(lt({event:e,item:u,itemInputValue:a,itemUrl:l,refresh:n,source:c,state:r.getState()},o))}}},onMouseDown:function(e){e.preventDefault()},onClick:function(e){var a=u.getItemInputValue({item:i,state:r.getState()}),l=u.getItemUrl({item:i,state:r.getState()});(l?Promise.resolve():$e(lt({event:e,nextState:{isOpen:!1},props:t,query:a,refresh:n,store:r},o))).then((function(){u.onSelect(lt({event:e,item:i,itemInputValue:a,itemUrl:l,refresh:n,source:u,state:r.getState()},o))}))}},a)}}}function pt(e){return pt="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e},pt(e)}function mt(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function vt(e){for(var t=1;t=5&&((o||!e&&5===r)&&(u.push(r,0,o,n),r=6),e&&(u.push(r,e,0,n),r=6)),o=""},l=0;l"===t?(r=1,o=""):o=t+o[0]:i?t===i?i="":o+=t:'"'===t||"'"===t?i=t:">"===t?(a(),r=1):r&&("="===t?(r=5,n=o,o=""):"/"===t&&(r<5||">"===e[l][c+1])?(a(),3===r&&(u=u[0]),r=u,(u=u[0]).push(2,0,r),r=0):" "===t||"\t"===t||"\n"===t||"\r"===t?(a(),r=2):o+=t),3===r&&"!--"===o&&(r=4,u=u[0])}return a(),u}(e)),t),arguments,[])).length>1?t:t[0]}var kt=function(e){var t=e.environment,n=t.document.createElementNS("http://www.w3.org/2000/svg","svg");n.setAttribute("class","aa-ClearIcon"),n.setAttribute("viewBox","0 0 24 24"),n.setAttribute("width","18"),n.setAttribute("height","18"),n.setAttribute("fill","currentColor");var r=t.document.createElementNS("http://www.w3.org/2000/svg","path");return r.setAttribute("d","M5.293 6.707l5.293 5.293-5.293 5.293c-0.391 0.391-0.391 1.024 0 1.414s1.024 0.391 1.414 0l5.293-5.293 5.293 5.293c0.391 0.391 1.024 0.391 1.414 0s0.391-1.024 0-1.414l-5.293-5.293 5.293-5.293c0.391-0.391 0.391-1.024 0-1.414s-1.024-0.391-1.414 0l-5.293 5.293-5.293-5.293c-0.391-0.391-1.024-0.391-1.414 0s-0.391 1.024 0 1.414z"),n.appendChild(r),n};function xt(e,t){if("string"==typeof t){var n=e.document.querySelector(t);return"The element ".concat(JSON.stringify(t)," is not in the document."),n}return t}function Nt(){for(var e=arguments.length,t=new Array(e),n=0;n2&&(u.children=arguments.length>3?Jt.call(arguments,2):n),"function"==typeof e&&null!=e.defaultProps)for(i in e.defaultProps)void 0===u[i]&&(u[i]=e.defaultProps[i]);return sn(e,u,r,o,null)}function sn(e,t,n,r,o){var i={type:e,props:t,key:n,ref:r,__k:null,__:null,__b:0,__e:null,__d:void 0,__c:null,__h:null,constructor:void 0,__v:null==o?++Yt:o};return null==o&&null!=Xt.vnode&&Xt.vnode(i),i}function fn(e){return e.children}function pn(e,t){this.props=e,this.context=t}function mn(e,t){if(null==t)return e.__?mn(e.__,e.__.__k.indexOf(e)+1):null;for(var n;tt&&Zt.sort(nn));yn.__r=0}function bn(e,t,n,r,o,i,u,a,l,c){var s,f,p,m,v,d,y,b=r&&r.__k||on,g=b.length;for(n.__k=[],s=0;s0?sn(m.type,m.props,m.key,m.ref?m.ref:null,m.__v):m)){if(m.__=n,m.__b=n.__b+1,null===(p=b[s])||p&&m.key==p.key&&m.type===p.type)b[s]=void 0;else for(f=0;f=0;t--)if((n=e.__k[t])&&(r=On(n)))return r;return null}function _n(e,t,n){"-"===t[0]?e.setProperty(t,null==n?"":n):e[t]=null==n?"":"number"!=typeof n||un.test(t)?n:n+"px"}function Sn(e,t,n,r,o){var i;e:if("style"===t)if("string"==typeof n)e.style.cssText=n;else{if("string"==typeof r&&(e.style.cssText=r=""),r)for(t in r)n&&t in n||_n(e.style,t,"");if(n)for(t in n)r&&n[t]===r[t]||_n(e.style,t,n[t])}else if("o"===t[0]&&"n"===t[1])i=t!==(t=t.replace(/Capture$/,"")),t=t.toLowerCase()in e?t.toLowerCase().slice(2):t.slice(2),e.l||(e.l={}),e.l[t+i]=n,n?r||e.addEventListener(t,i?Pn:jn,i):e.removeEventListener(t,i?Pn:jn,i);else if("dangerouslySetInnerHTML"!==t){if(o)t=t.replace(/xlink(H|:h)/,"h").replace(/sName$/,"s");else if("width"!==t&&"height"!==t&&"href"!==t&&"list"!==t&&"form"!==t&&"tabIndex"!==t&&"download"!==t&&t in e)try{e[t]=null==n?"":n;break e}catch(e){}"function"==typeof n||(null==n||!1===n&&"-"!==t[4]?e.removeAttribute(t):e.setAttribute(t,n))}}function jn(e){return this.l[e.type+!1](Xt.event?Xt.event(e):e)}function Pn(e){return this.l[e.type+!0](Xt.event?Xt.event(e):e)}function wn(e,t,n,r,o,i,u,a,l){var c,s,f,p,m,v,d,y,b,g,h,O,_,S,j,P=t.type;if(void 0!==t.constructor)return null;null!=n.__h&&(l=n.__h,a=t.__e=n.__e,t.__h=null,i=[a]),(c=Xt.__b)&&c(t);try{e:if("function"==typeof P){if(y=t.props,b=(c=P.contextType)&&r[c.__c],g=c?b?b.props.value:c.__:r,n.__c?d=(s=t.__c=n.__c).__=s.__E:("prototype"in P&&P.prototype.render?t.__c=s=new P(y,g):(t.__c=s=new pn(y,g),s.constructor=P,s.render=Cn),b&&b.sub(s),s.props=y,s.state||(s.state={}),s.context=g,s.__n=r,f=s.__d=!0,s.__h=[],s._sb=[]),null==s.__s&&(s.__s=s.state),null!=P.getDerivedStateFromProps&&(s.__s==s.state&&(s.__s=an({},s.__s)),an(s.__s,P.getDerivedStateFromProps(y,s.__s))),p=s.props,m=s.state,s.__v=t,f)null==P.getDerivedStateFromProps&&null!=s.componentWillMount&&s.componentWillMount(),null!=s.componentDidMount&&s.__h.push(s.componentDidMount);else{if(null==P.getDerivedStateFromProps&&y!==p&&null!=s.componentWillReceiveProps&&s.componentWillReceiveProps(y,g),!s.__e&&null!=s.shouldComponentUpdate&&!1===s.shouldComponentUpdate(y,s.__s,g)||t.__v===n.__v){for(t.__v!==n.__v&&(s.props=y,s.state=s.__s,s.__d=!1),s.__e=!1,t.__e=n.__e,t.__k=n.__k,t.__k.forEach((function(e){e&&(e.__=t)})),h=0;h0&&void 0!==arguments[0]?arguments[0]:[];return{get:function(){return e},add:function(t){var n=e[e.length-1];(null==n?void 0:n.isHighlighted)===t.isHighlighted?e[e.length-1]={value:n.value+t.value,isHighlighted:n.isHighlighted}:e.push(t)}}}(n?[{value:n,isHighlighted:!1}]:[]);return t.forEach((function(e){var t=e.split(xn);r.add({value:t[0],isHighlighted:!0}),""!==t[1]&&r.add({value:t[1],isHighlighted:!1})})),r.get()}function Tn(e){return function(e){if(Array.isArray(e))return qn(e)}(e)||function(e){if("undefined"!=typeof Symbol&&null!=e[Symbol.iterator]||null!=e["@@iterator"])return Array.from(e)}(e)||function(e,t){if(!e)return;if("string"==typeof e)return qn(e,t);var n=Object.prototype.toString.call(e).slice(8,-1);"Object"===n&&e.constructor&&(n=e.constructor.name);if("Map"===n||"Set"===n)return Array.from(e);if("Arguments"===n||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return qn(e,t)}(e)||function(){throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}()}function qn(e,t){(null==t||t>e.length)&&(t=e.length);for(var n=0,r=new Array(t);n",""":'"',"'":"'"},Fn=new RegExp(/\w/i),Ln=/&(amp|quot|lt|gt|#39);/g,Un=RegExp(Ln.source);function Mn(e,t){var n,r,o,i=e[t],u=(null===(n=e[t+1])||void 0===n?void 0:n.isHighlighted)||!0,a=(null===(r=e[t-1])||void 0===r?void 0:r.isHighlighted)||!0;return Fn.test((o=i.value)&&Un.test(o)?o.replace(Ln,(function(e){return Rn[e]})):o)||a!==u?i.isHighlighted:a}function Hn(e){return Hn="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e},Hn(e)}function Vn(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function Wn(e){for(var t=1;te.length)&&(t=e.length);for(var n=0,r=new Array(t);n=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}function ur(e){return function(e){if(Array.isArray(e))return ar(e)}(e)||function(e){if("undefined"!=typeof Symbol&&null!=e[Symbol.iterator]||null!=e["@@iterator"])return Array.from(e)}(e)||function(e,t){if(!e)return;if("string"==typeof e)return ar(e,t);var n=Object.prototype.toString.call(e).slice(8,-1);"Object"===n&&e.constructor&&(n=e.constructor.name);if("Map"===n||"Set"===n)return Array.from(e);if("Arguments"===n||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return ar(e,t)}(e)||function(){throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}()}function ar(e,t){(null==t||t>e.length)&&(t=e.length);for(var n=0,r=new Array(t);n0;if(!O.value.core.openOnFocus&&!t.query)return n;var r=Boolean(y.current||O.value.renderer.renderNoResults);return!n&&r||n},__autocomplete_metadata:{userAgents:br,options:e}}))})),j=f(n({collections:[],completion:null,context:{},isOpen:!1,query:"",activeItemId:null,status:"idle"},O.value.core.initialState)),P={getEnvironmentProps:O.value.renderer.getEnvironmentProps,getFormProps:O.value.renderer.getFormProps,getInputProps:O.value.renderer.getInputProps,getItemProps:O.value.renderer.getItemProps,getLabelProps:O.value.renderer.getLabelProps,getListProps:O.value.renderer.getListProps,getPanelProps:O.value.renderer.getPanelProps,getRootProps:O.value.renderer.getRootProps},w={setActiveItemId:S.value.setActiveItemId,setQuery:S.value.setQuery,setCollections:S.value.setCollections,setIsOpen:S.value.setIsOpen,setStatus:S.value.setStatus,setContext:S.value.setContext,refresh:S.value.refresh,navigator:S.value.navigator},I=m((function(){return Ct.bind(O.value.renderer.renderer.createElement)})),A=m((function(){return Gt({autocomplete:S.value,autocompleteScopeApi:w,classNames:O.value.renderer.classNames,environment:O.value.core.environment,isDetached:_.value,placeholder:O.value.core.placeholder,propGetters:P,setIsModalOpen:k,state:j.current,translations:O.value.renderer.translations})}));function E(){Ht(A.value.panel,{style:_.value?{}:yr({panelPlacement:O.value.renderer.panelPlacement,container:A.value.root,form:A.value.form,environment:O.value.core.environment})})}function D(e){j.current=e;var t={autocomplete:S.value,autocompleteScopeApi:w,classNames:O.value.renderer.classNames,components:O.value.renderer.components,container:O.value.renderer.container,html:I.value,dom:A.value,panelContainer:_.value?A.value.detachedContainer:O.value.renderer.panelContainer,propGetters:P,state:j.current,renderer:O.value.renderer.renderer},r=!b(e)&&!y.current&&O.value.renderer.renderNoResults||O.value.renderer.render;!function(e){var t=e.autocomplete,r=e.autocompleteScopeApi,o=e.dom,i=e.propGetters,u=e.state;Vt(o.root,i.getRootProps(n({state:u,props:t.getRootProps({})},r))),Vt(o.input,i.getInputProps(n({state:u,props:t.getInputProps({inputElement:o.input}),inputElement:o.input},r))),Ht(o.label,{hidden:"stalled"===u.status}),Ht(o.loadingIndicator,{hidden:"stalled"!==u.status}),Ht(o.clearButton,{hidden:!u.query}),Ht(o.detachedSearchButtonQuery,{textContent:u.query}),Ht(o.detachedSearchButtonPlaceholder,{hidden:Boolean(u.query)})}(t),function(e,t){var r=t.autocomplete,o=t.autocompleteScopeApi,u=t.classNames,a=t.html,l=t.dom,c=t.panelContainer,s=t.propGetters,f=t.state,p=t.components,m=t.renderer;if(f.isOpen){c.contains(l.panel)||"loading"===f.status||c.appendChild(l.panel),l.panel.classList.toggle("aa-Panel--stalled","stalled"===f.status);var v=f.collections.filter((function(e){var t=e.source,n=e.items;return t.templates.noResults||n.length>0})).map((function(e,t){var l=e.source,c=e.items;return m.createElement("section",{key:t,className:u.source,"data-autocomplete-source-id":l.sourceId},l.templates.header&&m.createElement("div",{className:u.sourceHeader},l.templates.header({components:p,createElement:m.createElement,Fragment:m.Fragment,items:c,source:l,state:f,html:a})),l.templates.noResults&&0===c.length?m.createElement("div",{className:u.sourceNoResults},l.templates.noResults({components:p,createElement:m.createElement,Fragment:m.Fragment,source:l,state:f,html:a})):m.createElement("ul",i({className:u.list},s.getListProps(n({state:f,props:r.getListProps({source:l})},o))),c.map((function(e){var t=r.getItemProps({item:e,source:l});return m.createElement("li",i({key:t.id,className:u.item},s.getItemProps(n({state:f,props:t},o))),l.templates.item({components:p,createElement:m.createElement,Fragment:m.Fragment,item:e,state:f,html:a}))}))),l.templates.footer&&m.createElement("div",{className:u.sourceFooter},l.templates.footer({components:p,createElement:m.createElement,Fragment:m.Fragment,items:c,source:l,state:f,html:a})))})),d=m.createElement(m.Fragment,null,m.createElement("div",{className:u.panelLayout},v),m.createElement("div",{className:"aa-GradientBottom"})),y=v.reduce((function(e,t){return e[t.props["data-autocomplete-source-id"]]=t,e}),{});e(n(n({children:d,state:f,sections:v,elements:y},m),{},{components:p,html:a},o),l.panel)}else c.contains(l.panel)&&c.removeChild(l.panel)}(r,t)}function C(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{};l();var t=O.value.renderer,n=t.components,r=u(t,gr);g.current=qt(r,O.value.core,{components:Bt(n,(function(e){return!e.value.hasOwnProperty("__autocomplete_componentName")})),initialState:j.current},e),v(),c(),S.value.refresh().then((function(){D(j.current)}))}function k(e){requestAnimationFrame((function(){var t=O.value.core.environment.document.body.contains(A.value.detachedOverlay);e!==t&&(e?(O.value.core.environment.document.body.appendChild(A.value.detachedOverlay),O.value.core.environment.document.body.classList.add("aa-Detached"),A.value.input.focus()):(O.value.core.environment.document.body.removeChild(A.value.detachedOverlay),O.value.core.environment.document.body.classList.remove("aa-Detached")))}))}return a((function(){var e=S.value.getEnvironmentProps({formElement:A.value.form,panelElement:A.value.panel,inputElement:A.value.input});return Ht(O.value.core.environment,e),function(){Ht(O.value.core.environment,Object.keys(e).reduce((function(e,t){return n(n({},e),{},o({},t,void 0))}),{}))}})),a((function(){var e=_.value?O.value.core.environment.document.body:O.value.renderer.panelContainer,t=_.value?A.value.detachedOverlay:A.value.panel;return _.value&&j.current.isOpen&&k(!0),D(j.current),function(){e.contains(t)&&e.removeChild(t)}})),a((function(){var e=O.value.renderer.container;return e.appendChild(A.value.root),function(){e.removeChild(A.value.root)}})),a((function(){var e=p((function(e){D(e.state)}),0);return h.current=function(t){var n=t.state,r=t.prevState;(_.value&&r.isOpen!==n.isOpen&&k(n.isOpen),_.value||!n.isOpen||r.isOpen||E(),n.query!==r.query)&&O.value.core.environment.document.querySelectorAll(".aa-Panel--scrollable").forEach((function(e){0!==e.scrollTop&&(e.scrollTop=0)}));e({state:n})},function(){h.current=void 0}})),a((function(){var e=p((function(){var e=_.value;_.value=O.value.core.environment.matchMedia(O.value.renderer.detachedMediaQuery).matches,e!==_.value?C({}):requestAnimationFrame(E)}),20);return O.value.core.environment.addEventListener("resize",e),function(){O.value.core.environment.removeEventListener("resize",e)}})),a((function(){if(!_.value)return function(){};function e(e){A.value.detachedContainer.classList.toggle("aa-DetachedContainer--modal",e)}function t(t){e(t.matches)}var n=O.value.core.environment.matchMedia(getComputedStyle(O.value.core.environment.document.documentElement).getPropertyValue("--aa-detached-modal-media-query"));e(n.matches);var r=Boolean(n.addEventListener);return r?n.addEventListener("change",t):n.addListener(t),function(){r?n.removeEventListener("change",t):n.removeListener(t)}})),a((function(){return requestAnimationFrame(E),function(){}})),n(n({},w),{},{update:C,destroy:function(){l()}})},e.getAlgoliaFacets=function(e){var t=hr({transformResponse:function(e){return e.facetHits}}),r=e.queries.map((function(e){return n(n({},e),{},{type:"facet"})}));return t(n(n({},e),{},{queries:r}))},e.getAlgoliaResults=Or,Object.defineProperty(e,"__esModule",{value:!0})})); + diff --git a/site_libs/quarto-search/fuse.min.js b/site_libs/quarto-search/fuse.min.js new file mode 100644 index 00000000..adc28356 --- /dev/null +++ b/site_libs/quarto-search/fuse.min.js @@ -0,0 +1,9 @@ +/** + * Fuse.js v6.6.2 - Lightweight fuzzy-search (http://fusejs.io) + * + * Copyright (c) 2022 Kiro Risk (http://kiro.me) + * All Rights Reserved. Apache Software License 2.0 + * + * http://www.apache.org/licenses/LICENSE-2.0 + */ +var e,t;e=this,t=function(){"use strict";function e(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function t(t){for(var n=1;ne.length)&&(t=e.length);for(var n=0,r=new Array(t);n0&&void 0!==arguments[0]?arguments[0]:1,t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:3,n=new Map,r=Math.pow(10,t);return{get:function(t){var i=t.match(C).length;if(n.has(i))return n.get(i);var o=1/Math.pow(i,.5*e),c=parseFloat(Math.round(o*r)/r);return n.set(i,c),c},clear:function(){n.clear()}}}var $=function(){function e(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},n=t.getFn,i=void 0===n?I.getFn:n,o=t.fieldNormWeight,c=void 0===o?I.fieldNormWeight:o;r(this,e),this.norm=E(c,3),this.getFn=i,this.isCreated=!1,this.setIndexRecords()}return o(e,[{key:"setSources",value:function(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:[];this.docs=e}},{key:"setIndexRecords",value:function(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:[];this.records=e}},{key:"setKeys",value:function(){var e=this,t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:[];this.keys=t,this._keysMap={},t.forEach((function(t,n){e._keysMap[t.id]=n}))}},{key:"create",value:function(){var e=this;!this.isCreated&&this.docs.length&&(this.isCreated=!0,g(this.docs[0])?this.docs.forEach((function(t,n){e._addString(t,n)})):this.docs.forEach((function(t,n){e._addObject(t,n)})),this.norm.clear())}},{key:"add",value:function(e){var t=this.size();g(e)?this._addString(e,t):this._addObject(e,t)}},{key:"removeAt",value:function(e){this.records.splice(e,1);for(var t=e,n=this.size();t2&&void 0!==arguments[2]?arguments[2]:{},r=n.getFn,i=void 0===r?I.getFn:r,o=n.fieldNormWeight,c=void 0===o?I.fieldNormWeight:o,a=new $({getFn:i,fieldNormWeight:c});return a.setKeys(e.map(_)),a.setSources(t),a.create(),a}function R(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},n=t.errors,r=void 0===n?0:n,i=t.currentLocation,o=void 0===i?0:i,c=t.expectedLocation,a=void 0===c?0:c,s=t.distance,u=void 0===s?I.distance:s,h=t.ignoreLocation,l=void 0===h?I.ignoreLocation:h,f=r/e.length;if(l)return f;var d=Math.abs(a-o);return u?f+d/u:d?1:f}function N(){for(var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:[],t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:I.minMatchCharLength,n=[],r=-1,i=-1,o=0,c=e.length;o=t&&n.push([r,i]),r=-1)}return e[o-1]&&o-r>=t&&n.push([r,o-1]),n}var P=32;function W(e){for(var t={},n=0,r=e.length;n1&&void 0!==arguments[1]?arguments[1]:{},o=i.location,c=void 0===o?I.location:o,a=i.threshold,s=void 0===a?I.threshold:a,u=i.distance,h=void 0===u?I.distance:u,l=i.includeMatches,f=void 0===l?I.includeMatches:l,d=i.findAllMatches,v=void 0===d?I.findAllMatches:d,g=i.minMatchCharLength,y=void 0===g?I.minMatchCharLength:g,p=i.isCaseSensitive,m=void 0===p?I.isCaseSensitive:p,k=i.ignoreLocation,M=void 0===k?I.ignoreLocation:k;if(r(this,e),this.options={location:c,threshold:s,distance:h,includeMatches:f,findAllMatches:v,minMatchCharLength:y,isCaseSensitive:m,ignoreLocation:M},this.pattern=m?t:t.toLowerCase(),this.chunks=[],this.pattern.length){var b=function(e,t){n.chunks.push({pattern:e,alphabet:W(e),startIndex:t})},x=this.pattern.length;if(x>P){for(var w=0,L=x%P,S=x-L;w3&&void 0!==arguments[3]?arguments[3]:{},i=r.location,o=void 0===i?I.location:i,c=r.distance,a=void 0===c?I.distance:c,s=r.threshold,u=void 0===s?I.threshold:s,h=r.findAllMatches,l=void 0===h?I.findAllMatches:h,f=r.minMatchCharLength,d=void 0===f?I.minMatchCharLength:f,v=r.includeMatches,g=void 0===v?I.includeMatches:v,y=r.ignoreLocation,p=void 0===y?I.ignoreLocation:y;if(t.length>P)throw new Error(w(P));for(var m,k=t.length,M=e.length,b=Math.max(0,Math.min(o,M)),x=u,L=b,S=d>1||g,_=S?Array(M):[];(m=e.indexOf(t,L))>-1;){var O=R(t,{currentLocation:m,expectedLocation:b,distance:a,ignoreLocation:p});if(x=Math.min(O,x),L=m+k,S)for(var j=0;j=z;q-=1){var B=q-1,J=n[e.charAt(B)];if(S&&(_[B]=+!!J),K[q]=(K[q+1]<<1|1)&J,F&&(K[q]|=(A[q+1]|A[q])<<1|1|A[q+1]),K[q]&$&&(C=R(t,{errors:F,currentLocation:B,expectedLocation:b,distance:a,ignoreLocation:p}))<=x){if(x=C,(L=B)<=b)break;z=Math.max(1,2*b-L)}}if(R(t,{errors:F+1,currentLocation:b,expectedLocation:b,distance:a,ignoreLocation:p})>x)break;A=K}var U={isMatch:L>=0,score:Math.max(.001,C)};if(S){var V=N(_,d);V.length?g&&(U.indices=V):U.isMatch=!1}return U}(e,n,i,{location:c+o,distance:a,threshold:s,findAllMatches:u,minMatchCharLength:h,includeMatches:r,ignoreLocation:l}),p=y.isMatch,m=y.score,k=y.indices;p&&(g=!0),v+=m,p&&k&&(d=[].concat(f(d),f(k)))}));var y={isMatch:g,score:g?v/this.chunks.length:1};return g&&r&&(y.indices=d),y}}]),e}(),z=function(){function e(t){r(this,e),this.pattern=t}return o(e,[{key:"search",value:function(){}}],[{key:"isMultiMatch",value:function(e){return D(e,this.multiRegex)}},{key:"isSingleMatch",value:function(e){return D(e,this.singleRegex)}}]),e}();function D(e,t){var n=e.match(t);return n?n[1]:null}var K=function(e){a(n,e);var t=l(n);function n(e){return r(this,n),t.call(this,e)}return o(n,[{key:"search",value:function(e){var t=e===this.pattern;return{isMatch:t,score:t?0:1,indices:[0,this.pattern.length-1]}}}],[{key:"type",get:function(){return"exact"}},{key:"multiRegex",get:function(){return/^="(.*)"$/}},{key:"singleRegex",get:function(){return/^=(.*)$/}}]),n}(z),q=function(e){a(n,e);var t=l(n);function n(e){return r(this,n),t.call(this,e)}return o(n,[{key:"search",value:function(e){var t=-1===e.indexOf(this.pattern);return{isMatch:t,score:t?0:1,indices:[0,e.length-1]}}}],[{key:"type",get:function(){return"inverse-exact"}},{key:"multiRegex",get:function(){return/^!"(.*)"$/}},{key:"singleRegex",get:function(){return/^!(.*)$/}}]),n}(z),B=function(e){a(n,e);var t=l(n);function n(e){return r(this,n),t.call(this,e)}return o(n,[{key:"search",value:function(e){var t=e.startsWith(this.pattern);return{isMatch:t,score:t?0:1,indices:[0,this.pattern.length-1]}}}],[{key:"type",get:function(){return"prefix-exact"}},{key:"multiRegex",get:function(){return/^\^"(.*)"$/}},{key:"singleRegex",get:function(){return/^\^(.*)$/}}]),n}(z),J=function(e){a(n,e);var t=l(n);function n(e){return r(this,n),t.call(this,e)}return o(n,[{key:"search",value:function(e){var t=!e.startsWith(this.pattern);return{isMatch:t,score:t?0:1,indices:[0,e.length-1]}}}],[{key:"type",get:function(){return"inverse-prefix-exact"}},{key:"multiRegex",get:function(){return/^!\^"(.*)"$/}},{key:"singleRegex",get:function(){return/^!\^(.*)$/}}]),n}(z),U=function(e){a(n,e);var t=l(n);function n(e){return r(this,n),t.call(this,e)}return o(n,[{key:"search",value:function(e){var t=e.endsWith(this.pattern);return{isMatch:t,score:t?0:1,indices:[e.length-this.pattern.length,e.length-1]}}}],[{key:"type",get:function(){return"suffix-exact"}},{key:"multiRegex",get:function(){return/^"(.*)"\$$/}},{key:"singleRegex",get:function(){return/^(.*)\$$/}}]),n}(z),V=function(e){a(n,e);var t=l(n);function n(e){return r(this,n),t.call(this,e)}return o(n,[{key:"search",value:function(e){var t=!e.endsWith(this.pattern);return{isMatch:t,score:t?0:1,indices:[0,e.length-1]}}}],[{key:"type",get:function(){return"inverse-suffix-exact"}},{key:"multiRegex",get:function(){return/^!"(.*)"\$$/}},{key:"singleRegex",get:function(){return/^!(.*)\$$/}}]),n}(z),G=function(e){a(n,e);var t=l(n);function n(e){var i,o=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},c=o.location,a=void 0===c?I.location:c,s=o.threshold,u=void 0===s?I.threshold:s,h=o.distance,l=void 0===h?I.distance:h,f=o.includeMatches,d=void 0===f?I.includeMatches:f,v=o.findAllMatches,g=void 0===v?I.findAllMatches:v,y=o.minMatchCharLength,p=void 0===y?I.minMatchCharLength:y,m=o.isCaseSensitive,k=void 0===m?I.isCaseSensitive:m,M=o.ignoreLocation,b=void 0===M?I.ignoreLocation:M;return r(this,n),(i=t.call(this,e))._bitapSearch=new T(e,{location:a,threshold:u,distance:l,includeMatches:d,findAllMatches:g,minMatchCharLength:p,isCaseSensitive:k,ignoreLocation:b}),i}return o(n,[{key:"search",value:function(e){return this._bitapSearch.searchIn(e)}}],[{key:"type",get:function(){return"fuzzy"}},{key:"multiRegex",get:function(){return/^"(.*)"$/}},{key:"singleRegex",get:function(){return/^(.*)$/}}]),n}(z),H=function(e){a(n,e);var t=l(n);function n(e){return r(this,n),t.call(this,e)}return o(n,[{key:"search",value:function(e){for(var t,n=0,r=[],i=this.pattern.length;(t=e.indexOf(this.pattern,n))>-1;)n=t+i,r.push([t,n-1]);var o=!!r.length;return{isMatch:o,score:o?0:1,indices:r}}}],[{key:"type",get:function(){return"include"}},{key:"multiRegex",get:function(){return/^'"(.*)"$/}},{key:"singleRegex",get:function(){return/^'(.*)$/}}]),n}(z),Q=[K,H,B,J,V,U,q,G],X=Q.length,Y=/ +(?=(?:[^\"]*\"[^\"]*\")*[^\"]*$)/;function Z(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{};return e.split("|").map((function(e){for(var n=e.trim().split(Y).filter((function(e){return e&&!!e.trim()})),r=[],i=0,o=n.length;i1&&void 0!==arguments[1]?arguments[1]:{},i=n.isCaseSensitive,o=void 0===i?I.isCaseSensitive:i,c=n.includeMatches,a=void 0===c?I.includeMatches:c,s=n.minMatchCharLength,u=void 0===s?I.minMatchCharLength:s,h=n.ignoreLocation,l=void 0===h?I.ignoreLocation:h,f=n.findAllMatches,d=void 0===f?I.findAllMatches:f,v=n.location,g=void 0===v?I.location:v,y=n.threshold,p=void 0===y?I.threshold:y,m=n.distance,k=void 0===m?I.distance:m;r(this,e),this.query=null,this.options={isCaseSensitive:o,includeMatches:a,minMatchCharLength:u,findAllMatches:d,ignoreLocation:l,location:g,threshold:p,distance:k},this.pattern=o?t:t.toLowerCase(),this.query=Z(this.pattern,this.options)}return o(e,[{key:"searchIn",value:function(e){var t=this.query;if(!t)return{isMatch:!1,score:1};var n=this.options,r=n.includeMatches;e=n.isCaseSensitive?e:e.toLowerCase();for(var i=0,o=[],c=0,a=0,s=t.length;a-1&&(n.refIndex=e.idx),t.matches.push(n)}}))}function ve(e,t){t.score=e.score}function ge(e,t){var n=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{},r=n.includeMatches,i=void 0===r?I.includeMatches:r,o=n.includeScore,c=void 0===o?I.includeScore:o,a=[];return i&&a.push(de),c&&a.push(ve),e.map((function(e){var n=e.idx,r={item:t[n],refIndex:n};return a.length&&a.forEach((function(t){t(e,r)})),r}))}var ye=function(){function e(n){var i=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},o=arguments.length>2?arguments[2]:void 0;r(this,e),this.options=t(t({},I),i),this.options.useExtendedSearch,this._keyStore=new S(this.options.keys),this.setCollection(n,o)}return o(e,[{key:"setCollection",value:function(e,t){if(this._docs=e,t&&!(t instanceof $))throw new Error("Incorrect 'index' type");this._myIndex=t||F(this.options.keys,this._docs,{getFn:this.options.getFn,fieldNormWeight:this.options.fieldNormWeight})}},{key:"add",value:function(e){k(e)&&(this._docs.push(e),this._myIndex.add(e))}},{key:"remove",value:function(){for(var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:function(){return!1},t=[],n=0,r=this._docs.length;n1&&void 0!==arguments[1]?arguments[1]:{},n=t.limit,r=void 0===n?-1:n,i=this.options,o=i.includeMatches,c=i.includeScore,a=i.shouldSort,s=i.sortFn,u=i.ignoreFieldNorm,h=g(e)?g(this._docs[0])?this._searchStringList(e):this._searchObjectList(e):this._searchLogical(e);return fe(h,{ignoreFieldNorm:u}),a&&h.sort(s),y(r)&&r>-1&&(h=h.slice(0,r)),ge(h,this._docs,{includeMatches:o,includeScore:c})}},{key:"_searchStringList",value:function(e){var t=re(e,this.options),n=this._myIndex.records,r=[];return n.forEach((function(e){var n=e.v,i=e.i,o=e.n;if(k(n)){var c=t.searchIn(n),a=c.isMatch,s=c.score,u=c.indices;a&&r.push({item:n,idx:i,matches:[{score:s,value:n,norm:o,indices:u}]})}})),r}},{key:"_searchLogical",value:function(e){var t=this,n=function(e,t){var n=(arguments.length>2&&void 0!==arguments[2]?arguments[2]:{}).auto,r=void 0===n||n,i=function e(n){var i=Object.keys(n),o=ue(n);if(!o&&i.length>1&&!se(n))return e(le(n));if(he(n)){var c=o?n[ce]:i[0],a=o?n[ae]:n[c];if(!g(a))throw new Error(x(c));var s={keyId:j(c),pattern:a};return r&&(s.searcher=re(a,t)),s}var u={children:[],operator:i[0]};return i.forEach((function(t){var r=n[t];v(r)&&r.forEach((function(t){u.children.push(e(t))}))})),u};return se(e)||(e=le(e)),i(e)}(e,this.options),r=function e(n,r,i){if(!n.children){var o=n.keyId,c=n.searcher,a=t._findMatches({key:t._keyStore.get(o),value:t._myIndex.getValueForItemAtKeyId(r,o),searcher:c});return a&&a.length?[{idx:i,item:r,matches:a}]:[]}for(var s=[],u=0,h=n.children.length;u1&&void 0!==arguments[1]?arguments[1]:{},n=t.getFn,r=void 0===n?I.getFn:n,i=t.fieldNormWeight,o=void 0===i?I.fieldNormWeight:i,c=e.keys,a=e.records,s=new $({getFn:r,fieldNormWeight:o});return s.setKeys(c),s.setIndexRecords(a),s},ye.config=I,function(){ne.push.apply(ne,arguments)}(te),ye},"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define(t):(e="undefined"!=typeof globalThis?globalThis:e||self).Fuse=t(); \ No newline at end of file diff --git a/site_libs/quarto-search/quarto-search.js b/site_libs/quarto-search/quarto-search.js new file mode 100644 index 00000000..d788a958 --- /dev/null +++ b/site_libs/quarto-search/quarto-search.js @@ -0,0 +1,1290 @@ +const kQueryArg = "q"; +const kResultsArg = "show-results"; + +// If items don't provide a URL, then both the navigator and the onSelect +// function aren't called (and therefore, the default implementation is used) +// +// We're using this sentinel URL to signal to those handlers that this +// item is a more item (along with the type) and can be handled appropriately +const kItemTypeMoreHref = "0767FDFD-0422-4E5A-BC8A-3BE11E5BBA05"; + +window.document.addEventListener("DOMContentLoaded", function (_event) { + // Ensure that search is available on this page. If it isn't, + // should return early and not do anything + var searchEl = window.document.getElementById("quarto-search"); + if (!searchEl) return; + + const { autocomplete } = window["@algolia/autocomplete-js"]; + + let quartoSearchOptions = {}; + let language = {}; + const searchOptionEl = window.document.getElementById( + "quarto-search-options" + ); + if (searchOptionEl) { + const jsonStr = searchOptionEl.textContent; + quartoSearchOptions = JSON.parse(jsonStr); + language = quartoSearchOptions.language; + } + + // note the search mode + if (quartoSearchOptions.type === "overlay") { + searchEl.classList.add("type-overlay"); + } else { + searchEl.classList.add("type-textbox"); + } + + // Used to determine highlighting behavior for this page + // A `q` query param is expected when the user follows a search + // to this page + const currentUrl = new URL(window.location); + const query = currentUrl.searchParams.get(kQueryArg); + const showSearchResults = currentUrl.searchParams.get(kResultsArg); + const mainEl = window.document.querySelector("main"); + + // highlight matches on the page + if (query && mainEl) { + // perform any highlighting + highlight(escapeRegExp(query), mainEl); + + // fix up the URL to remove the q query param + const replacementUrl = new URL(window.location); + replacementUrl.searchParams.delete(kQueryArg); + window.history.replaceState({}, "", replacementUrl); + } + + // function to clear highlighting on the page when the search query changes + // (e.g. if the user edits the query or clears it) + let highlighting = true; + const resetHighlighting = (searchTerm) => { + if (mainEl && highlighting && query && searchTerm !== query) { + clearHighlight(query, mainEl); + highlighting = false; + } + }; + + // Clear search highlighting when the user scrolls sufficiently + const resetFn = () => { + resetHighlighting(""); + window.removeEventListener("quarto-hrChanged", resetFn); + window.removeEventListener("quarto-sectionChanged", resetFn); + }; + + // Register this event after the initial scrolling and settling of events + // on the page + window.addEventListener("quarto-hrChanged", resetFn); + window.addEventListener("quarto-sectionChanged", resetFn); + + // Responsively switch to overlay mode if the search is present on the navbar + // Note that switching the sidebar to overlay mode requires more coordinate (not just + // the media query since we generate different HTML for sidebar overlays than we do + // for sidebar input UI) + const detachedMediaQuery = + quartoSearchOptions.type === "overlay" ? "all" : "(max-width: 991px)"; + + // If configured, include the analytics client to send insights + const plugins = configurePlugins(quartoSearchOptions); + + let lastState = null; + const { setIsOpen, setQuery, setCollections } = autocomplete({ + container: searchEl, + detachedMediaQuery: detachedMediaQuery, + defaultActiveItemId: 0, + panelContainer: "#quarto-search-results", + panelPlacement: quartoSearchOptions["panel-placement"], + debug: false, + openOnFocus: true, + plugins, + classNames: { + form: "d-flex", + }, + placeholder: language["search-text-placeholder"], + translations: { + clearButtonTitle: language["search-clear-button-title"], + detachedCancelButtonText: language["search-detached-cancel-button-title"], + submitButtonTitle: language["search-submit-button-title"], + }, + initialState: { + query, + }, + getItemUrl({ item }) { + return item.href; + }, + onStateChange({ state }) { + // If this is a file URL, note that + + // Perhaps reset highlighting + resetHighlighting(state.query); + + // If the panel just opened, ensure the panel is positioned properly + if (state.isOpen) { + if (lastState && !lastState.isOpen) { + setTimeout(() => { + positionPanel(quartoSearchOptions["panel-placement"]); + }, 150); + } + } + + // Perhaps show the copy link + showCopyLink(state.query, quartoSearchOptions); + + lastState = state; + }, + reshape({ sources, state }) { + return sources.map((source) => { + try { + const items = source.getItems(); + + // Validate the items + validateItems(items); + + // group the items by document + const groupedItems = new Map(); + items.forEach((item) => { + const hrefParts = item.href.split("#"); + const baseHref = hrefParts[0]; + const isDocumentItem = hrefParts.length === 1; + + const items = groupedItems.get(baseHref); + if (!items) { + groupedItems.set(baseHref, [item]); + } else { + // If the href for this item matches the document + // exactly, place this item first as it is the item that represents + // the document itself + if (isDocumentItem) { + items.unshift(item); + } else { + items.push(item); + } + groupedItems.set(baseHref, items); + } + }); + + const reshapedItems = []; + let count = 1; + for (const [_key, value] of groupedItems) { + const firstItem = value[0]; + reshapedItems.push({ + ...firstItem, + type: kItemTypeDoc, + }); + + const collapseMatches = quartoSearchOptions["collapse-after"]; + const collapseCount = + typeof collapseMatches === "number" ? collapseMatches : 1; + + if (value.length > 1) { + const target = `search-more-${count}`; + const isExpanded = + state.context.expanded && + state.context.expanded.includes(target); + + const remainingCount = value.length - collapseCount; + + for (let i = 1; i < value.length; i++) { + if (collapseMatches && i === collapseCount) { + reshapedItems.push({ + target, + title: isExpanded + ? language["search-hide-matches-text"] + : remainingCount === 1 + ? `${remainingCount} ${language["search-more-match-text"]}` + : `${remainingCount} ${language["search-more-matches-text"]}`, + type: kItemTypeMore, + href: kItemTypeMoreHref, + }); + } + + if (isExpanded || !collapseMatches || i < collapseCount) { + reshapedItems.push({ + ...value[i], + type: kItemTypeItem, + target, + }); + } + } + } + count += 1; + } + + return { + ...source, + getItems() { + return reshapedItems; + }, + }; + } catch (error) { + // Some form of error occurred + return { + ...source, + getItems() { + return [ + { + title: error.name || "An Error Occurred While Searching", + text: + error.message || + "An unknown error occurred while attempting to perform the requested search.", + type: kItemTypeError, + }, + ]; + }, + }; + } + }); + }, + navigator: { + navigate({ itemUrl }) { + if (itemUrl !== offsetURL(kItemTypeMoreHref)) { + window.location.assign(itemUrl); + } + }, + navigateNewTab({ itemUrl }) { + if (itemUrl !== offsetURL(kItemTypeMoreHref)) { + const windowReference = window.open(itemUrl, "_blank", "noopener"); + if (windowReference) { + windowReference.focus(); + } + } + }, + navigateNewWindow({ itemUrl }) { + if (itemUrl !== offsetURL(kItemTypeMoreHref)) { + window.open(itemUrl, "_blank", "noopener"); + } + }, + }, + getSources({ state, setContext, setActiveItemId, refresh }) { + return [ + { + sourceId: "documents", + getItemUrl({ item }) { + if (item.href) { + return offsetURL(item.href); + } else { + return undefined; + } + }, + onSelect({ + item, + state, + setContext, + setIsOpen, + setActiveItemId, + refresh, + }) { + if (item.type === kItemTypeMore) { + toggleExpanded(item, state, setContext, setActiveItemId, refresh); + + // Toggle more + setIsOpen(true); + } + }, + getItems({ query }) { + if (query === null || query === "") { + return []; + } + + const limit = quartoSearchOptions.limit; + if (quartoSearchOptions.algolia) { + return algoliaSearch(query, limit, quartoSearchOptions.algolia); + } else { + // Fuse search options + const fuseSearchOptions = { + isCaseSensitive: false, + shouldSort: true, + minMatchCharLength: 2, + limit: limit, + }; + + return readSearchData().then(function (fuse) { + return fuseSearch(query, fuse, fuseSearchOptions); + }); + } + }, + templates: { + noResults({ createElement }) { + const hasQuery = lastState.query; + + return createElement( + "div", + { + class: `quarto-search-no-results${ + hasQuery ? "" : " no-query" + }`, + }, + language["search-no-results-text"] + ); + }, + header({ items, createElement }) { + // count the documents + const count = items.filter((item) => { + return item.type === kItemTypeDoc; + }).length; + + if (count > 0) { + return createElement( + "div", + { class: "search-result-header" }, + `${count} ${language["search-matching-documents-text"]}` + ); + } else { + return createElement( + "div", + { class: "search-result-header-no-results" }, + `` + ); + } + }, + footer({ _items, createElement }) { + if ( + quartoSearchOptions.algolia && + quartoSearchOptions.algolia["show-logo"] + ) { + const libDir = quartoSearchOptions.algolia["libDir"]; + const logo = createElement("img", { + src: offsetURL( + `${libDir}/quarto-search/search-by-algolia.svg` + ), + class: "algolia-search-logo", + }); + return createElement( + "a", + { href: "http://www.algolia.com/" }, + logo + ); + } + }, + + item({ item, createElement }) { + return renderItem( + item, + createElement, + state, + setActiveItemId, + setContext, + refresh, + quartoSearchOptions + ); + }, + }, + }, + ]; + }, + }); + + window.quartoOpenSearch = () => { + setIsOpen(false); + setIsOpen(true); + focusSearchInput(); + }; + + document.addEventListener("keyup", (event) => { + const { key } = event; + const kbds = quartoSearchOptions["keyboard-shortcut"]; + const focusedEl = document.activeElement; + + const isFormElFocused = [ + "input", + "select", + "textarea", + "button", + "option", + ].find((tag) => { + return focusedEl.tagName.toLowerCase() === tag; + }); + + if ( + kbds && + kbds.includes(key) && + !isFormElFocused && + !document.activeElement.isContentEditable + ) { + event.preventDefault(); + window.quartoOpenSearch(); + } + }); + + // Remove the labeleledby attribute since it is pointing + // to a non-existent label + if (quartoSearchOptions.type === "overlay") { + const inputEl = window.document.querySelector( + "#quarto-search .aa-Autocomplete" + ); + if (inputEl) { + inputEl.removeAttribute("aria-labelledby"); + } + } + + function throttle(func, wait) { + let waiting = false; + return function () { + if (!waiting) { + func.apply(this, arguments); + waiting = true; + setTimeout(function () { + waiting = false; + }, wait); + } + }; + } + + // If the main document scrolls dismiss the search results + // (otherwise, since they're floating in the document they can scroll with the document) + window.document.body.onscroll = throttle(() => { + // Only do this if we're not detached + // Bug #7117 + // This will happen when the keyboard is shown on ios (resulting in a scroll) + // which then closed the search UI + if (!window.matchMedia(detachedMediaQuery).matches) { + setIsOpen(false); + } + }, 50); + + if (showSearchResults) { + setIsOpen(true); + focusSearchInput(); + } +}); + +function configurePlugins(quartoSearchOptions) { + const autocompletePlugins = []; + const algoliaOptions = quartoSearchOptions.algolia; + if ( + algoliaOptions && + algoliaOptions["analytics-events"] && + algoliaOptions["search-only-api-key"] && + algoliaOptions["application-id"] + ) { + const apiKey = algoliaOptions["search-only-api-key"]; + const appId = algoliaOptions["application-id"]; + + // Aloglia insights may not be loaded because they require cookie consent + // Use deferred loading so events will start being recorded when/if consent + // is granted. + const algoliaInsightsDeferredPlugin = deferredLoadPlugin(() => { + if ( + window.aa && + window["@algolia/autocomplete-plugin-algolia-insights"] + ) { + window.aa("init", { + appId, + apiKey, + useCookie: true, + }); + + const { createAlgoliaInsightsPlugin } = + window["@algolia/autocomplete-plugin-algolia-insights"]; + // Register the insights client + const algoliaInsightsPlugin = createAlgoliaInsightsPlugin({ + insightsClient: window.aa, + onItemsChange({ insights, insightsEvents }) { + const events = insightsEvents.flatMap((event) => { + // This API limits the number of items per event to 20 + const chunkSize = 20; + const itemChunks = []; + const eventItems = event.items; + for (let i = 0; i < eventItems.length; i += chunkSize) { + itemChunks.push(eventItems.slice(i, i + chunkSize)); + } + // Split the items into multiple events that can be sent + const events = itemChunks.map((items) => { + return { + ...event, + items, + }; + }); + return events; + }); + + for (const event of events) { + insights.viewedObjectIDs(event); + } + }, + }); + return algoliaInsightsPlugin; + } + }); + + // Add the plugin + autocompletePlugins.push(algoliaInsightsDeferredPlugin); + return autocompletePlugins; + } +} + +// For plugins that may not load immediately, create a wrapper +// plugin and forward events and plugin data once the plugin +// is initialized. This is useful for cases like cookie consent +// which may prevent the analytics insights event plugin from initializing +// immediately. +function deferredLoadPlugin(createPlugin) { + let plugin = undefined; + let subscribeObj = undefined; + const wrappedPlugin = () => { + if (!plugin && subscribeObj) { + plugin = createPlugin(); + if (plugin && plugin.subscribe) { + plugin.subscribe(subscribeObj); + } + } + return plugin; + }; + + return { + subscribe: (obj) => { + subscribeObj = obj; + }, + onStateChange: (obj) => { + const plugin = wrappedPlugin(); + if (plugin && plugin.onStateChange) { + plugin.onStateChange(obj); + } + }, + onSubmit: (obj) => { + const plugin = wrappedPlugin(); + if (plugin && plugin.onSubmit) { + plugin.onSubmit(obj); + } + }, + onReset: (obj) => { + const plugin = wrappedPlugin(); + if (plugin && plugin.onReset) { + plugin.onReset(obj); + } + }, + getSources: (obj) => { + const plugin = wrappedPlugin(); + if (plugin && plugin.getSources) { + return plugin.getSources(obj); + } else { + return Promise.resolve([]); + } + }, + data: (obj) => { + const plugin = wrappedPlugin(); + if (plugin && plugin.data) { + plugin.data(obj); + } + }, + }; +} + +function validateItems(items) { + // Validate the first item + if (items.length > 0) { + const item = items[0]; + const missingFields = []; + if (item.href == undefined) { + missingFields.push("href"); + } + if (!item.title == undefined) { + missingFields.push("title"); + } + if (!item.text == undefined) { + missingFields.push("text"); + } + + if (missingFields.length === 1) { + throw { + name: `Error: Search index is missing the ${missingFields[0]} field.`, + message: `The items being returned for this search do not include all the required fields. Please ensure that your index items include the ${missingFields[0]} field or use index-fields in your _quarto.yml file to specify the field names.`, + }; + } else if (missingFields.length > 1) { + const missingFieldList = missingFields + .map((field) => { + return `${field}`; + }) + .join(", "); + + throw { + name: `Error: Search index is missing the following fields: ${missingFieldList}.`, + message: `The items being returned for this search do not include all the required fields. Please ensure that your index items includes the following fields: ${missingFieldList}, or use index-fields in your _quarto.yml file to specify the field names.`, + }; + } + } +} + +let lastQuery = null; +function showCopyLink(query, options) { + const language = options.language; + lastQuery = query; + // Insert share icon + const inputSuffixEl = window.document.body.querySelector( + ".aa-Form .aa-InputWrapperSuffix" + ); + + if (inputSuffixEl) { + let copyButtonEl = window.document.body.querySelector( + ".aa-Form .aa-InputWrapperSuffix .aa-CopyButton" + ); + + if (copyButtonEl === null) { + copyButtonEl = window.document.createElement("button"); + copyButtonEl.setAttribute("class", "aa-CopyButton"); + copyButtonEl.setAttribute("type", "button"); + copyButtonEl.setAttribute("title", language["search-copy-link-title"]); + copyButtonEl.onmousedown = (e) => { + e.preventDefault(); + e.stopPropagation(); + }; + + const linkIcon = "bi-clipboard"; + const checkIcon = "bi-check2"; + + const shareIconEl = window.document.createElement("i"); + shareIconEl.setAttribute("class", `bi ${linkIcon}`); + copyButtonEl.appendChild(shareIconEl); + inputSuffixEl.prepend(copyButtonEl); + + const clipboard = new window.ClipboardJS(".aa-CopyButton", { + text: function (_trigger) { + const copyUrl = new URL(window.location); + copyUrl.searchParams.set(kQueryArg, lastQuery); + copyUrl.searchParams.set(kResultsArg, "1"); + return copyUrl.toString(); + }, + }); + clipboard.on("success", function (e) { + // Focus the input + + // button target + const button = e.trigger; + const icon = button.querySelector("i.bi"); + + // flash "checked" + icon.classList.add(checkIcon); + icon.classList.remove(linkIcon); + setTimeout(function () { + icon.classList.remove(checkIcon); + icon.classList.add(linkIcon); + }, 1000); + }); + } + + // If there is a query, show the link icon + if (copyButtonEl) { + if (lastQuery && options["copy-button"]) { + copyButtonEl.style.display = "flex"; + } else { + copyButtonEl.style.display = "none"; + } + } + } +} + +/* Search Index Handling */ +// create the index +var fuseIndex = undefined; +var shownWarning = false; + +// fuse index options +const kFuseIndexOptions = { + keys: [ + { name: "title", weight: 20 }, + { name: "section", weight: 20 }, + { name: "text", weight: 10 }, + ], + ignoreLocation: true, + threshold: 0.1, +}; + +async function readSearchData() { + // Initialize the search index on demand + if (fuseIndex === undefined) { + if (window.location.protocol === "file:" && !shownWarning) { + window.alert( + "Search requires JavaScript features disabled when running in file://... URLs. In order to use search, please run this document in a web server." + ); + shownWarning = true; + return; + } + const fuse = new window.Fuse([], kFuseIndexOptions); + + // fetch the main search.json + const response = await fetch(offsetURL("search.json")); + if (response.status == 200) { + return response.json().then(function (searchDocs) { + searchDocs.forEach(function (searchDoc) { + fuse.add(searchDoc); + }); + fuseIndex = fuse; + return fuseIndex; + }); + } else { + return Promise.reject( + new Error( + "Unexpected status from search index request: " + response.status + ) + ); + } + } + + return fuseIndex; +} + +function inputElement() { + return window.document.body.querySelector(".aa-Form .aa-Input"); +} + +function focusSearchInput() { + setTimeout(() => { + const inputEl = inputElement(); + if (inputEl) { + inputEl.focus(); + } + }, 50); +} + +/* Panels */ +const kItemTypeDoc = "document"; +const kItemTypeMore = "document-more"; +const kItemTypeItem = "document-item"; +const kItemTypeError = "error"; + +function renderItem( + item, + createElement, + state, + setActiveItemId, + setContext, + refresh, + quartoSearchOptions +) { + switch (item.type) { + case kItemTypeDoc: + return createDocumentCard( + createElement, + "file-richtext", + item.title, + item.section, + item.text, + item.href, + item.crumbs, + quartoSearchOptions + ); + case kItemTypeMore: + return createMoreCard( + createElement, + item, + state, + setActiveItemId, + setContext, + refresh + ); + case kItemTypeItem: + return createSectionCard( + createElement, + item.section, + item.text, + item.href + ); + case kItemTypeError: + return createErrorCard(createElement, item.title, item.text); + default: + return undefined; + } +} + +function createDocumentCard( + createElement, + icon, + title, + section, + text, + href, + crumbs, + quartoSearchOptions +) { + const iconEl = createElement("i", { + class: `bi bi-${icon} search-result-icon`, + }); + const titleEl = createElement("p", { class: "search-result-title" }, title); + const titleContents = [iconEl, titleEl]; + const showParent = quartoSearchOptions["show-item-context"]; + if (crumbs && showParent) { + let crumbsOut = undefined; + const crumbClz = ["search-result-crumbs"]; + if (showParent === "root") { + crumbsOut = crumbs.length > 1 ? crumbs[0] : undefined; + } else if (showParent === "parent") { + crumbsOut = crumbs.length > 1 ? crumbs[crumbs.length - 2] : undefined; + } else { + crumbsOut = crumbs.length > 1 ? crumbs.join(" > ") : undefined; + crumbClz.push("search-result-crumbs-wrap"); + } + + const crumbEl = createElement( + "p", + { class: crumbClz.join(" ") }, + crumbsOut + ); + titleContents.push(crumbEl); + } + + const titleContainerEl = createElement( + "div", + { class: "search-result-title-container" }, + titleContents + ); + + const textEls = []; + if (section) { + const sectionEl = createElement( + "p", + { class: "search-result-section" }, + section + ); + textEls.push(sectionEl); + } + const descEl = createElement("p", { + class: "search-result-text", + dangerouslySetInnerHTML: { + __html: text, + }, + }); + textEls.push(descEl); + + const textContainerEl = createElement( + "div", + { class: "search-result-text-container" }, + textEls + ); + + const containerEl = createElement( + "div", + { + class: "search-result-container", + }, + [titleContainerEl, textContainerEl] + ); + + const linkEl = createElement( + "a", + { + href: offsetURL(href), + class: "search-result-link", + }, + containerEl + ); + + const classes = ["search-result-doc", "search-item"]; + if (!section) { + classes.push("document-selectable"); + } + + return createElement( + "div", + { + class: classes.join(" "), + }, + linkEl + ); +} + +function createMoreCard( + createElement, + item, + state, + setActiveItemId, + setContext, + refresh +) { + const moreCardEl = createElement( + "div", + { + class: "search-result-more search-item", + onClick: (e) => { + // Handle expanding the sections by adding the expanded + // section to the list of expanded sections + toggleExpanded(item, state, setContext, setActiveItemId, refresh); + e.stopPropagation(); + }, + }, + item.title + ); + + return moreCardEl; +} + +function toggleExpanded(item, state, setContext, setActiveItemId, refresh) { + const expanded = state.context.expanded || []; + if (expanded.includes(item.target)) { + setContext({ + expanded: expanded.filter((target) => target !== item.target), + }); + } else { + setContext({ expanded: [...expanded, item.target] }); + } + + refresh(); + setActiveItemId(item.__autocomplete_id); +} + +function createSectionCard(createElement, section, text, href) { + const sectionEl = createSection(createElement, section, text, href); + return createElement( + "div", + { + class: "search-result-doc-section search-item", + }, + sectionEl + ); +} + +function createSection(createElement, title, text, href) { + const descEl = createElement("p", { + class: "search-result-text", + dangerouslySetInnerHTML: { + __html: text, + }, + }); + + const titleEl = createElement("p", { class: "search-result-section" }, title); + const linkEl = createElement( + "a", + { + href: offsetURL(href), + class: "search-result-link", + }, + [titleEl, descEl] + ); + return linkEl; +} + +function createErrorCard(createElement, title, text) { + const descEl = createElement("p", { + class: "search-error-text", + dangerouslySetInnerHTML: { + __html: text, + }, + }); + + const titleEl = createElement("p", { + class: "search-error-title", + dangerouslySetInnerHTML: { + __html: ` ${title}`, + }, + }); + const errorEl = createElement("div", { class: "search-error" }, [ + titleEl, + descEl, + ]); + return errorEl; +} + +function positionPanel(pos) { + const panelEl = window.document.querySelector( + "#quarto-search-results .aa-Panel" + ); + const inputEl = window.document.querySelector( + "#quarto-search .aa-Autocomplete" + ); + + if (panelEl && inputEl) { + panelEl.style.top = `${Math.round(panelEl.offsetTop)}px`; + if (pos === "start") { + panelEl.style.left = `${Math.round(inputEl.left)}px`; + } else { + panelEl.style.right = `${Math.round(inputEl.offsetRight)}px`; + } + } +} + +/* Highlighting */ +// highlighting functions +function highlightMatch(query, text) { + if (text) { + const start = text.toLowerCase().indexOf(query.toLowerCase()); + if (start !== -1) { + const startMark = ""; + const endMark = ""; + + const end = start + query.length; + text = + text.slice(0, start) + + startMark + + text.slice(start, end) + + endMark + + text.slice(end); + const startInfo = clipStart(text, start); + const endInfo = clipEnd( + text, + startInfo.position + startMark.length + endMark.length + ); + text = + startInfo.prefix + + text.slice(startInfo.position, endInfo.position) + + endInfo.suffix; + + return text; + } else { + return text; + } + } else { + return text; + } +} + +function clipStart(text, pos) { + const clipStart = pos - 50; + if (clipStart < 0) { + // This will just return the start of the string + return { + position: 0, + prefix: "", + }; + } else { + // We're clipping before the start of the string, walk backwards to the first space. + const spacePos = findSpace(text, pos, -1); + return { + position: spacePos.position, + prefix: "", + }; + } +} + +function clipEnd(text, pos) { + const clipEnd = pos + 200; + if (clipEnd > text.length) { + return { + position: text.length, + suffix: "", + }; + } else { + const spacePos = findSpace(text, clipEnd, 1); + return { + position: spacePos.position, + suffix: spacePos.clipped ? "…" : "", + }; + } +} + +function findSpace(text, start, step) { + let stepPos = start; + while (stepPos > -1 && stepPos < text.length) { + const char = text[stepPos]; + if (char === " " || char === "," || char === ":") { + return { + position: step === 1 ? stepPos : stepPos - step, + clipped: stepPos > 1 && stepPos < text.length, + }; + } + stepPos = stepPos + step; + } + + return { + position: stepPos - step, + clipped: false, + }; +} + +// removes highlighting as implemented by the mark tag +function clearHighlight(searchterm, el) { + const childNodes = el.childNodes; + for (let i = childNodes.length - 1; i >= 0; i--) { + const node = childNodes[i]; + if (node.nodeType === Node.ELEMENT_NODE) { + if ( + node.tagName === "MARK" && + node.innerText.toLowerCase() === searchterm.toLowerCase() + ) { + el.replaceChild(document.createTextNode(node.innerText), node); + } else { + clearHighlight(searchterm, node); + } + } + } +} + +function escapeRegExp(string) { + return string.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"); // $& means the whole matched string +} + +// highlight matches +function highlight(term, el) { + const termRegex = new RegExp(term, "ig"); + const childNodes = el.childNodes; + + // walk back to front avoid mutating elements in front of us + for (let i = childNodes.length - 1; i >= 0; i--) { + const node = childNodes[i]; + + if (node.nodeType === Node.TEXT_NODE) { + // Search text nodes for text to highlight + const text = node.nodeValue; + + let startIndex = 0; + let matchIndex = text.search(termRegex); + if (matchIndex > -1) { + const markFragment = document.createDocumentFragment(); + while (matchIndex > -1) { + const prefix = text.slice(startIndex, matchIndex); + markFragment.appendChild(document.createTextNode(prefix)); + + const mark = document.createElement("mark"); + mark.appendChild( + document.createTextNode( + text.slice(matchIndex, matchIndex + term.length) + ) + ); + markFragment.appendChild(mark); + + startIndex = matchIndex + term.length; + matchIndex = text.slice(startIndex).search(new RegExp(term, "ig")); + if (matchIndex > -1) { + matchIndex = startIndex + matchIndex; + } + } + if (startIndex < text.length) { + markFragment.appendChild( + document.createTextNode(text.slice(startIndex, text.length)) + ); + } + + el.replaceChild(markFragment, node); + } + } else if (node.nodeType === Node.ELEMENT_NODE) { + // recurse through elements + highlight(term, node); + } + } +} + +/* Link Handling */ +// get the offset from this page for a given site root relative url +function offsetURL(url) { + var offset = getMeta("quarto:offset"); + return offset ? offset + url : url; +} + +// read a meta tag value +function getMeta(metaName) { + var metas = window.document.getElementsByTagName("meta"); + for (let i = 0; i < metas.length; i++) { + if (metas[i].getAttribute("name") === metaName) { + return metas[i].getAttribute("content"); + } + } + return ""; +} + +function algoliaSearch(query, limit, algoliaOptions) { + const { getAlgoliaResults } = window["@algolia/autocomplete-preset-algolia"]; + + const applicationId = algoliaOptions["application-id"]; + const searchOnlyApiKey = algoliaOptions["search-only-api-key"]; + const indexName = algoliaOptions["index-name"]; + const indexFields = algoliaOptions["index-fields"]; + const searchClient = window.algoliasearch(applicationId, searchOnlyApiKey); + const searchParams = algoliaOptions["params"]; + const searchAnalytics = !!algoliaOptions["analytics-events"]; + + return getAlgoliaResults({ + searchClient, + queries: [ + { + indexName: indexName, + query, + params: { + hitsPerPage: limit, + clickAnalytics: searchAnalytics, + ...searchParams, + }, + }, + ], + transformResponse: (response) => { + if (!indexFields) { + return response.hits.map((hit) => { + return hit.map((item) => { + return { + ...item, + text: highlightMatch(query, item.text), + }; + }); + }); + } else { + const remappedHits = response.hits.map((hit) => { + return hit.map((item) => { + const newItem = { ...item }; + ["href", "section", "title", "text", "crumbs"].forEach( + (keyName) => { + const mappedName = indexFields[keyName]; + if ( + mappedName && + item[mappedName] !== undefined && + mappedName !== keyName + ) { + newItem[keyName] = item[mappedName]; + delete newItem[mappedName]; + } + } + ); + newItem.text = highlightMatch(query, newItem.text); + return newItem; + }); + }); + return remappedHits; + } + }, + }); +} + +let subSearchTerm = undefined; +let subSearchFuse = undefined; +const kFuseMaxWait = 125; + +async function fuseSearch(query, fuse, fuseOptions) { + let index = fuse; + // Fuse.js using the Bitap algorithm for text matching which runs in + // O(nm) time (no matter the structure of the text). In our case this + // means that long search terms mixed with large index gets very slow + // + // This injects a subIndex that will be used once the terms get long enough + // Usually making this subindex is cheap since there will typically be + // a subset of results matching the existing query + if (subSearchFuse !== undefined && query.startsWith(subSearchTerm)) { + // Use the existing subSearchFuse + index = subSearchFuse; + } else if (subSearchFuse !== undefined) { + // The term changed, discard the existing fuse + subSearchFuse = undefined; + subSearchTerm = undefined; + } + + // Search using the active fuse + const then = performance.now(); + const resultsRaw = await index.search(query, fuseOptions); + const now = performance.now(); + + const results = resultsRaw.map((result) => { + const addParam = (url, name, value) => { + const anchorParts = url.split("#"); + const baseUrl = anchorParts[0]; + const sep = baseUrl.search("\\?") > 0 ? "&" : "?"; + anchorParts[0] = baseUrl + sep + name + "=" + value; + return anchorParts.join("#"); + }; + + return { + title: result.item.title, + section: result.item.section, + href: addParam(result.item.href, kQueryArg, query), + text: highlightMatch(query, result.item.text), + crumbs: result.item.crumbs, + }; + }); + + // If we don't have a subfuse and the query is long enough, go ahead + // and create a subfuse to use for subsequent queries + if ( + now - then > kFuseMaxWait && + subSearchFuse === undefined && + resultsRaw.length < fuseOptions.limit + ) { + subSearchTerm = query; + subSearchFuse = new window.Fuse([], kFuseIndexOptions); + resultsRaw.forEach((rr) => { + subSearchFuse.add(rr.item); + }); + } + return results; +} diff --git a/sitemap.xml b/sitemap.xml new file mode 100644 index 00000000..dc46d696 --- /dev/null +++ b/sitemap.xml @@ -0,0 +1,103 @@ + + + + https://AnswerDotAI.github.io/fasthtml/ref/live_reload.html + 2025-01-12T17:38:53.788Z + + + https://AnswerDotAI.github.io/fasthtml/ref/defining_xt_component.html + 2025-01-12T17:38:53.764Z + + + https://AnswerDotAI.github.io/fasthtml/explains/minidataapi.html + 2025-01-12T17:38:53.772Z + + + https://AnswerDotAI.github.io/fasthtml/explains/websockets.html + 2025-01-12T17:38:53.708Z + + + https://AnswerDotAI.github.io/fasthtml/explains/oauth.html + 2025-01-12T17:38:53.664Z + + + https://AnswerDotAI.github.io/fasthtml/api/components.html + 2025-01-12T17:38:53.664Z + + + https://AnswerDotAI.github.io/fasthtml/api/jupyter.html + 2025-01-12T17:38:53.480Z + + + https://AnswerDotAI.github.io/fasthtml/api/cli.html + 2025-01-12T17:38:53.208Z + + + https://AnswerDotAI.github.io/fasthtml/api/core.html + 2025-01-12T17:38:53.712Z + + + https://AnswerDotAI.github.io/fasthtml/unpublished/tutorial_for_web_devs.html + 2025-01-12T17:38:53.108Z + + + https://AnswerDotAI.github.io/fasthtml/tutorials/quickstart_for_web_devs.html + 2025-01-12T17:38:53.108Z + + + https://AnswerDotAI.github.io/fasthtml/tutorials/index.html + 2025-01-12T17:38:31.112Z + + + https://AnswerDotAI.github.io/fasthtml/tutorials/e2e.html + 2025-01-12T17:38:53.056Z + + + https://AnswerDotAI.github.io/fasthtml/tutorials/by_example.html + 2025-01-12T17:38:53.096Z + + + https://AnswerDotAI.github.io/fasthtml/tutorials/jupyter_and_fasthtml.html + 2025-01-12T17:38:53.072Z + + + https://AnswerDotAI.github.io/fasthtml/index.html + 2025-01-12T17:38:53.048Z + + + https://AnswerDotAI.github.io/fasthtml/api/oauth.html + 2025-01-12T17:38:53.384Z + + + https://AnswerDotAI.github.io/fasthtml/api/js.html + 2025-01-12T17:38:53.348Z + + + https://AnswerDotAI.github.io/fasthtml/api/svg.html + 2025-01-12T17:38:53.620Z + + + https://AnswerDotAI.github.io/fasthtml/api/pico.html + 2025-01-12T17:38:53.500Z + + + https://AnswerDotAI.github.io/fasthtml/api/xtend.html + 2025-01-12T17:38:53.728Z + + + https://AnswerDotAI.github.io/fasthtml/explains/routes.html + 2025-01-12T17:38:53.696Z + + + https://AnswerDotAI.github.io/fasthtml/explains/faq.html + 2025-01-12T17:38:53.732Z + + + https://AnswerDotAI.github.io/fasthtml/explains/explaining_xt_components.html + 2025-01-12T17:38:53.760Z + + + https://AnswerDotAI.github.io/fasthtml/ref/handlers.html + 2025-01-12T17:38:53.792Z + + diff --git a/styles.css b/styles.css new file mode 100644 index 00000000..e5ea544a --- /dev/null +++ b/styles.css @@ -0,0 +1,47 @@ +.cell { margin-bottom: 1rem; } +.cell > .sourceCode { margin-bottom: 0; } +.cell-output > pre { margin-bottom: 0; } + +.cell-output > pre, .cell-output > .sourceCode > pre, .cell-output-stdout > pre { + margin-left: 0.8rem; + margin-top: 0; + background: none; + border-left: 2px solid lightsalmon; + border-top-left-radius: 0; + border-top-right-radius: 0; +} + +.cell-output > .sourceCode { border: none; } + +.cell-output > .sourceCode { + background: none; + margin-top: 0; +} + +div.description { + padding-left: 2px; + padding-top: 5px; + font-size: 1.25rem; + color: rgba(0, 0, 0, 0.60); + opacity: 70%; +} + + +div.sidebar-item-container .active { + background-color: #E8E8FC; + color: #000; +} + +div.sidebar-item-container a { + color: #000; + padding: 2px 4px; + border-radius: 8px; + font-size: 1rem; +} + +div.sidebar-item-container a.text-start { + padding-bottom: 0px; +} + +.navbar-container { max-width: 1282px; } + diff --git a/tutorials/by_example.html b/tutorials/by_example.html new file mode 100644 index 00000000..9053edd6 --- /dev/null +++ b/tutorials/by_example.html @@ -0,0 +1,1902 @@ + + + + + + + + + + +FastHTML By Example – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

FastHTML By Example

+
+ +
+
+ An introduction to FastHTML from the ground up, with four complete examples +
+
+ + +
+ + + + +
+ + + +
+ + + +

This tutorial provides an alternate introduction to FastHTML by building out example applications. We also illustrate how to use FastHTML foundations to create custom web apps. Finally, this document serves as minimal context for a LLM to turn it into a FastHTML assistant.

+

Let’s get started.

+
+

FastHTML Basics

+

FastHTML is just Python. You can install it with pip install python-fasthtml. Extensions/components built for it can likewise be distributed via PyPI or as simple Python files.

+

The core usage of FastHTML is to define routes, and then to define what to do at each route. This is similar to the FastAPI web framework (in fact we implemented much of the functionality to match the FastAPI usage examples), but where FastAPI focuses on returning JSON data to build APIs, FastHTML focuses on returning HTML data.

+

Here’s a simple FastHTML app that returns a “Hello, World” message:

+
+
from fasthtml.common import FastHTML, serve
+
+app = FastHTML()
+
+@app.get("/")
+def home():
+    return "<h1>Hello, World</h1>"
+
+serve()
+
+

To run this app, place it in a file, say app.py, and then run it with python app.py.

+
INFO:     Will watch for changes in these directories: ['/home/jonathan/fasthtml-example']
+INFO:     Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit)
+INFO:     Started reloader process [871942] using WatchFiles
+INFO:     Started server process [871945]
+INFO:     Waiting for application startup.
+INFO:     Application startup complete.
+

If you navigate to http://127.0.0.1:8000 in a browser, you’ll see your “Hello, World”. If you edit the app.py file and save it, the server will reload and you’ll see the updated message when you refresh the page in your browser.

+
+
+

Constructing HTML

+

Notice we wrote some HTML in the previous example. We don’t want to do that! Some web frameworks require that you learn HTML, CSS, JavaScript AND some templating language AND python. We want to do as much as possible with just one language. Fortunately, the Python module fastcore.xml has all we need for constructing HTML from Python, and FastHTML includes all the tags you need to get started. For example:

+
+
from fasthtml.common import *
+page = Html(
+    Head(Title('Some page')),
+    Body(Div('Some text, ', A('A link', href='https://example.com'), Img(src="https://placehold.co/200"), cls='myclass')))
+print(to_xml(page))
+
+
<!doctype html></!doctype>
+
+<html>
+  <head>
+    <title>Some page</title>
+  </head>
+  <body>
+    <div class="myclass">
+Some text, 
+      <a href="https://example.com">A link</a>
+      <img src="https://placehold.co/200">
+    </div>
+  </body>
+</html>
+
+
+
+
+
show(page)
+
+ + + + + Some page + + +
+Some text, + A link + +
+ + +
+
+

If that import * worries you, you can always import only the tags you need.

+

FastHTML is smart enough to know about fastcore.xml, and so you don’t need to use the to_xml function to convert your FT objects to HTML. You can just return them as you would any other Python object. For example, if we modify our previous example to use fastcore.xml, we can return an FT object directly:

+
+
from fasthtml.common import *
+app = FastHTML()
+
+@app.get("/")
+def home():
+    page = Html(
+        Head(Title('Some page')),
+        Body(Div('Some text, ', A('A link', href='https://example.com'), Img(src="https://placehold.co/200"), cls='myclass')))
+    return page
+
+serve()
+
+

This will render the HTML in the browser.

+

For debugging, you can right-click on the rendered HTML in the browser and select “Inspect” to see the underlying HTML that was generated. There you’ll also find the ‘network’ tab, which shows you the requests that were made to render the page. Refresh and look for the request to 127.0.0.1 - and you’ll see it’s just a GET request to /, and the response body is the HTML you just returned.

+
+
+
+ +
+
+Live Reloading +
+
+
+

You can also enable live reloading so you don’t have to manually refresh your browser to view updates.

+
+
+

You can also use Starlette’s TestClient to try it out in a notebook:

+
+
from starlette.testclient import TestClient
+client = TestClient(app)
+r = client.get("/")
+print(r.text)
+
+
<html>
+  <head><title>Some page</title>
+</head>
+  <body><div class="myclass">
+Some text, 
+  <a href="https://example.com">A link</a>
+  <img src="https://placehold.co/200">
+</div>
+</body>
+</html>
+
+
+
+

FastHTML wraps things in an Html tag if you don’t do it yourself (unless the request comes from htmx, in which case you get the element directly). See FT objects and HTML for more on creating custom components or adding HTML rendering to existing Python objects. To give the page a non-default title, return a Title before your main content:

+
+
app = FastHTML()
+
+@app.get("/")
+def home():
+    return Title("Page Demo"), Div(H1('Hello, World'), P('Some text'), P('Some more text'))
+
+client = TestClient(app)
+print(client.get("/").text)
+
+
<!doctype html></!doctype>
+
+<html>
+  <head>
+    <title>Page Demo</title>
+    <meta charset="utf-8"></meta>
+    <meta name="viewport" content="width=device-width, initial-scale=1, viewport-fit=cover"></meta>
+    <script src="https://unpkg.com/htmx.org@next/dist/htmx.min.js"></script>
+    <script src="https://cdn.jsdelivr.net/gh/answerdotai/surreal@1.3.0/surreal.js"></script>
+    <script src="https://cdn.jsdelivr.net/gh/gnat/css-scope-inline@main/script.js"></script>
+  </head>
+  <body>
+<div>
+  <h1>Hello, World</h1>
+  <p>Some text</p>
+  <p>Some more text</p>
+</div>
+  </body>
+</html>
+
+
+
+

We’ll use this pattern often in the examples to follow.

+
+
+

Defining Routes

+

The HTTP protocol defines a number of methods (‘verbs’) to send requests to a server. The most common are GET, POST, PUT, DELETE, and HEAD. We saw ‘GET’ in action before - when you navigate to a URL, you’re making a GET request to that URL. We can do different things on a route for different HTTP methods. For example:

+
@app.route("/", methods='get')
+def home():
+    return H1('Hello, World')
+
+@app.route("/", methods=['post', 'put'])
+def post_or_put():
+    return "got a POST or PUT request"
+

This says that when someone navigates to the root URL “/” (i.e. sends a GET request), they will see the big “Hello, World” heading. When someone submits a POST or PUT request to the same URL, the server should return the string “got a post or put request”.

+
+
+
+ +
+
+Test the POST request +
+
+
+

You can test the POST request with curl -X POST http://127.0.0.1:8000 -d "some data". This sends some data to the server, you should see the response “got a post or put request” printed in the terminal.

+
+
+

There are a few other ways you can specify the route+method - FastHTML has .get, .post, etc. as shorthand for route(..., methods=['get']), etc.

+
+
@app.get("/")
+def my_function():
+    return "Hello World from a GET request"
+
+

Or you can use the @rt decorator without a method but specify the method with the name of the function. For example:

+
+
rt = app.route
+
+@rt("/")
+def post():
+    return "Hello World from a POST request"
+
+
+
client.post("/").text
+
+
'Hello World from a POST request'
+
+
+

You’re welcome to pick whichever style you prefer. Using routes lets you show different content on different pages - ‘/home’, ‘/about’ and so on. You can also respond differently to different kinds of requests to the same route, as shown above. You can also pass data via the route:

+
+ +
+
+
+
@app.get("/greet/{nm}")
+def greet(nm:str):
+    return f"Good day to you, {nm}!"
+
+client.get("/greet/Dave").text
+
+
'Good day to you, Dave!'
+
+
+
+
+
+
@rt("/greet/{nm}")
+def get(nm:str):
+    return f"Good day to you, {nm}!"
+
+client.get("/greet/Dave").text
+
+
'Good day to you, Dave!'
+
+
+
+
+
+

More on this in the More on Routing and Request Parameters section, which goes deeper into the different ways to get information from a request.

+
+
+

Styling Basics

+

Plain HTML probably isn’t quite what you imagine when you visualize your beautiful web app. CSS is the go-to language for styling HTML. But again, we don’t want to learn extra languages unless we absolutely have to! Fortunately, there are ways to get much more visually appealing sites by relying on the hard work of others, using existing CSS libraries. One of our favourites is PicoCSS. A common way to add CSS files to web pages is to use a <link> tag inside your HTML header, like this:

+
<header>
+    ...
+    <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@picocss/pico@latest/css/pico.min.css">
+</header>
+

For convenience, FastHTML already defines a Pico component for you with picolink:

+
+
print(to_xml(picolink))
+
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@picocss/pico@latest/css/pico.min.css">
+
+<style>:root { --pico-font-size: 100%; }</style>
+
+
+
+
+
+
+ +
+
+Note +
+
+
+

picolink also includes a <style> tag, as we found that setting the font-size to 100% to be a good default. We show you how to override this below.

+
+
+

Since we typically want CSS styling on all pages of our app, FastHTML lets you define a shared HTML header with the hdrs argument as shown below:

+
+
from fasthtml.common import *
+1css = Style(':root {--pico-font-size:90%,--pico-font-family: Pacifico, cursive;}')
+2app = FastHTML(hdrs=(picolink, css))
+
+@app.route("/")
+def get():
+    return (Title("Hello World"), 
+3            Main(H1('Hello, World'), cls="container"))
+
+
+
1
+
+Custom styling to override the pico defaults +
+
2
+
+Define shared headers for all pages +
+
3
+
+As per the pico docs, we put all of our content inside a <main> tag with a class of container: +
+
+
+
+
+
+
+ +
+
+Returning Tuples +
+
+
+

We’re returning a tuple here (a title and the main page). Returning a tuple, list, FT object, or an object with a __ft__ method tells FastHTML to turn the main body into a full HTML page that includes the headers (including the pico link and our custom css) which we passed in. This only occurs if the request isn’t from HTMX (for HTMX requests we need only return the rendered components).

+
+
+

You can check out the Pico examples page to see how different elements will look. If everything is working, the page should now render nice text with our custom font, and it should respect the user’s light/dark mode preferences too.

+

If you want to override the default styles or add more custom CSS, you can do so by adding a <style> tag to the headers as shown above. So you are allowed to write CSS to your heart’s content - we just want to make sure you don’t necessarily have to! Later on we’ll see examples using other component libraries and tailwind css to do more fancy styling things, along with tips to get an LLM to write all those fiddly bits so you don’t have to.

+
+
+

Web Page -> Web App

+

Showing content is all well and good, but we typically expect a bit more interactivity from something calling itself a web app! So, let’s add a few different pages, and use a form to let users add messages to a list:

+
+
app = FastHTML()
+messages = ["This is a message, which will get rendered as a paragraph"]
+
+@app.get("/")
+def home():
+    return Main(H1('Messages'), 
+                *[P(msg) for msg in messages],
+                A("Link to Page 2 (to add messages)", href="/page2"))
+
+@app.get("/page2")
+def page2():
+    return Main(P("Add a message with the form below:"),
+                Form(Input(type="text", name="data"),
+                     Button("Submit"),
+                     action="/", method="post"))
+
+@app.post("/")
+def add_message(data:str):
+    messages.append(data)
+    return home()
+
+

We re-render the entire homepage to show the newly added message. This is fine, but modern web apps often don’t re-render the entire page, they just update a part of the page. In fact even very complicated applications are often implemented as ‘Single Page Apps’ (SPAs). This is where HTMX comes in.

+
+
+

HTMX

+

HTMX addresses some key limitations of HTML. In vanilla HTML, links can trigger a GET request to show a new page, and forms can send requests containing data to the server. A lot of ‘Web 1.0’ design revolved around ways to use these to do everything we wanted. But why should only some elements be allowed to trigger requests? And why should we refresh the entire page with the result each time one does? HTMX extends HTML to allow us to trigger requests from any element on all kinds of events, and to update a part of the page without refreshing the entire page. It’s a powerful tool for building modern web apps.

+

It does this by adding attributes to HTML tags to make them do things. For example, here’s a page with a counter and a button that increments it:

+
+
app = FastHTML()
+
+count = 0
+
+@app.get("/")
+def home():
+    return Title("Count Demo"), Main(
+        H1("Count Demo"),
+        P(f"Count is set to {count}", id="count"),
+        Button("Increment", hx_post="/increment", hx_target="#count", hx_swap="innerHTML")
+    )
+
+@app.post("/increment")
+def increment():
+    print("incrementing")
+    global count
+    count += 1
+    return f"Count is set to {count}"
+
+

The button triggers a POST request to /increment (since we set hx_post="/increment"), which increments the count and returns the new count. The hx_target attribute tells HTMX where to put the result. If no target is specified it replaces the element that triggered the request. The hx_swap attribute specifies how it adds the result to the page. Useful options are:

+
    +
  • innerHTML: Replace the target element’s content with the result.
  • +
  • outerHTML: Replace the target element with the result.
  • +
  • beforebegin: Insert the result before the target element.
  • +
  • beforeend: Insert the result inside the target element, after its last child.
  • +
  • afterbegin: Insert the result inside the target element, before its first child.
  • +
  • afterend: Insert the result after the target element.
  • +
+

You can also use an hx_swap of delete to delete the target element regardless of response, or of none to do nothing.

+

By default, requests are triggered by the “natural” event of an element - click in the case of a button (and most other elements). You can also specify different triggers, along with various modifiers - see the HTMX docs for more.

+

This pattern of having elements trigger requests that modify or replace other elements is a key part of the HTMX philosophy. It takes a little getting used to, but once mastered it is extremely powerful.

+
+

Replacing Elements Besides the Target

+

Sometimes having a single target is not enough, and we’d like to specify some additional elements to update or remove. In these cases, returning elements with an id that matches the element to be replaced and hx_swap_oob='true' will replace those elements too. We’ll use this in the next example to clear an input field when we submit a form.

+
+
+
+

Full Example #1 - ToDo App

+

The canonical demo web app! A TODO list. Rather than create yet another variant for this tutorial, we recommend starting with this video tutorial from Jeremy:

+
+
+
+

+
image.png
+
+
+

We’ve made a number of variants of this app - so in addition to the version shown in the video you can browse this series of examples with increasing complexity, the heavily-commented “idiomatic” version here, and the example linked from the FastHTML homepage.

+
+
+

Full Example #2 - Image Generation App

+

Let’s create an image generation app. We’d like to wrap a text-to-image model in a nice UI, where the user can type in a prompt and see a generated image appear. We’ll use a model hosted by Replicate to actually generate the images. Let’s start with the homepage, with a form to submit prompts and a div to hold the generated images:

+
# Main page
+@app.get("/")
+def get():
+    inp = Input(id="new-prompt", name="prompt", placeholder="Enter a prompt")
+    add = Form(Group(inp, Button("Generate")), hx_post="/", target_id='gen-list', hx_swap="afterbegin")
+    gen_list = Div(id='gen-list')
+    return Title('Image Generation Demo'), Main(H1('Magic Image Generation'), add, gen_list, cls='container')
+

Submitting the form will trigger a POST request to /, so next we need to generate an image and add it to the list. One problem: generating images is slow! We’ll start the generation in a separate thread, but this now surfaces a different problem: we want to update the UI right away, but our image will only be ready a few seconds later. This is a common pattern - think about how often you see a loading spinner online. We need a way to return a temporary bit of UI which will eventually be replaced by the final image. Here’s how we might do this:

+
def generation_preview(id):
+    if os.path.exists(f"gens/{id}.png"):
+        return Div(Img(src=f"/gens/{id}.png"), id=f'gen-{id}')
+    else:
+        return Div("Generating...", id=f'gen-{id}', 
+                   hx_post=f"/generations/{id}",
+                   hx_trigger='every 1s', hx_swap='outerHTML')
+    
+@app.post("/generations/{id}")
+def get(id:int): return generation_preview(id)
+
+@app.post("/")
+def post(prompt:str):
+    id = len(generations)
+    generate_and_save(prompt, id)
+    generations.append(prompt)
+    clear_input =  Input(id="new-prompt", name="prompt", placeholder="Enter a prompt", hx_swap_oob='true')
+    return generation_preview(id), clear_input
+
+@threaded
+def generate_and_save(prompt, id): ... 
+

The form sends the prompt to the / route, which starts the generation in a separate thread then returns two things:

+
    +
  • A generation preview element that will be added to the top of the gen-list div (since that is the target_id of the form which triggered the request)
  • +
  • An input field that will replace the form’s input field (that has the same id), using the hx_swap_oob=‘true’ trick. This clears the prompt field so the user can type another prompt.
  • +
+

The generation preview first returns a temporary “Generating…” message, which polls the /generations/{id} route every second. This is done by setting hx_post to the route and hx_trigger to ‘every 1s’. The /generations/{id} route returns the preview element every second until the image is ready, at which point it returns the final image. Since the final image replaces the temporary one (hx_swap=‘outerHTML’), the polling stops running and the generation preview is now complete.

+

This works nicely - the user can submit several prompts without having to wait for the first one to generate, and as the images become available they are added to the list. You can see the full code of this version here.

+
+

Again, with Style

+

The app is functional, but can be improved. The next version adds more stylish generation previews, lays out the images in a grid layout that is responsive to different screen sizes, and adds a database to track generations and make them persistent. The database part is very similar to the todo list example, so let’s just quickly look at how we add the nice grid layout. This is what the result looks like:

+
+
+

+
image.png
+
+
+

Step one was looking around for existing components. The Pico CSS library we’ve been using has a rudimentary grid but recommends using an alternative layout system. One of the options listed was Flexbox.

+

To use Flexbox you create a “row” with one or more elements. You can specify how wide things should be with a specific syntax in the class name. For example, col-xs-12 means a box that will take up 12 columns (out of 12 total) of the row on extra small screens, col-sm-6 means a column that will take up 6 columns of the row on small screens, and so on. So if you want four columns on large screens you would use col-lg-3 for each item (i.e. each item is using 3 columns out of 12).

+
<div class="row">
+    <div class="col-xs-12">
+        <div class="box">This takes up the full width</div>
+    </div>
+</div>
+

This was non-intuitive to me. Thankfully ChatGPT et al know web stuff quite well, and we can also experiment in a notebook to test things out:

+
+
grid = Html(
+    Link(rel="stylesheet", href="https://cdnjs.cloudflare.com/ajax/libs/flexboxgrid/6.3.1/flexboxgrid.min.css", type="text/css"),
+    Div(
+        Div(Div("This takes up the full width", cls="box", style="background-color: #800000;"), cls="col-xs-12"),
+        Div(Div("This takes up half", cls="box", style="background-color: #008000;"), cls="col-xs-6"),
+        Div(Div("This takes up half", cls="box", style="background-color: #0000B0;"), cls="col-xs-6"),
+        cls="row", style="color: #fff;"
+    )
+)
+show(grid)
+
+ + + + +
+
+
This takes up the full width
+
+
+
This takes up half
+
+
+
This takes up half
+
+
+ +
+
+

Aside: when in doubt with CSS stuff, add a background color or a border so you can see what’s happening!

+

Translating this into our app, we have a new homepage with a div (class="row") to store the generated images / previews, and a generation_preview function that returns boxes with the appropriate classes and styles to make them appear in the grid. I chose a layout with different numbers of columns for different screen sizes, but you could also just specify the col-xs class if you wanted the same layout on all devices.

+
gridlink = Link(rel="stylesheet", href="https://cdnjs.cloudflare.com/ajax/libs/flexboxgrid/6.3.1/flexboxgrid.min.css", type="text/css")
+app = FastHTML(hdrs=(picolink, gridlink))
+
+# Main page
+@app.get("/")
+def get():
+    inp = Input(id="new-prompt", name="prompt", placeholder="Enter a prompt")
+    add = Form(Group(inp, Button("Generate")), hx_post="/", target_id='gen-list', hx_swap="afterbegin")
+    gen_containers = [generation_preview(g) for g in gens(limit=10)] # Start with last 10
+    gen_list = Div(*gen_containers[::-1], id='gen-list', cls="row") # flexbox container: class = row
+    return Title('Image Generation Demo'), Main(H1('Magic Image Generation'), add, gen_list, cls='container')
+
+# Show the image (if available) and prompt for a generation
+def generation_preview(g):
+    grid_cls = "box col-xs-12 col-sm-6 col-md-4 col-lg-3"
+    image_path = f"{g.folder}/{g.id}.png"
+    if os.path.exists(image_path):
+        return Div(Card(
+                       Img(src=image_path, alt="Card image", cls="card-img-top"),
+                       Div(P(B("Prompt: "), g.prompt, cls="card-text"),cls="card-body"),
+                   ), id=f'gen-{g.id}', cls=grid_cls)
+    return Div(f"Generating gen {g.id} with prompt {g.prompt}", 
+            id=f'gen-{g.id}', hx_get=f"/gens/{g.id}", 
+            hx_trigger="every 2s", hx_swap="outerHTML", cls=grid_cls)
+

You can see the final result in main.py in the image_app_simple example directory, along with info on deploying it (tl;dr don’t!). We’ve also deployed a version that only shows your generations (tied to browser session) and has a credit system to save our bank accounts. You can access that here. Now for the next question: how do we keep track of different users?

+
+
+

Again, with Sessions

+

At the moment everyone sees all images! How do we keep some sort of unique identifier tied to a user? Before going all the way to setting up users, login pages etc., let’s look at a way to at least limit generations to the user’s session. You could do this manually with cookies. For convenience and security, fasthtml (via Starlette) has a special mechanism for storing small amounts of data in the user’s browser via the session argument to your route. This acts like a dictionary and you can set and get values from it. For example, here we look for a session_id key, and if it doesn’t exist we generate a new one:

+
@app.get("/")
+def get(session):
+    if 'session_id' not in session: session['session_id'] = str(uuid.uuid4())
+    return H1(f"Session ID: {session['session_id']}")
+

Refresh the page a few times - you’ll notice that the session ID remains the same. If you clear your browsing data, you’ll get a new session ID. And if you load the page in a different browser (but not a different tab), you’ll get a new session ID. This will persist within the current browser, letting us use it as a key for our generations. As a bonus, someone can’t spoof this session id by passing it in another way (for example, sending a query parameter). Behind the scenes, the data is stored in a browser cookie but it is signed with a secret key that stops the user or anyone nefarious from being able to tamper with it. The cookie is decoded back into a dictionary by something called a middleware function, which we won’t cover here. All you need to know is that we can use this to store bits of state in the user’s browser.

+

In the image app example, we can add a session_id column to our database, and modify our homepage like so:

+
@app.get("/")
+def get(session):
+    if 'session_id' not in session: session['session_id'] = str(uuid.uuid4())
+    inp = Input(id="new-prompt", name="prompt", placeholder="Enter a prompt")
+    add = Form(Group(inp, Button("Generate")), hx_post="/", target_id='gen-list', hx_swap="afterbegin")
+    gen_containers = [generation_preview(g) for g in gens(limit=10, where=f"session_id == '{session['session_id']}'")]
+    ...
+

So we check if the session id exists in the session, add one if not, and then limit the generations shown to only those tied to this session id. We filter the database with a where clause - see [TODO link Jeremy’s example for a more reliable way to do this]. The only other change we need to make is to store the session id in the database when a generation is made. You can check out this version here. You could instead write this app without relying on a database at all - simply storing the filenames of the generated images in the session, for example. But this more general approach of linking some kind of unique session identifier to users or data in our tables is a useful general pattern for more complex examples.

+
+
+

Again, with Credits!

+

Generating images with replicate costs money. So next let’s add a pool of credits that get used up whenever anyone generates an image. To recover our lost funds, we’ll also set up a payment system so that generous users can buy more credits for everyone. You could modify this to let users buy credits tied to their session ID, but at that point you risk having angry customers losing their money after wiping their browser history, and should consider setting up proper account management :)

+

Taking payments with Stripe is intimidating but very doable. Here’s a tutorial that shows the general principle using Flask. As with other popular tasks in the web-dev world, ChatGPT knows a lot about Stripe - but you should exercise extra caution when writing code that handles money!

+

For the finished example we add the bare minimum:

+
    +
  • A way to create a Stripe checkout session and redirect the user to the session URL
  • +
  • ‘Success’ and ‘Cancel’ routes to handle the result of the checkout
  • +
  • A route that listens for a webhook from Stripe to update the number of credits when a payment is made.
  • +
+

In a typical application you’ll want to keep track of which users make payments, catch other kinds of stripe events and so on. This example is more a ‘this is possible, do your own research’ than ‘this is how you do it’. But hopefully it does illustrate the key idea: there is no magic here. Stripe (and many other technologies) relies on sending users to different routes and shuttling data back and forth in requests. And we know how to do that!

+
+
+
+

More on Routing and Request Parameters

+

There are a number of ways information can be passed to the server. When you specify arguments to a route, FastHTML will search the request for values with the same name, and convert them to the correct type. In order, it searches

+
    +
  • The path parameters
  • +
  • The query parameters
  • +
  • The cookies
  • +
  • The headers
  • +
  • The session
  • +
  • Form data
  • +
+

There are also a few special arguments

+
    +
  • request (or any prefix like req): gets the raw Starlette Request object
  • +
  • session (or any prefix like sess): gets the session object
  • +
  • auth
  • +
  • htmx
  • +
  • app
  • +
+

In this section let’s quickly look at some of these in action.

+
+
from fasthtml.common import *
+from starlette.testclient import TestClient
+
+app = FastHTML()
+cli = TestClient(app)
+
+

Part of the route (path parameters):

+
+
@app.get('/user/{nm}')
+def _(nm:str): return f"Good day to you, {nm}!"
+
+cli.get('/user/jph').text
+
+
'Good day to you, jph!'
+
+
+

Matching with a regex:

+
+
reg_re_param("imgext", "ico|gif|jpg|jpeg|webm")
+
+@app.get(r'/static/{path:path}/{fn}.{ext:imgext}')
+def get_img(fn:str, path:str, ext:str): return f"Getting {fn}.{ext} from /{path}"
+
+cli.get('/static/foo/jph.ico').text
+
+
'Getting jph.ico from /foo/'
+
+
+

Using an enum (try using a string that isn’t in the enum):

+
+
ModelName = str_enum('ModelName', "alexnet", "resnet", "lenet")
+
+@app.get("/models/{nm}")
+def model(nm:ModelName): return nm
+
+print(cli.get('/models/alexnet').text)
+
+
alexnet
+
+
+

Casting to a Path:

+
+
@app.get("/files/{path}")
+def txt(path: Path): return path.with_suffix('.txt')
+
+print(cli.get('/files/foo').text)
+
+
foo.txt
+
+
+

An integer with a default value:

+
+
fake_db = [{"name": "Foo"}, {"name": "Bar"}]
+
+@app.get("/items/")
+def read_item(idx: int = 0): return fake_db[idx]
+
+print(cli.get('/items/?idx=1').text)
+
+
{"name":"Bar"}
+
+
+
+
# Equivalent to `/items/?idx=0`.
+print(cli.get('/items/').text)
+
+
{"name":"Foo"}
+
+
+

Boolean values (takes anything “truthy” or “falsy”):

+
+
@app.get("/booly/")
+def booly(coming:bool=True): return 'Coming' if coming else 'Not coming'
+
+print(cli.get('/booly/?coming=true').text)
+
+
Coming
+
+
+
+
print(cli.get('/booly/?coming=no').text)
+
+
Not coming
+
+
+

Getting dates:

+
+
@app.get("/datie/")
+def datie(d:parsed_date): return d
+
+date_str = "17th of May, 2024, 2p"
+print(cli.get(f'/datie/?d={date_str}').text)
+
+
2024-05-17 14:00:00
+
+
+

Matching a dataclass:

+
+
from dataclasses import dataclass, asdict
+
+@dataclass
+class Bodie:
+    a:int;b:str
+
+@app.route("/bodie/{nm}")
+def post(nm:str, data:Bodie):
+    res = asdict(data)
+    res['nm'] = nm
+    return res
+
+cli.post('/bodie/me', data=dict(a=1, b='foo')).text
+
+
'{"a":1,"b":"foo","nm":"me"}'
+
+
+
+

Cookies

+

Cookies can be set via a Starlette Response object, and can be read back by specifying the name:

+
+
from datetime import datetime
+
+@app.get("/setcookie")
+def setc(req):
+    now = datetime.now()
+    res = Response(f'Set to {now}')
+    res.set_cookie('now', str(now))
+    return res
+
+cli.get('/setcookie').text
+
+
'Set to 2024-07-20 23:14:54.364793'
+
+
+
+
@app.get("/getcookie")
+def getc(now:parsed_date): return f'Cookie was set at time {now.time()}'
+
+cli.get('/getcookie').text
+
+
'Cookie was set at time 23:14:54.364793'
+
+
+
+
+

User Agent and HX-Request

+

An argument of user_agent will match the header User-Agent. This holds for special headers like HX-Request (used by HTMX to signal when a request comes from an HTMX request) - the general pattern is that “-” is replaced with “_” and strings are turned to lowercase.

+
+
@app.get("/ua")
+async def ua(user_agent:str): return user_agent
+
+cli.get('/ua', headers={'User-Agent':'FastHTML'}).text
+
+
'FastHTML'
+
+
+
+
@app.get("/hxtest")
+def hxtest(htmx): return htmx.request
+
+cli.get('/hxtest', headers={'HX-Request':'1'}).text
+
+
'1'
+
+
+
+
+

Starlette Requests

+

If you add an argument called request(or any prefix of that, for example req) it will be populated with the Starlette Request object. This is useful if you want to do your own processing manually. For example, although FastHTML will parse forms for you, you could instead get form data like so:

+
@app.get("/form")
+async def form(request:Request):
+    form_data = await request.form()
+    a = form_data.get('a')
+

See the Starlette docs for more information on the Request object.

+
+
+

Starlette Responses

+

You can return a Starlette Response object from a route to control the response. For example:

+
@app.get("/redirect")
+def redirect():
+    return RedirectResponse(url="/")
+

We used this to set cookies in the previous example. See the Starlette docs for more information on the Response object.

+
+
+

Static Files

+

We often want to serve static files like images. This is easily done! For common file types (images, CSS etc) we can create a route that returns a Starlette FileResponse like so:

+
# For images, CSS, etc.
+@app.get("/{fname:path}.{ext:static}")
+def static(fname: str, ext: str):
+  return FileResponse(f'{fname}.{ext}')
+

You can customize it to suit your needs (for example, only serving files in a certain directory). You’ll notice some variant of this route in all our complete examples - even for apps with no static files the browser will typically request a /favicon.ico file, for example, and as the astute among you will have noticed this has sparked a bit of competition between Johno and Jeremy regarding which country flag should serve as the default!

+
+
+

WebSockets

+

For certain applications such as multiplayer games, websockets can be a powerful feature. Luckily HTMX and FastHTML has you covered! Simply specify that you wish to include the websocket header extension from HTMX:

+
app = FastHTML(exts='ws')
+rt = app.route
+

With that, you are now able to specify the different websocket specific HTMX goodies. For example, say we have a website we want to setup a websocket, you can simply:

+
def mk_inp(): return Input(id='msg')
+
+@rt('/')
+async def get(request):
+    cts = Div(
+        Div(id='notifications'),
+        Form(mk_inp(), id='form', ws_send=True),
+        hx_ext='ws', ws_connect='/ws')
+    return Titled('Websocket Test', cts)
+

And this will setup a connection on the route /ws along with a form that will send a message to the websocket whenever the form is submitted. Let’s go ahead and handle this route:

+
@app.ws('/ws')
+async def ws(msg:str, send):
+    await send(Div('Hello ' + msg, id="notifications"))
+    await sleep(2)
+    return Div('Goodbye ' + msg, id="notifications"), mk_inp()
+

One thing you might have noticed is a lack of target id for our websocket trigger for swapping HTML content. This is because HTMX always swaps content with websockets with Out of Band Swaps. Therefore, HTMX will look for the id in the returned HTML content from the server for determining what to swap. To send stuff to the client, you can either use the send parameter or simply return the content or both!

+

Now, sometimes you might want to perform actions when a client connects or disconnects such as add or remove a user from a player queue. To hook into these events, you can pass your connection or disconnection function to the app.ws decorator:

+
async def on_connect(send):
+    print('Connected!')
+    await send(Div('Hello, you have connected', id="notifications"))
+
+async def on_disconnect(ws):
+    print('Disconnected!')
+
+@app.ws('/ws', conn=on_connect, disconn=on_disconnect)
+async def ws(msg:str, send):
+    await send(Div('Hello ' + msg, id="notifications"))
+    await sleep(2)
+    return Div('Goodbye ' + msg, id="notifications"), mk_inp()
+
+
+
+

Full Example #3 - Chatbot Example with DaisyUI Components

+

Let’s go back to the topic of adding components or styling beyond the simple PicoCSS examples so far. How might we adopt a component or framework? In this example, let’s build a chatbot UI leveraging the DaisyUI chat bubble. The final result will look like this:

+
+
+

+
image.png
+
+
+

At first glance, DaisyUI’s chat component looks quite intimidating. The examples look like this:

+
<div class="chat chat-start">
+  <div class="chat-image avatar">
+    <div class="w-10 rounded-full">
+      <img alt="Tailwind CSS chat bubble component" src="https://img.daisyui.com/images/stock/photo-1534528741775-53994a69daeb.jpg" />
+    </div>
+  </div>
+  <div class="chat-header">
+    Obi-Wan Kenobi
+    <time class="text-xs opacity-50">12:45</time>
+  </div>
+  <div class="chat-bubble">You were the Chosen One!</div>
+  <div class="chat-footer opacity-50">
+    Delivered
+  </div>
+</div>
+<div class="chat chat-end">
+  <div class="chat-image avatar">
+    <div class="w-10 rounded-full">
+      <img alt="Tailwind CSS chat bubble component" src="https://img.daisyui.com/images/stock/photo-1534528741775-53994a69daeb.jpg" />
+    </div>
+  </div>
+  <div class="chat-header">
+    Anakin
+    <time class="text-xs opacity-50">12:46</time>
+  </div>
+  <div class="chat-bubble">I hate you!</div>
+  <div class="chat-footer opacity-50">
+    Seen at 12:46
+  </div>
+</div>
+

We have several things going for us however.

+
    +
  • ChatGPT knows DaisyUI and Tailwind (DaisyUI is a Tailwind component library)
  • +
  • We can build things up piece by piece with AI standing by to help.
  • +
+

https://h2f.answer.ai/ is a tool that can convert HTML to FT (fastcore.xml) and back, which is useful for getting a quick starting point when you have an HTML example to start from.

+

We can strip out some unnecessary bits and try to get the simplest possible example working in a notebook first:

+
+
# Loading tailwind and daisyui
+headers = (Script(src="https://cdn.tailwindcss.com"),
+           Link(rel="stylesheet", href="https://cdn.jsdelivr.net/npm/daisyui@4.11.1/dist/full.min.css"))
+
+# Displaying a single message
+d = Div(
+    Div("Chat header here", cls="chat-header"),
+    Div("My message goes here", cls="chat-bubble chat-bubble-primary"),
+    cls="chat chat-start"
+)
+# show(Html(*headers, d)) # uncomment to view
+
+

Now we can extend this to render multiple messages, with the message being on the left (chat-start) or right (chat-end) depending on the role. While we’re at it, we can also change the color (chat-bubble-primary) of the message and put them all in a chat-box div:

+
+
messages = [
+    {"role":"user", "content":"Hello"},
+    {"role":"assistant", "content":"Hi, how can I assist you?"}
+]
+
+def ChatMessage(msg):
+    return Div(
+        Div(msg['role'], cls="chat-header"),
+        Div(msg['content'], cls=f"chat-bubble chat-bubble-{'primary' if msg['role'] == 'user' else 'secondary'}"),
+        cls=f"chat chat-{'end' if msg['role'] == 'user' else 'start'}")
+
+chatbox = Div(*[ChatMessage(msg) for msg in messages], cls="chat-box", id="chatlist")
+
+# show(Html(*headers, chatbox)) # Uncomment to view
+
+

Next, it was back to the ChatGPT to tweak the chat box so it wouldn’t grow as messages were added. I asked:

+
"I have something like this (it's working now) 
+[code]
+The messages are added to this div so it grows over time. 
+Is there a way I can set it's height to always be 80% of the total window height with a scroll bar if needed?"
+

Based on this query GPT4o helpfully shared that “This can be achieved using Tailwind CSS utility classes. Specifically, you can use h-[80vh] to set the height to 80% of the viewport height, and overflow-y-auto to add a vertical scroll bar when needed.”

+

To put it another way: none of the CSS classes in the following example were written by a human, and what edits I did make were informed by advice from the AI that made it relatively painless!

+

The actual chat functionality of the app is based on our claudette library. As with the image example, we face a potential hiccup in that getting a response from an LLM is slow. We need a way to have the user message added to the UI immediately, and then have the response added once it’s available. We could do something similar to the image generation example above, or use websockets. Check out the full example for implementations of both, along with further details.

+
+
+

Full Example #4 - Multiplayer Game of Life Example with Websockets

+

Let’s see how we can implement a collaborative website using Websockets in FastHTML. To showcase this, we will use the famous Conway’s Game of Life, which is a game that takes place in a grid world. Each cell in the grid can be either alive or dead. The cell’s state is initially given by a user before the game is started and then evolves through the iteration of the grid world once the clock starts. Whether a cell’s state will change from the previous state depends on simple rules based on its neighboring cells’ states. Here is the standard Game of Life logic implemented in Python courtesy of ChatGPT:

+
grid = [[0 for _ in range(20)] for _ in range(20)]
+def update_grid(grid: list[list[int]]) -> list[list[int]]:
+    new_grid = [[0 for _ in range(20)] for _ in range(20)]
+    def count_neighbors(x, y):
+        directions = [(-1, -1), (-1, 0), (-1, 1), (0, -1), (0, 1), (1, -1), (1, 0), (1, 1)]
+        count = 0
+        for dx, dy in directions:
+            nx, ny = x + dx, y + dy
+            if 0 <= nx < len(grid) and 0 <= ny < len(grid[0]): count += grid[nx][ny]
+        return count
+    for i in range(len(grid)):
+        for j in range(len(grid[0])):
+            neighbors = count_neighbors(i, j)
+            if grid[i][j] == 1:
+                if neighbors < 2 or neighbors > 3: new_grid[i][j] = 0
+                else: new_grid[i][j] = 1
+            elif neighbors == 3: new_grid[i][j] = 1
+    return new_grid
+

This would be a very dull game if we were to run it, since the initial state of everything would remain dead. Therefore, we need a way of letting the user give an initial state before starting the game. FastHTML to the rescue!

+
def Grid():
+    cells = []
+    for y, row in enumerate(game_state['grid']):
+        for x, cell in enumerate(row):
+            cell_class = 'alive' if cell else 'dead'
+            cell = Div(cls=f'cell {cell_class}', hx_put='/update', hx_vals={'x': x, 'y': y}, hx_swap='none', hx_target='#gol', hx_trigger='click')
+            cells.append(cell)
+    return Div(*cells, id='grid')
+
+@rt('/update')
+async def put(x: int, y: int):
+    grid[y][x] = 1 if grid[y][x] == 0 else 0
+

Above is a component for representing the game’s state that the user can interact with and update on the server using cool HTMX features such as hx_vals for determining which cell was clicked to make it dead or alive. Now, you probably noticed that the HTTP request in this case is a PUT request, which does not return anything and this means our client’s view of the grid world and the server’s game state will immediately become out of sync :(. We could of course just return a new Grid component with the updated state, but that would only work for a single client, if we had more, they quickly get out of sync with each other and the server. Now Websockets to the rescue!

+

Websockets are a way for the server to keep a persistent connection with clients and send data to the client without explicitly being requested for information, which is not possible with HTTP. Luckily FastHTML and HTMX work well with Websockets. Simply state you wish to use websockets for your app and define a websocket route:

+
...
+app = FastHTML(hdrs=(picolink, gridlink, css, htmx_ws), exts='ws')
+
+player_queue = []
+async def update_players():
+    for i, player in enumerate(player_queue):
+        try: await player(Grid())
+        except: player_queue.pop(i)
+async def on_connect(send): player_queue.append(send)
+async def on_disconnect(send): await update_players()
+
+@app.ws('/gol', conn=on_connect, disconn=on_disconnect)
+async def ws(msg:str, send): pass
+
+def Home(): return Title('Game of Life'), Main(gol, Div(Grid(), id='gol', cls='row center-xs'), hx_ext="ws", ws_connect="/gol")
+
+@rt('/update')
+async def put(x: int, y: int):
+    grid[y][x] = 1 if grid[y][x] == 0 else 0
+    await update_players()
+...
+

Here we simply keep track of all the players that have connected or disconnected to our site and when an update occurs, we send updates to all the players still connected via websockets. Via HTMX, you are still simply exchanging HTML from the server to the client and will swap in the content based on how you setup your hx_swap attribute. There is only one difference, that being all swaps are OOB. You can find more information on the HTMX websocket extension documentation page here. You can find a full fledge hosted example of this app here.

+
+
+

FT objects and HTML

+

These FT objects create a ‘FastTag’ structure [tag,children,attrs] for to_xml(). When we call Div(...), the elements we pass in are the children. Attributes are passed in as keywords. class and for are special words in python, so we use cls, klass or _class instead of class and fr or _for instead of for. Note these objects are just 3-element lists - you can create custom ones too as long as they’re also 3-element lists. Alternately, leaf nodes can be strings instead (which is why you can do Div('some text')). If you pass something that isn’t a 3-element list or a string, it will be converted to a string using str()… unless (our final trick) you define a __ft__ method that will run before str(), so you can render things a custom way.

+

For example, here’s one way we could make a custom class that can be rendered into HTML:

+
+
class Person:
+    def __init__(self, name, age):
+        self.name = name
+        self.age = age
+
+    def __ft__(self):
+        return ['div', [f'{self.name} is {self.age} years old.'], {}]
+
+p = Person('Jonathan', 28)
+print(to_xml(Div(p, "more text", cls="container")))
+
+
<div class="container">
+  <div>Jonathan is 28 years old.</div>
+more text
+</div>
+
+
+
+

In the examples, you’ll see we often patch in __ft__ methods to existing classes to control how they’re rendered. For example, if Person didn’t have a __ft__ method or we wanted to override it, we could add a new one like this:

+
+
from fastcore.all import patch
+
+@patch
+def __ft__(self:Person):
+    return Div("Person info:", Ul(Li("Name:",self.name), Li("Age:", self.age)))
+
+show(p)
+
+
+Person info: +
    +
  • +Name: +Jonathan +
  • +
  • +Age: +28 +
  • +
+
+
+
+

Some tags from fastcore.xml are overwritten by fasthtml.core and a few are further extended by fasthtml.xtend using this method. Over time, we hope to see others developing custom components too, giving us a larger and larger ecosystem of reusable components.

+
+
+

Custom Scripts and Styling

+

There are many popular JavaScript and CSS libraries that can be used via a simple Script or Style tag. But in some cases you will need to write more custom code. FastHTML’s js.py contains a few examples that may be useful as reference.

+

For example, to use the marked.js library to render markdown in a div, including in components added after the page has loaded via htmx, we do something like this:

+
import { marked } from "https://cdn.jsdelivr.net/npm/marked/lib/marked.esm.js";
+proc_htmx('%s', e => e.innerHTML = marked.parse(e.textContent));
+

proc_htmx is a shortcut that we wrote to apply a function to elements matching a selector, including the element that triggered the event. Here’s the code for reference:

+
export function proc_htmx(sel, func) {
+  htmx.onLoad(elt => {
+    const elements = htmx.findAll(elt, sel);
+    if (elt.matches(sel)) elements.unshift(elt)
+    elements.forEach(func);
+  });
+}
+

The AI Pictionary example uses a larger chunk of custom JavaScript to handle the drawing canvas. It’s a good example of the type of application where running code on the client side makes the most sense, but still shows how you can integrate it with FastHTML on the server side to add functionality (like the AI responses) easily.

+

Adding styling with custom CSS and libraries such as tailwind is done the same way we add custom JavaScript. The doodle example uses Doodle.CSS to style the page in a quirky way.

+
+
+

Deploying Your App

+

We can deploy FastHTML almost anywhere you can deploy python apps. We’ve tested Railway, Replit, HuggingFace, and PythonAnywhere.

+
+

Railway

+
    +
  1. Install the Railway CLI and sign up for an account.
  2. +
  3. Set up a folder with our app as main.py
  4. +
  5. In the folder, run railway login.
  6. +
  7. Use the fh_railway_deploy script to deploy our project:
  8. +
+
fh_railway_deploy MY_APP_NAME
+

What the script does for us:

+
    +
  1. Do we have an existing railway project? +
      +
    • Yes: Link the project folder to our existing Railway project.
    • +
    • No: Create a new Railway project.
    • +
  2. +
  3. Deploy the project. We’ll see the logs as the service is built and run!
  4. +
  5. Fetches and displays the URL of our app.
  6. +
  7. By default, mounts a /app/data folder on the cloud to our app’s root folder. The app is run in /app by default, so from our app anything we store in /data will persist across restarts.
  8. +
+

A final note about Railway: We can add secrets like API keys that can be accessed as environment variables from our apps via ‘Variables’. For example, for the image generation app, we can add a REPLICATE_API_KEY variable, and then in main.py we can access it as os.environ['REPLICATE_API_KEY'].

+
+
+

Replit

+

Fork this repl for a minimal example you can edit to your heart’s content. .replit has been edited to add the right run command (run = ["uvicorn", "main:app", "--reload"]) and to set up the ports correctly. FastHTML was installed with poetry add python-fasthtml, you can add additional packages as needed in the same way. Running the app in Replit will show you a webview, but you may need to open in a new tab for all features (such as cookies) to work. When you’re ready, you can deploy your app by clicking the ‘Deploy’ button. You pay for usage - for an app that is mostly idle the cost is usually a few cents per month.

+

You can store secrets like API keys via the ‘Secrets’ tab in the Replit project settings.

+
+
+

HuggingFace

+

Follow the instructions in this repository to deploy to HuggingFace spaces.

+
+
+
+

Where Next?

+

We’ve covered a lot of ground here! Hopefully this has given you plenty to work with in building your own FastHTML apps. If you have any questions, feel free to ask in the #fasthtml Discord channel (in the fastai Discord community). You can look through the other examples in the fasthtml-example repository for more ideas, and keep an eye on Jeremy’s YouTube channel where we’ll be releasing a number of “dev chats” related to FastHTML in the near future.

+ + +
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/tutorials/by_example.html.md b/tutorials/by_example.html.md new file mode 100644 index 00000000..86917c19 --- /dev/null +++ b/tutorials/by_example.html.md @@ -0,0 +1,1563 @@ +# FastHTML By Example + + + + +This tutorial provides an alternate introduction to FastHTML by building +out example applications. We also illustrate how to use FastHTML +foundations to create custom web apps. Finally, this document serves as +minimal context for a LLM to turn it into a FastHTML assistant. + +Let’s get started. + +## FastHTML Basics + +FastHTML is *just Python*. You can install it with +`pip install python-fasthtml`. Extensions/components built for it can +likewise be distributed via PyPI or as simple Python files. + +The core usage of FastHTML is to define routes, and then to define what +to do at each route. This is similar to the +[FastAPI](https://fastapi.tiangolo.com/) web framework (in fact we +implemented much of the functionality to match the FastAPI usage +examples), but where FastAPI focuses on returning JSON data to build +APIs, FastHTML focuses on returning HTML data. + +Here’s a simple FastHTML app that returns a “Hello, World” message: + +``` python +from fasthtml.common import FastHTML, serve + +app = FastHTML() + +@app.get("/") +def home(): + return "

Hello, World

" + +serve() +``` + +To run this app, place it in a file, say `app.py`, and then run it with +`python app.py`. + + INFO: Will watch for changes in these directories: ['/home/jonathan/fasthtml-example'] + INFO: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) + INFO: Started reloader process [871942] using WatchFiles + INFO: Started server process [871945] + INFO: Waiting for application startup. + INFO: Application startup complete. + +If you navigate to in a browser, you’ll see your +“Hello, World”. If you edit the `app.py` file and save it, the server +will reload and you’ll see the updated message when you refresh the page +in your browser. + +## Constructing HTML + +Notice we wrote some HTML in the previous example. We don’t want to do +that! Some web frameworks require that you learn HTML, CSS, JavaScript +AND some templating language AND python. We want to do as much as +possible with just one language. Fortunately, the Python module +[fastcore.xml](https://fastcore.fast.ai/xml.html) has all we need for +constructing HTML from Python, and FastHTML includes all the tags you +need to get started. For example: + +``` python +from fasthtml.common import * +page = Html( + Head(Title('Some page')), + Body(Div('Some text, ', A('A link', href='https://example.com'), Img(src="https://placehold.co/200"), cls='myclass'))) +print(to_xml(page)) +``` + + + + + + Some page + + +
+ Some text, + A link + +
+ + + +``` python +show(page) +``` + + + + + Some page + + +
+Some text, + A link + +
+ + + +If that `import *` worries you, you can always import only the tags you +need. + +FastHTML is smart enough to know about fastcore.xml, and so you don’t +need to use the `to_xml` function to convert your FT objects to HTML. +You can just return them as you would any other Python object. For +example, if we modify our previous example to use fastcore.xml, we can +return an FT object directly: + +``` python +from fasthtml.common import * +app = FastHTML() + +@app.get("/") +def home(): + page = Html( + Head(Title('Some page')), + Body(Div('Some text, ', A('A link', href='https://example.com'), Img(src="https://placehold.co/200"), cls='myclass'))) + return page + +serve() +``` + +This will render the HTML in the browser. + +For debugging, you can right-click on the rendered HTML in the browser +and select “Inspect” to see the underlying HTML that was generated. +There you’ll also find the ‘network’ tab, which shows you the requests +that were made to render the page. Refresh and look for the request to +`127.0.0.1` - and you’ll see it’s just a `GET` request to `/`, and the +response body is the HTML you just returned. + +
+ +> **Live Reloading** +> +> You can also enable [live reloading](../ref/live_reload.ipynb) so you +> don’t have to manually refresh your browser to view updates. + +
+ +You can also use Starlette’s `TestClient` to try it out in a notebook: + +``` python +from starlette.testclient import TestClient +client = TestClient(app) +r = client.get("/") +print(r.text) +``` + + + Some page + +
+ Some text, + A link + +
+ + + +FastHTML wraps things in an Html tag if you don’t do it yourself (unless +the request comes from htmx, in which case you get the element +directly). See [FT objects and HTML](#ft-objects-and-html) for more on +creating custom components or adding HTML rendering to existing Python +objects. To give the page a non-default title, return a Title before +your main content: + +``` python +app = FastHTML() + +@app.get("/") +def home(): + return Title("Page Demo"), Div(H1('Hello, World'), P('Some text'), P('Some more text')) + +client = TestClient(app) +print(client.get("/").text) +``` + + + + + + Page Demo + + + + + + + +
+

Hello, World

+

Some text

+

Some more text

+
+ + + +We’ll use this pattern often in the examples to follow. + +## Defining Routes + +The HTTP protocol defines a number of methods (‘verbs’) to send requests +to a server. The most common are GET, POST, PUT, DELETE, and HEAD. We +saw ‘GET’ in action before - when you navigate to a URL, you’re making a +GET request to that URL. We can do different things on a route for +different HTTP methods. For example: + +``` python +@app.route("/", methods='get') +def home(): + return H1('Hello, World') + +@app.route("/", methods=['post', 'put']) +def post_or_put(): + return "got a POST or PUT request" +``` + +This says that when someone navigates to the root URL “/” (i.e. sends a +GET request), they will see the big “Hello, World” heading. When someone +submits a POST or PUT request to the same URL, the server should return +the string “got a post or put request”. + +
+ +> **Test the POST request** +> +> You can test the POST request with +> `curl -X POST http://127.0.0.1:8000 -d "some data"`. This sends some +> data to the server, you should see the response “got a post or put +> request” printed in the terminal. + +
+ +There are a few other ways you can specify the route+method - FastHTML +has `.get`, `.post`, etc. as shorthand for +`route(..., methods=['get'])`, etc. + +``` python +@app.get("/") +def my_function(): + return "Hello World from a GET request" +``` + +Or you can use the `@rt` decorator without a method but specify the +method with the name of the function. For example: + +``` python +rt = app.route + +@rt("/") +def post(): + return "Hello World from a POST request" +``` + +``` python +client.post("/").text +``` + + 'Hello World from a POST request' + +You’re welcome to pick whichever style you prefer. Using routes lets you +show different content on different pages - ‘/home’, ‘/about’ and so on. +You can also respond differently to different kinds of requests to the +same route, as shown above. You can also pass data via the route: + +
+ +## `@app.get` + +``` python +@app.get("/greet/{nm}") +def greet(nm:str): + return f"Good day to you, {nm}!" + +client.get("/greet/Dave").text +``` + + 'Good day to you, Dave!' + +## `@rt` + +``` python +@rt("/greet/{nm}") +def get(nm:str): + return f"Good day to you, {nm}!" + +client.get("/greet/Dave").text +``` + + 'Good day to you, Dave!' + +
+ +More on this in the [More on Routing and Request +Parameters](#more-on-routing-and-request-parameters) section, which goes +deeper into the different ways to get information from a request. + +## Styling Basics + +Plain HTML probably isn’t quite what you imagine when you visualize your +beautiful web app. CSS is the go-to language for styling HTML. But +again, we don’t want to learn extra languages unless we absolutely have +to! Fortunately, there are ways to get much more visually appealing +sites by relying on the hard work of others, using existing CSS +libraries. One of our favourites is [PicoCSS](https://picocss.com/). A +common way to add CSS files to web pages is to use a +[``](https://www.w3schools.com/tags/tag_link.asp) tag inside your +[HTML header](https://www.w3schools.com/tags/tag_header.asp), like this: + +``` html +
+ ... + +
+``` + +For convenience, FastHTML already defines a Pico component for you with +`picolink`: + +``` python +print(to_xml(picolink)) +``` + + + + + +
+ +> **Note** +> +> `picolink` also includes a ` + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

JS App Walkthrough

+
+ +
+
+ How to build a website with custom JavaScript in FastHTML step-by-step +
+
+ + +
+ + + + +
+ + + +
+ + + +
+

Installation

+

You’ll need the following software to complete the tutorial, read on for specific installation instructions:

+
    +
  1. Python
  2. +
  3. A Python package manager such as pip (which normally comes with Python) or uv
  4. +
  5. FastHTML
  6. +
  7. Web browser
  8. +
  9. Railway.app account
  10. +
+

If you haven’t worked with Python before, we recommend getting started with Miniconda.

+

Note that you will only need to follow the steps in the installation section once per environment. If you create a new repo, you won’t need to redo these.

+
+

Install FastHTML

+

For Mac, Windows and Linux, enter:

+
pip install python-fasthtml
+
+
+
+

First steps

+

By the end of this section you’ll have your own FastHTML website with tests deployed to railway.app.

+
+

Create a hello world

+

Create a new folder to organize all the files for your project. Inside this folder, create a file called main.py and add the following code to it:

+
+
+
main.py
+
+
from fasthtml.common import *
+
+app = FastHTML()
+rt = app.route
+
+@rt('/')
+def get():
+    return 'Hello, world!'
+
+serve()
+
+

Finally, run python main.py in your terminal and open your browser to the ‘Link’ that appears.

+
+
+

QuickDraw: A FastHTML Adventure 🎨✨

+

The end result of this tutorial will be QuickDraw, a real-time collaborative drawing app using FastHTML. Here is what the final site will look like:

+
+
+

+
QuickDraw
+
+
+
+

Drawing Rooms

+

Drawing rooms are the core concept of our application. Each room represents a separate drawing space where a user can let their inner Picasso shine. Here’s a detailed breakdown:

+
    +
  1. Room Creation and Storage
  2. +
+
+
+
main.py
+
+
db = database('data/drawapp.db')
+rooms = db.t.rooms
+if rooms not in db.t:
+    rooms.create(id=int, name=str, created_at=str, pk='id')
+Room = rooms.dataclass()
+
+@patch
+def __ft__(self:Room):
+    return Li(A(self.name, href=f"/rooms/{self.id}"))
+
+

Or you can use our fast_app function to create a FastHTML app with a SQLite database and dataclass in one line:

+
+
+
main.py
+
+
def render(room):
+    return Li(A(room.name, href=f"/rooms/{room.id}"))
+
+app,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, pk='id')
+
+

We are specifying a render function to convert our dataclass into HTML, which is the same as extending the __ft__ method from the patch decorator we used before. We will use this method for the rest of the tutorial since it is a lot cleaner and easier to read.

+
    +
  • We’re using a SQLite database (via FastLite) to store our rooms.
  • +
  • Each room has an id (integer), a name (string), and a created_at timestamp (string).
  • +
  • The Room dataclass is automatically generated based on this structure.
  • +
+
    +
  1. Creating a room
  2. +
+
+
+
main.py
+
+
@rt("/")
+def get():
+    # The 'Input' id defaults to the same as the name, so you can omit it if you wish
+    create_room = Form(Input(id="name", name="name", placeholder="New Room Name"),
+                       Button("Create Room"),
+                       hx_post="/rooms", hx_target="#rooms-list", hx_swap="afterbegin")
+    rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list')
+    return Titled("DrawCollab", 
+                  H1("DrawCollab"),
+                  create_room, rooms_list)
+
+@rt("/rooms")
+async def post(room:Room):
+    room.created_at = datetime.now().isoformat()
+    return rooms.insert(room)
+
+
    +
  • When a user submits the “Create Room” form, this route is called.
  • +
  • It creates a new Room object, sets the creation time, and inserts it into the database.
  • +
  • It returns an HTML list item with a link to the new room, which is dynamically added to the room list on the homepage thanks to HTMX.
  • +
+
    +
  1. Let’s give our rooms shape
  2. +
+
+
+
main.py
+
+
@rt("/rooms/{id}")
+async def get(id:int):
+    room = rooms[id]
+    return Titled(f"Room: {room.name}", H1(f"Welcome to {room.name}"), A(Button("Leave Room"), href="/"))
+
+
    +
  • This route renders the interface for a specific room.
  • +
  • It fetches the room from the database and renders a title, heading, and paragraph.
  • +
+

Here is the full code so far:

+
+
+
main.py
+
+
from fasthtml.common import *
+from datetime import datetime
+
+def render(room):
+    return Li(A(room.name, href=f"/rooms/{room.id}"))
+
+app,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, pk='id')
+
+@rt("/")
+def get():
+    create_room = Form(Input(id="name", name="name", placeholder="New Room Name"),
+                       Button("Create Room"),
+                       hx_post="/rooms", hx_target="#rooms-list", hx_swap="afterbegin")
+    rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list')
+    return Titled("DrawCollab", create_room, rooms_list)
+
+@rt("/rooms")
+async def post(room:Room):
+    room.created_at = datetime.now().isoformat()
+    return rooms.insert(room)
+
+@rt("/rooms/{id}")
+async def get(id:int):
+    room = rooms[id]
+    return Titled(f"Room: {room.name}", H1(f"Welcome to {room.name}"), A(Button("Leave Room"), href="/"))
+
+serve()
+
+

Now run python main.py in your terminal and open your browser to the ‘Link’ that appears. You should see a page with a form to create a new room and a list of existing rooms.

+
+
+

The Canvas - Let’s Get Drawing! 🖌️

+

Time to add the actual drawing functionality. We’ll use Fabric.js for this:

+
+
+
main.py
+
+
# ... (keep the previous imports and database setup)
+
+@rt("/rooms/{id}")
+async def get(id:int):
+    room = rooms[id]
+    canvas = Canvas(id="canvas", width="800", height="600")
+    color_picker = Input(type="color", id="color-picker", value="#3CDD8C")
+    brush_size = Input(type="range", id="brush-size", min="1", max="50", value="10")
+    
+    js = """
+    var canvas = new fabric.Canvas('canvas');
+    canvas.isDrawingMode = true;
+    canvas.freeDrawingBrush.color = '#3CDD8C';
+    canvas.freeDrawingBrush.width = 10;
+    
+    document.getElementById('color-picker').onchange = function() {
+        canvas.freeDrawingBrush.color = this.value;
+    };
+    
+    document.getElementById('brush-size').oninput = function() {
+        canvas.freeDrawingBrush.width = parseInt(this.value, 10);
+    };
+    """
+    
+    return Titled(f"Room: {room.name}",
+                  A(Button("Leave Room"), href="/"),
+                  canvas,
+                  Div(color_picker, brush_size),
+                  Script(src="https://cdnjs.cloudflare.com/ajax/libs/fabric.js/5.3.1/fabric.min.js"),
+                  Script(js))
+
+# ... (keep the serve() part)
+
+

Now we’ve got a drawing canvas! FastHTML makes it easy to include external libraries and add custom JavaScript.

+
+
+

Saving and Loading Canvases 💾

+

Now that we have a working drawing canvas, let’s add the ability to save and load drawings. We’ll modify our database schema to include a canvas_data field, and add new routes for saving and loading canvas data. Here’s how we’ll update our code:

+
    +
  1. Modify the database schema:
  2. +
+
+
+
main.py
+
+
app,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, canvas_data=str, pk='id')
+
+
    +
  1. Add a save button that grabs the canvas’ state and sends it to the server:
  2. +
+
+
+
main.py
+
+
@rt("/rooms/{id}")
+async def get(id:int):
+    room = rooms[id]
+    canvas = Canvas(id="canvas", width="800", height="600")
+    color_picker = Input(type="color", id="color-picker", value="#3CDD8C")
+    brush_size = Input(type="range", id="brush-size", min="1", max="50", value="10")
+    save_button = Button("Save Canvas", id="save-canvas", hx_post=f"/rooms/{id}/save", hx_vals="js:{canvas_data: JSON.stringify(canvas.toJSON())}")
+    # ... (rest of the function remains the same)
+
+
    +
  1. Add routes for saving and loading canvas data:
  2. +
+
+
+
main.py
+
+
@rt("/rooms/{id}/save")
+async def post(id:int, canvas_data:str):
+    rooms.update({'canvas_data': canvas_data}, id)
+    return "Canvas saved successfully"
+
+@rt("/rooms/{id}/load")
+async def get(id:int):
+    room = rooms[id]
+    return room.canvas_data if room.canvas_data else "{}"
+
+
    +
  1. Update the JavaScript to load existing canvas data:
  2. +
+
+
+
main.py
+
+
js = f"""
+    var canvas = new fabric.Canvas('canvas');
+    canvas.isDrawingMode = true;
+    canvas.freeDrawingBrush.color = '#3CDD8C';
+    canvas.freeDrawingBrush.width = 10;
+    // Load existing canvas data
+    fetch(`/rooms/{id}/load`)
+    .then(response => response.json())
+    .then(data => {{
+        if (data && Object.keys(data).length > 0) {{
+            canvas.loadFromJSON(data, canvas.renderAll.bind(canvas));
+        }}
+    }});
+    
+    // ... (rest of the JavaScript remains the same)
+"""
+
+

With these changes, users can now save their drawings and load them when they return to the room. The canvas data is stored as a JSON string in the database, allowing for easy serialization and deserialization. Try it out! Create a new room, make a drawing, save it, and then reload the page. You should see your drawing reappear, ready for further editing.

+

Here is the completed code:

+
+
+
main.py
+
+
from fasthtml.common import *
+from datetime import datetime
+
+def render(room):
+    return Li(A(room.name, href=f"/rooms/{room.id}"))
+
+app,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, canvas_data=str, pk='id')
+
+@rt("/")
+def get():
+    create_room = Form(Input(id="name", name="name", placeholder="New Room Name"),
+                       Button("Create Room"),
+                       hx_post="/rooms", hx_target="#rooms-list", hx_swap="afterbegin")
+    rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list')
+    return Titled("QuickDraw", 
+                  create_room, rooms_list)
+
+@rt("/rooms")
+async def post(room:Room):
+    room.created_at = datetime.now().isoformat()
+    return rooms.insert(room)
+
+@rt("/rooms/{id}")
+async def get(id:int):
+    room = rooms[id]
+    canvas = Canvas(id="canvas", width="800", height="600")
+    color_picker = Input(type="color", id="color-picker", value="#000000")
+    brush_size = Input(type="range", id="brush-size", min="1", max="50", value="10")
+    save_button = Button("Save Canvas", id="save-canvas", hx_post=f"/rooms/{id}/save", hx_vals="js:{canvas_data: JSON.stringify(canvas.toJSON())}")
+
+    js = f"""
+    var canvas = new fabric.Canvas('canvas');
+    canvas.isDrawingMode = true;
+    canvas.freeDrawingBrush.color = '#000000';
+    canvas.freeDrawingBrush.width = 10;
+
+    // Load existing canvas data
+    fetch(`/rooms/{id}/load`)
+    .then(response => response.json())
+    .then(data => {{
+        if (data && Object.keys(data).length > 0) {{
+            canvas.loadFromJSON(data, canvas.renderAll.bind(canvas));
+        }}
+    }});
+    
+    document.getElementById('color-picker').onchange = function() {{
+        canvas.freeDrawingBrush.color = this.value;
+    }};
+    
+    document.getElementById('brush-size').oninput = function() {{
+        canvas.freeDrawingBrush.width = parseInt(this.value, 10);
+    }};
+    """
+    
+    return Titled(f"Room: {room.name}",
+                  A(Button("Leave Room"), href="/"),
+                  canvas,
+                  Div(color_picker, brush_size, save_button),
+                  Script(src="https://cdnjs.cloudflare.com/ajax/libs/fabric.js/5.3.1/fabric.min.js"),
+                  Script(js))
+
+@rt("/rooms/{id}/save")
+async def post(id:int, canvas_data:str):
+    rooms.update({'canvas_data': canvas_data}, id)
+    return "Canvas saved successfully"
+
+@rt("/rooms/{id}/load")
+async def get(id:int):
+    room = rooms[id]
+    return room.canvas_data if room.canvas_data else "{}"
+
+serve()
+
+
+
+
+

Deploying to Railway

+

You can deploy your website to a number of hosting providers, for this tutorial we’ll be using Railway. To get started, make sure you create an account and install the Railway CLI. Once installed, make sure to run railway login to log in to your account.

+

To make deploying your website as easy as possible, FastHTMl comes with a built in CLI tool that will handle most of the deployment process for you. To deploy your website, run the following command in your terminal in the root directory of your project:

+
fh_railway_deploy quickdraw
+
+
+
+ +
+
+Note +
+
+
+

Your app must be located in a main.py file for this to work.

+
+
+
+
+

Conclusion: You’re a FastHTML Artist Now! 🎨🚀

+

Congratulations! You’ve just built a sleek, interactive web application using FastHTML. Let’s recap what we’ve learned:

+
    +
  1. FastHTML allows you to create dynamic web apps with minimal code.
  2. +
  3. We used FastHTML’s routing system to handle different pages and actions.
  4. +
  5. We integrated with a SQLite database to store room information and canvas data.
  6. +
  7. We utilized Fabric.js to create an interactive drawing canvas.
  8. +
  9. We implemented features like color picking, brush size adjustment, and canvas saving.
  10. +
  11. We used HTMX for seamless, partial page updates without full reloads.
  12. +
  13. We learned how to deploy our FastHTML application to Railway for easy hosting.
  14. +
+

You’ve taken your first steps into the world of FastHTML development. From here, the possibilities are endless! You could enhance the drawing app further by adding features like:

+
    +
  • Implementing different drawing tools (e.g., shapes, text)
  • +
  • Adding user authentication
  • +
  • Creating a gallery of saved drawings
  • +
  • Implementing real-time collaborative drawing using WebSockets
  • +
+

Whatever you choose to build next, FastHTML has got your back. Now go forth and create something awesome! Happy coding! 🖼️🚀

+ + +
+
+ +
+ +
+ + + + + \ No newline at end of file diff --git a/tutorials/e2e.html.md b/tutorials/e2e.html.md new file mode 100644 index 00000000..81bf8603 --- /dev/null +++ b/tutorials/e2e.html.md @@ -0,0 +1,499 @@ +# JS App Walkthrough + + + + +## Installation + +You’ll need the following software to complete the tutorial, read on for +specific installation instructions: + +1. Python +2. A Python package manager such as pip (which normally comes with + Python) or uv +3. FastHTML +4. Web browser +5. Railway.app account + +If you haven’t worked with Python before, we recommend getting started +with [Miniconda](https://docs.anaconda.com/miniconda/). + +Note that you will only need to follow the steps in the installation +section once per environment. If you create a new repo, you won’t need +to redo these. + +### Install FastHTML + +For Mac, Windows and Linux, enter: + +``` sh +pip install python-fasthtml +``` + +## First steps + +By the end of this section you’ll have your own FastHTML website with +tests deployed to railway.app. + +### Create a hello world + +Create a new folder to organize all the files for your project. Inside +this folder, create a file called `main.py` and add the following code +to it: + +
+ +**main.py** + +``` python +from fasthtml.common import * + +app = FastHTML() +rt = app.route + +@rt('/') +def get(): + return 'Hello, world!' + +serve() +``` + +
+ +Finally, run `python main.py` in your terminal and open your browser to +the ‘Link’ that appears. + +### QuickDraw: A FastHTML Adventure 🎨✨ + +The end result of this tutorial will be QuickDraw, a real-time +collaborative drawing app using FastHTML. Here is what the final site +will look like: + +
+QuickDraw + +
+ +#### Drawing Rooms + +Drawing rooms are the core concept of our application. Each room +represents a separate drawing space where a user can let their inner +Picasso shine. Here’s a detailed breakdown: + +1. Room Creation and Storage + +
+ +**main.py** + +``` python +db = database('data/drawapp.db') +rooms = db.t.rooms +if rooms not in db.t: + rooms.create(id=int, name=str, created_at=str, pk='id') +Room = rooms.dataclass() + +@patch +def __ft__(self:Room): + return Li(A(self.name, href=f"/rooms/{self.id}")) +``` + +
+ +Or you can use our `fast_app` function to create a FastHTML app with a +SQLite database and dataclass in one line: + +
+ +**main.py** + +``` python +def render(room): + return Li(A(room.name, href=f"/rooms/{room.id}")) + +app,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, pk='id') +``` + +
+ +We are specifying a render function to convert our dataclass into HTML, +which is the same as extending the `__ft__` method from the `patch` +decorator we used before. We will use this method for the rest of the +tutorial since it is a lot cleaner and easier to read. + +- We’re using a SQLite database (via FastLite) to store our rooms. +- Each room has an id (integer), a name (string), and a created_at + timestamp (string). +- The Room dataclass is automatically generated based on this structure. + +2. Creating a room + +
+ +**main.py** + +``` python +@rt("/") +def get(): + # The 'Input' id defaults to the same as the name, so you can omit it if you wish + create_room = Form(Input(id="name", name="name", placeholder="New Room Name"), + Button("Create Room"), + hx_post="/rooms", hx_target="#rooms-list", hx_swap="afterbegin") + rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list') + return Titled("DrawCollab", + H1("DrawCollab"), + create_room, rooms_list) + +@rt("/rooms") +async def post(room:Room): + room.created_at = datetime.now().isoformat() + return rooms.insert(room) +``` + +
+ +- When a user submits the “Create Room” form, this route is called. +- It creates a new Room object, sets the creation time, and inserts it + into the database. +- It returns an HTML list item with a link to the new room, which is + dynamically added to the room list on the homepage thanks to HTMX. + +3. Let’s give our rooms shape + +
+ +**main.py** + +``` python +@rt("/rooms/{id}") +async def get(id:int): + room = rooms[id] + return Titled(f"Room: {room.name}", H1(f"Welcome to {room.name}"), A(Button("Leave Room"), href="/")) +``` + +
+ +- This route renders the interface for a specific room. +- It fetches the room from the database and renders a title, heading, + and paragraph. + +Here is the full code so far: + +
+ +**main.py** + +``` python +from fasthtml.common import * +from datetime import datetime + +def render(room): + return Li(A(room.name, href=f"/rooms/{room.id}")) + +app,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, pk='id') + +@rt("/") +def get(): + create_room = Form(Input(id="name", name="name", placeholder="New Room Name"), + Button("Create Room"), + hx_post="/rooms", hx_target="#rooms-list", hx_swap="afterbegin") + rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list') + return Titled("DrawCollab", create_room, rooms_list) + +@rt("/rooms") +async def post(room:Room): + room.created_at = datetime.now().isoformat() + return rooms.insert(room) + +@rt("/rooms/{id}") +async def get(id:int): + room = rooms[id] + return Titled(f"Room: {room.name}", H1(f"Welcome to {room.name}"), A(Button("Leave Room"), href="/")) + +serve() +``` + +
+ +Now run `python main.py` in your terminal and open your browser to the +‘Link’ that appears. You should see a page with a form to create a new +room and a list of existing rooms. + +#### The Canvas - Let’s Get Drawing! 🖌️ + +Time to add the actual drawing functionality. We’ll use Fabric.js for +this: + +
+ +**main.py** + +``` python +# ... (keep the previous imports and database setup) + +@rt("/rooms/{id}") +async def get(id:int): + room = rooms[id] + canvas = Canvas(id="canvas", width="800", height="600") + color_picker = Input(type="color", id="color-picker", value="#3CDD8C") + brush_size = Input(type="range", id="brush-size", min="1", max="50", value="10") + + js = """ + var canvas = new fabric.Canvas('canvas'); + canvas.isDrawingMode = true; + canvas.freeDrawingBrush.color = '#3CDD8C'; + canvas.freeDrawingBrush.width = 10; + + document.getElementById('color-picker').onchange = function() { + canvas.freeDrawingBrush.color = this.value; + }; + + document.getElementById('brush-size').oninput = function() { + canvas.freeDrawingBrush.width = parseInt(this.value, 10); + }; + """ + + return Titled(f"Room: {room.name}", + A(Button("Leave Room"), href="/"), + canvas, + Div(color_picker, brush_size), + Script(src="https://cdnjs.cloudflare.com/ajax/libs/fabric.js/5.3.1/fabric.min.js"), + Script(js)) + +# ... (keep the serve() part) +``` + +
+ +Now we’ve got a drawing canvas! FastHTML makes it easy to include +external libraries and add custom JavaScript. + +#### Saving and Loading Canvases 💾 + +Now that we have a working drawing canvas, let’s add the ability to save +and load drawings. We’ll modify our database schema to include a +`canvas_data` field, and add new routes for saving and loading canvas +data. Here’s how we’ll update our code: + +1. Modify the database schema: + +
+ +**main.py** + +``` python +app,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, canvas_data=str, pk='id') +``` + +
+ +2. Add a save button that grabs the canvas’ state and sends it to the + server: + +
+ +**main.py** + +``` python +@rt("/rooms/{id}") +async def get(id:int): + room = rooms[id] + canvas = Canvas(id="canvas", width="800", height="600") + color_picker = Input(type="color", id="color-picker", value="#3CDD8C") + brush_size = Input(type="range", id="brush-size", min="1", max="50", value="10") + save_button = Button("Save Canvas", id="save-canvas", hx_post=f"/rooms/{id}/save", hx_vals="js:{canvas_data: JSON.stringify(canvas.toJSON())}") + # ... (rest of the function remains the same) +``` + +
+ +3. Add routes for saving and loading canvas data: + +
+ +**main.py** + +``` python +@rt("/rooms/{id}/save") +async def post(id:int, canvas_data:str): + rooms.update({'canvas_data': canvas_data}, id) + return "Canvas saved successfully" + +@rt("/rooms/{id}/load") +async def get(id:int): + room = rooms[id] + return room.canvas_data if room.canvas_data else "{}" +``` + +
+ +4. Update the JavaScript to load existing canvas data: + +
+ +**main.py** + +``` javascript +js = f""" + var canvas = new fabric.Canvas('canvas'); + canvas.isDrawingMode = true; + canvas.freeDrawingBrush.color = '#3CDD8C'; + canvas.freeDrawingBrush.width = 10; + // Load existing canvas data + fetch(`/rooms/{id}/load`) + .then(response => response.json()) + .then(data => {{ + if (data && Object.keys(data).length > 0) {{ + canvas.loadFromJSON(data, canvas.renderAll.bind(canvas)); + }} + }}); + + // ... (rest of the JavaScript remains the same) +""" +``` + +
+ +With these changes, users can now save their drawings and load them when +they return to the room. The canvas data is stored as a JSON string in +the database, allowing for easy serialization and deserialization. Try +it out! Create a new room, make a drawing, save it, and then reload the +page. You should see your drawing reappear, ready for further editing. + +Here is the completed code: + +
+ +**main.py** + +``` python +from fasthtml.common import * +from datetime import datetime + +def render(room): + return Li(A(room.name, href=f"/rooms/{room.id}")) + +app,rt,rooms,Room = fast_app('data/drawapp.db', render=render, id=int, name=str, created_at=str, canvas_data=str, pk='id') + +@rt("/") +def get(): + create_room = Form(Input(id="name", name="name", placeholder="New Room Name"), + Button("Create Room"), + hx_post="/rooms", hx_target="#rooms-list", hx_swap="afterbegin") + rooms_list = Ul(*rooms(order_by='id DESC'), id='rooms-list') + return Titled("QuickDraw", + create_room, rooms_list) + +@rt("/rooms") +async def post(room:Room): + room.created_at = datetime.now().isoformat() + return rooms.insert(room) + +@rt("/rooms/{id}") +async def get(id:int): + room = rooms[id] + canvas = Canvas(id="canvas", width="800", height="600") + color_picker = Input(type="color", id="color-picker", value="#000000") + brush_size = Input(type="range", id="brush-size", min="1", max="50", value="10") + save_button = Button("Save Canvas", id="save-canvas", hx_post=f"/rooms/{id}/save", hx_vals="js:{canvas_data: JSON.stringify(canvas.toJSON())}") + + js = f""" + var canvas = new fabric.Canvas('canvas'); + canvas.isDrawingMode = true; + canvas.freeDrawingBrush.color = '#000000'; + canvas.freeDrawingBrush.width = 10; + + // Load existing canvas data + fetch(`/rooms/{id}/load`) + .then(response => response.json()) + .then(data => {{ + if (data && Object.keys(data).length > 0) {{ + canvas.loadFromJSON(data, canvas.renderAll.bind(canvas)); + }} + }}); + + document.getElementById('color-picker').onchange = function() {{ + canvas.freeDrawingBrush.color = this.value; + }}; + + document.getElementById('brush-size').oninput = function() {{ + canvas.freeDrawingBrush.width = parseInt(this.value, 10); + }}; + """ + + return Titled(f"Room: {room.name}", + A(Button("Leave Room"), href="/"), + canvas, + Div(color_picker, brush_size, save_button), + Script(src="https://cdnjs.cloudflare.com/ajax/libs/fabric.js/5.3.1/fabric.min.js"), + Script(js)) + +@rt("/rooms/{id}/save") +async def post(id:int, canvas_data:str): + rooms.update({'canvas_data': canvas_data}, id) + return "Canvas saved successfully" + +@rt("/rooms/{id}/load") +async def get(id:int): + room = rooms[id] + return room.canvas_data if room.canvas_data else "{}" + +serve() +``` + +
+ +### Deploying to Railway + +You can deploy your website to a number of hosting providers, for this +tutorial we’ll be using Railway. To get started, make sure you create an +[account](https://railway.app/) and install the [Railway +CLI](https://docs.railway.app/guides/cli). Once installed, make sure to +run `railway login` to log in to your account. + +To make deploying your website as easy as possible, FastHTMl comes with +a built in CLI tool that will handle most of the deployment process for +you. To deploy your website, run the following command in your terminal +in the root directory of your project: + +``` sh +fh_railway_deploy quickdraw +``` + +
+ +> **Note** +> +> Your app must be located in a `main.py` file for this to work. + +
+ +### Conclusion: You’re a FastHTML Artist Now! 🎨🚀 + +Congratulations! You’ve just built a sleek, interactive web application +using FastHTML. Let’s recap what we’ve learned: + +1. FastHTML allows you to create dynamic web apps with minimal code. +2. We used FastHTML’s routing system to handle different pages and + actions. +3. We integrated with a SQLite database to store room information and + canvas data. +4. We utilized Fabric.js to create an interactive drawing canvas. +5. We implemented features like color picking, brush size adjustment, + and canvas saving. +6. We used HTMX for seamless, partial page updates without full + reloads. +7. We learned how to deploy our FastHTML application to Railway for + easy hosting. + +You’ve taken your first steps into the world of FastHTML development. +From here, the possibilities are endless! You could enhance the drawing +app further by adding features like: + +- Implementing different drawing tools (e.g., shapes, text) +- Adding user authentication +- Creating a gallery of saved drawings +- Implementing real-time collaborative drawing using WebSockets + +Whatever you choose to build next, FastHTML has got your back. Now go +forth and create something awesome! Happy coding! 🖼️🚀 diff --git a/tutorials/imgs/quickdraw.png b/tutorials/imgs/quickdraw.png new file mode 100644 index 0000000000000000000000000000000000000000..37c8a7cdb43a79c0caf51444ffecaba002c8fff8 GIT binary patch literal 67243 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

s=+cJbxuj~R){8-D ztWZ+%Y(jA`kOY$kH9dLuEzvZ3A$qa@FYYz;M{KLxOcjEe2g{2x%xd4Qqh&63ZYr|# zPH}%Pap9y^khyn{vwI+ET#d*{Y_cdSBD>ZOgtn^41evI*3z(T4byufnlGf0AvQW>G z%#xvutoNUVlD{HjveOc^Cv&yABq#qvzyw3rN%48$$fNcLaUh$;8>{SB9Q2-vny4Cvi)m@G7_@|!r zDyituSs$V>;(F)a?*$I$7^2~82@9mH>v+lNcgi}}rZ9XSMv3IqsrSC48SR4F>)xqy zjVr%rH(U#ZN30h!l?dpFH40}aJYJ)+zvcDOBHKyeD^*U-NborA?LTr@MTJ!KRYCD) zdy8n2^Yy})y=k~;?&8UbdmhW{hJdb_rjM|<7uGx8TeXxXApC8{wql+Hs4+1-9p8u9 z;f=!Scw!MB!-TAD&OuHoh53Rz@w>>NyPLehm6Y1cYbPOpaWUe~(Og35ilsZIyTi%D zvOU*)sooGLxhZ>){r5kz-(wxu^&>|i{%Xs~5~)R#&W`_HGh0qwi0lim@ykf{r%Q#F zP3YbaiURtYm0ho&s6itYIl~k^bgTdRHv`LctG{&t51X(NCXj=2O z_mnbT{n!M{$N7FhV(!}5`jl@R1HA-{OYEiQ)3iJ<|4)_^jXBEz{50U3z>sJN=1Em< z$5X2B44et)q~9Gi);R;AO0Uw_&2F^WCG4K`K2dmaPAc7qURiT8Kdx_zCDYX7!bz<5 zu8z5!vC)I=2#C4GpZkF>)mE>Iv`1!1Z4G2nv7P=VM^?Vo54j&4a^DHpf4-OfK`bwb zMX;N?(!XKbovd$7y{;nHHlw%lQ_aPgPUCT3H)(^pF5>jQ_o)pz6}ILR^?ED|O@0Yq z2RSTX(tg~!R8Y17e~hs zy|UkKR`rE5EO&;OzJAeyyQOG``@pC{qiNT~Z}eY)?D`(sUP#V5(AzHra8|AvvzsmM zHV3NUi+v=Pm_yZ(iI0>7rBk2%aiVZ{{P9{c2WKqScPZ$uS|E0`=Wszl;PjmE|c{+mF#w5)^aNP zM7G+$5QFHi?Np&2JJQ_Lrx*^kxySXGtk&Ci1ClnET1l!ICwa`FEL=G??U-6zzzN3rD30^SuD z;!eCQICIX-o_E-d?RQv}%`>0t{s6%ym18HoonN08tMiz1UScON6L^PhMwV;#Xk|6K z4>B4H5cERbI-ZIFzlnVI3GMZLIlGKLHU1g;M70mi9_N}5F(x!2#T6I}lY7#Q6+d|7 z%G~)yU0qS?%R;;vb~;sQ2?`z@@ixNx;}aH z-S!_Wq8hUtFA>4uQ6}rZCB7)9WkEYnjvEPq4({kzACFeH0;1E6_WOq2qf2Tm=R%STClDA`ds;L zen)9Pb@;IBI(1U}|G`c=1m0;zbgvG!KH-8Dnl-4;10tv!p-w}uwt5nODFWc8rA+iEZKDbdNND>o6c^=K1G&(kus_h3&aMI zl5HYxe2JRtgxQ&&q*Erl*&2=$`UStXF1d3psGKo}T{Rw^Qf3T)J;->8)DIfsc!bm~ zJJ=S_if~9P0EPOg69eOa;~Ogy#s5`Z=b!v;{Ycflb{2hMs~rSvspg9#Pp&4W!>mU> zyZI`)h3X@5&`2!v_AsWgOTyh!>A z8Kfdc4Apa}3`>*o#iS1Rr4XM7PM+E!8+dXD$(1(qBW#DSM39Sv%FtCFWNhasq*y2g zdI}&inQ%k}C7?PTUt@QU;8hSHO)EBc`e^C`12H!8#yQ?t-DohLDHRHnxmty6l=Ej} z!a)?*Xlrp2sH-4;qCCu+^+oBT*121L0=9oh=A+JnJ_l3c)aL$9v$;Pn#RpYO&3q)t z*p6&vu6jC)S4_ueUcWC9l*jTOqdcn9I|4(a-7+JkVdmmHrNTZ2gAp*p47;;rrFGJ& zp6E|FWfBlv{DyT~qqLfP+U)Mzmnc|2bJ2P=m?d^F$X>@otx=4nI%#~VW!a>fvzbAT=fM& zu)6X($|{MT=)B+5YCoJyxH;yUWR}Sk~5D?0}Qi3^RZ7-&jMt=x}=8n{F3Hc(C6BePwuLi ze526zd*rZLD*65Kv-okp-`}hq<+38cC`t6rR|p8$qHj?-o7$( zKm_RQxim5+*_tR^NO#5Gbu()E9Qp~fZC}?7S}(E`8##VAULru~jm$b!8WDC+e7@6& zJsW3{yXM;S`(fLR`HkgiA!XR`*F{Ms8m-JLH#8!J8@MiuUYA!3wqe^kEs-!Qc~>L; zRO5SH3Yg_{Aq_H~(oE|L=$3{w7W1P{R&x_qR~XBx?Oy42>(|~gae9<9kT?n;R}8;4 z3ZgqXYJ)!f_RA9F-cbUA1Vrs@9-zsn(S`f|+{;V$YSW2s%}-y+ra%Z^=$$ zL<0O-!3K&WV1~>{1-C=@bg<)%>O~y;tg;G2>jXuV+hzA(tRs~;AW1o^=Y?>{McH&@cOPAvOJmd z$ns)T?51&A^XuDPi7z|nPdUhLH9LM+vLXPVW9$N-U{c;{!Y?uuD4=mn7RP zN)5>!58RChNW7V_jF+madTORfq3-GRtgQAh;pS+Jyi3k+9xl&5fT<3dx#>=`C2Py> z;_71qk2IQ-!}J83`_lElSvyl-dP@-=!(4TNQ-lbX^L4e)TR@%QaogL*<4g3Tq$;k( z8lc_yG^dj`1}&*B#4f8mnG&-EIMaGLoI{kG*$U5LlY%f#P7#E0ayjeL*{iZ}cNTOz zKC`p#nA;LQKz*!>QZn5iH@aUk=~8R^QLX*Z{=Pe20BJEYPwp|?JaT#Be!9W|1fBA6 zw!iMUj}9)}aGf*!;SbW1hr`1vs?=y&ZygeTv$b#t*ka@ZW9)mXjWblwL9lkqRROD` zgQ+Y8mDMBA`O>}$yHgH-Ez-N~lQObulvBl& zd*2nqD7x%{i2T7{V0*>7f5|e-8+QGXK<;z5{)uOu?->c4rsyti#K>^d^DLTAp?eu! z>Tps+`QIX=EakzboH(`?r~3A=hf*TYb?^Co_#{)za{3B+Jr^|1rYpP6US-Qn0%cg+ z(%#Cmaf}(3S;uWjKeZpHzwpA)VwX<~%ke4n?)kDH14%J@P43pBYiGS-px_wkq&WXH z$=>1|+9fj$MwMYYLzY?|_iFK```>)}tU#bd^pB_F|JD_qdn1xtt$?OinQ!`x2g%xm z|Ni=IPY#NQmI_8c2yaE54^go;aiY&u_ui#`JV4xoJ|1`m$63V&Y#WeQ3F2jZFie%QTqL;Y( zh(r)3w2koso@L)dd89 zRV>hT_HP|JU_HM4{qgKfV1igh9!`gCVcDxD8Fw|J_HW6SpiVkA#mLK6cDvT;;{qM> zOj#x4Wg7szV1(J!lMX<=yKF_p=3}6mK}_rZX5P*F?Re3U5QWba-92Pn|H3-O!NvOR z(-O4h3O44A{?9Mc;J3?hOUZOF)p74vF>;h#w-%Cfyh}SaxjpcD+N~BfB+6DJkQq%g z+>MwPAYA^4Ju7)G4Q7LFcN`)2vv}0&SksiPJ&sK_vzrviZdQ@a zS!sI%23yyr1Q;i}R@~3OW2H@Q6Gk49?M+kFcO+}pX-0e5kINE~e+yhbUbkec;mkZ}>6rCI-d%$_ z!ZfV)02anf+$xvLaF=1B_`#C&Kh0NTj`uJ5+=bo~Q1>L=uV03xkF=0qB4p(}v-K?{ zcKK)Cp*d=s{rStQqkZWBJIcC$E#%+a{Oju!rZJklw}SXVHqlz|+us%zX$+Q^q-@5r zv~*Zy$1ydhGoPFfCtRlsJlY_}M{nMF2Xj|g?vQha0=RZ9=b4zJB_asPe^?CY6~}pJ z84Pc6AeOe)uYP~aN3{A)g5q8P(3gb*ZpDyc@p*RQ*1-3z=su>!4^||`cLyuW@=7#V68XJOMAZ{OK|sX;o+ODEVD4=WRELiE$1Q;{_8bny zEzz#*1aC7S6OSoMV}b4k_^H3j%(lgdmH|Edb(?$8FHKaWL?Xp-f+Ts!Niwg5A*P5m zo>}A-c!F^o#B(#hp8qO$aLL1_X? zkrIk70s>N`cT_qEC@pkES5QA?DXwB4kh{`=HDZC+No#Pb=$JhSpqKEsM$M}mELlL zL{aS#?lS(voKK+cLR1bEpJ**-}uPUNYK_B)V{|)TH!Pm9ho|!>BCrgt-m1X z*1_%h-J=ZhsYI(V^|!9o4u7?=Z63$-I4;q_VPLL>0YbEuoh}=w9RhP7$XKbSZ8m5K zf~T1MfUqpa==Vq?EMEF>BcDe<-VqjYRLo%jU&J(Rap<=Qb+@*u*em zB(n%uN<4H-jm5RYZh(r|VzR!-C$2CepH6e@6AHTO4vzqH3AZ0S?=4eh6-`>Tb9mDV z@SNey3${`ZOj-Swp$>}h_}I9>5r69-69Yk>r5>hvxcgH9(d9-4;uQRdbX5GmwV?A^Wv|SO) zp1GzYLfxenVv#g=;yGZqq1|q!V?9-?(|1rh3E>HYT}byEa`#*?RQ@#SB6}!$mu;-a z4R+mni)Zn1>(R^GO?(!`YoFiq41VjiK-Qy1s4J#Y61t8bK=fCs7xDDbn0>o+uy?C~ zovCJ3ytOH-jSjI^=9i>1!D*(pF#zHiEt=N`c;}z4Zov1G>r@uBzv{N1l&$>mPY^Z0Ym>M)Vb7t>7r@|ofd>54uRNu7Fk9;; z(rncHcA~5ZqB0F%cY%nvjh0y-=h?-eGJrz+9s*FVPmrn3|1ywwDCxDaw{-i)RG;y) zmN4`FNC4OYXJZI_x0nE5eLcMggl>9jTLdLr9Zvh!2iDVj&r(F$e>@^#=ffl@8@#aI zJ)g@JTi@AlS9`Qpi>Ek6de|fD@$k)l=1p6O&VA9BjvVJmx+Ps-NTrK9B|D!(-Od+k z{W12UCrBH2Qv0aMbFD_en==3gqq*T9dWr!~A!ycIKH!WYT;<`5*PHb%g73P9+UU%&7OfF3a%-DbfAUANmh_63pKNLv7-esMF3V1~^Jh zS0)0F)Yi*Vc0G8k!m16vKn=%A_&tD3WHOU^wGgFGJW zcnPc14=4qeI$43>9%h4HD*ZzAwy-QK-fH%k*{`H(E_9WnAmMl12)u_kLaI*<6}z%X z$ULbMe|Tx|;M+#i#umPM)*qhPTJW-IEcCC5Di!xJH865q{>(HNcGZ57?EHkTH=aULAcfBF&YOO z0vw{L9=D%`Y?Yd&Giut?<}IH{TDVpXl!{qB=(GVG4F93VBOa~c^mnSBZqYvzS6aGn zc)iw8DbY~z|8i6*(w@6FL7jtP(c1mdo7>%5P#Ae$^0esJnO{o={?dLSMyDvm)-8exwKj{HtjEe)$4FMGuG>v^KFQ;14^`Lo3Y)G@_E6t zAl;isBz#?>`LpGY)$f`|EAoo4XhjHtnaeHjfQ}Z^);ah}hPx^tU8yu@z(L!e{vYNm z9L<0)UT*w-)raO!pdNYp{2PVcsDdzpF18=)wu}y#s8$5*LIVaWt{$cb`)ERd)pldg zqg%jRI8R~ELH)QAK=+}@TBEXlRZ;RmM(?X z6i4tQ1m%%uPgd;a(z~pJ+HV%n{Ec?gu9KA7dCytD-B5n=s>ntH>)%=uU!g#N(*`r~ z)1X`x|I@JUMmzmU80-N_W~8E!6Jfb^<#uCTZ$y!&vW^M7VvbR|*ewj=L3MWBx&y53 zBu5$OR1VI$#tg%e@5n_zmfFAQBv2$|L7On*{q9?tmG*FMDQg!)QpPRD3NfH8``pl_ z?8+JM)}2E%f}MCyB;Fg-Rl~Y-i$v?#X24u`56S8P9scNIGsSHK*$ryOb1E!JZr1gR zxq6zp>JcDh`WYvVlluCQ9Q}+o9E4EF8SeS6oG!dEU8MBdt&G+62cQsntE?&*>0B5W ztl*9Q2JQw-L}I-N{*c*1rfzd1*Dm%4&1Nt&WNn@sf5YHCchsnJC=S@D(`UC5Mu$Wo z^|@5(zg-`#{t!h;&;K!(g#GX$N{tsZSgUe`x2I%vdUOC2vGOCnN2EQBUaK&i?DpiK z(MvIcN%To6+WJDi{@{!3=DvUGoEARZXYl>HV(2T&>vYR1HlY1@Z6r06yA1Ryj&t2G zN{w~jF8=2QQ0cwSK-d2*D{|DZLjgLx+)=&vIK(83iuu0jM?;v-tA@@DeXy6>8fUSv z5=HgSwH2!%+@F@Wpa*81@9l|dpGB_qrQi3QTcEvr6ZU=3vce_?80R9ZYvE~FPm|fs z#FcY&Ke+N?BvH;{?P1jzG$65ZvyrGh(v)3g{J^-MO}22s64wrao@K5VjR5hxZy@sO z%oeQR4aSZ(2`pQ0DuCRjS2m)kxB{UJJg$W&6;M&#vMpWXv3hB+GacXSn3rKP;CD2@ zCN3!LK3iD^6A%>(=+$g?Q@K?zp7Ofnbvffb0zF(+Pnb4GiP>L;lflRMY3#;^PHQU# zz8YPb#`$m?C319^nv2hT#w+@34e)0{725aX_856G>HW%1F1^lWB{1X;IZ6%{&#=lvmPPf6T>)(k&_{Ld|Bb6S;c^?K@KLsaAEXW4 zPjmI4u(l#_09=hwh5Pe=FV+f*xL9a)11kFUI?sMlx`(KJ0NBaCV_(dl@(fZrB1-zQ zuUo3v1TEWIS>F5vn=uY2hQ5f%P$ltI`G*v1`KKz{S^C7ewq|7*Atjhm^|k z)TyMNpMsBeSs^ZIC;I4GEL+xar-cu?v|Q5;Pms_tdwQCjUh$uMw) z5Ss-;YF$KJ)V%@g-3}!eo8R6uKSBj!4Ir^G?HuIFY0m6|-Ea03Otw(}LXOt=KkH5w z>NmHCNm1^wUhjl}rFNU7$2>QT;8f&9c8BzwFLjw#eV84U&c_rb*jbcu23%+^NTtzI zqgn(UWWYi0B{sf$*N|4LZ6RN7%4)|vV?A|$48FhLYqrO&ST6%}7zj)+qSA%QUEKY5 zDmUKvtnOV$oz#a&rX!+jV_e7X*3;WDeaCpOh~9Nkz@+yX&~R7j)iSNJ)G|MZSJZj+ zIK!XbI$Ycd1=U|Tv$bzExCzK7C3V24^+UGbW@sDbvop-nxw<{B{f58|k0)QiDS!Bn zo?B6b7N~YN5YL*;VPzFU z9Gv6UQv)9oN!|1=b7mMewSD{!dh(HTWKq|Wi_ep>hs$+1sU0EZ~<`pXxqTcbbIBbFfJyk@YgECQ@` zBRxK)wub0{j2L*I=}-W`hvJ3z2I*bqaG(=oK$x;nsC3kn-$0~A%RP^8cS9Wgr)X^cP^PN0a|Rx?l}o&tWvoT{P&F2W`bAeOdL zZ{gU8f8D<72(xHI`8v|*yOGGM4*n*g;7Ad6=P0lVXWSMTE%noY6*D{3m!vs5>u-7P zHtc;~(InjP@<_*8<;h+CJ!UwwKT7^cpghGxHvfZ-{8zxK<~MZCfGC5k00ggZb!oSg z_ltE-8c@5Mg2h(wRc`V5I$;-tjNHD`+okoHES#>B@VA!@4`=ompYni%e~I<7kV?omo0E zMqLAe$^%>0+9>Bia8gk#{%ud})|?a^yr^`Uxvod4$GQvu-OWjAwAP{4FHOBAm}Ty#0VUy_jL=wQ$QpFuYKDfKdBBcFLYBR@pv4GeI|W9MEdXnk+U{*>r9asHv&Qy6i-FRe3; z*6{qKTxa#)J+-gaIznH)=uSwgRc5!I$m6uUr7HQJTy(f%owEbtGsk(~i#O&*7-kZ)-u#oxm}SRM`5C2BS6 z0&mhaWW()4VR~-;c&J(93-O@T6eVicY{!cJt|CZgp6NR{QK=00ScDEetp}U%a`T&M zJ?!>#2_;%rySZ#6(z+=42Ecgm)sipqQtlGwdJb>ii@r2hFw&9J?Bv{GDTVxuy3c*} zjZXW`{P&sBw&g)Ql@|YAMD~*Db7cOf@Xlq+l}BBW&Hg##OxRJyB*-ue8M#1&+Er4; zc~93|>pKL@LTm7a@x0_E+uCkTHfk+r0C_IZnb#6Oe0zHV4k?mrCk|=fKg+KloOu#g;e`9FH`W>}?)#<)SYE5v3BdKkp-3$=ws-HhCRU3N%eWl=|^@bGU`!fN^8i z$5O9%hno?i;QBl0+#7;XfEeD74W+XyB{%0UWE2vKlWHNP!zg(1IWP_iQho96Zo_vP=eK z<@ztUh90_@$?&eH*%jx;TlhX$<8Y=GJ&)ApC#S}177F&_kMMUI&VqnS*$3JH7mxKK zwkUZK(7`pTw)n#=t*fI8MW`TF$ARea_sav%)9%N*yBa{joNq00KC5f>Bm)wwSsY7M zau#(EU1!Ir7qUfYuzP;q=Rw3*_pwExNUb7qfU`C!+|aqi)zNtRL8hzC@+n!>FP#mz zN3XHM8_&kyoGU;)vFOP^K#7*5bD70XYAWWB-RmIgj44-X#fqm4`ceExwrOR$QIzrY zot5R0Zn!&xLq}X`ed&78KIIV=k^)26mtn8CM0)r*88EoNXaWrFtVy5KEXlL%t=J9q z2-W!ddkn31MjYpmgf(?{$C<4l@n(kl`c+`t7eymMYCyk%RS(sa@=q^|X1jSvygP@B z#rH5E4?;yByR#+^$(nlO^FHQZk31lqDrxL1OaCJUdCud>%d%q#NUPlxdDi7FT`vrx z4nPcZzbw~nvq-Z{yjDK{Zs$3b>PItUUp`z1Jwoc@lk=mAxYt-8T~YFq@i1wGua&Zn zfle3-$jGeirsVz)4qA*nR?SYl<2cvgFc zM@km-Wk7cUb~e<*X@$f;zIvVk!-R-o|0|9Ha5Tw9vf%l1fphg$_AGLfn7J1Ks9O0W zz(T40N7(q{mhU^Iyd2O7m(Y|*30;<*9p{&mPG9?0%;y?HoI_-!`(n2+Yq&RCpLlF6 zWraquxr3Ay{~5n)DJ`{p5k;x8cV4gSGfr0o#Z_iS)mEFNnQ@PVsaFZyD1niIuxwp! zjL5Le?FgmzTVuxt^jmBf(N%sY6_6Ce;nZP4T_nUu`=0|#R=5V9hZll%I zWY&lKMLe^xT@)zpnuh3Jm_r$aNV|YHOH+U7+%wyZzUC{`5Qqx$+_xC6iRmSsiop08 z3dwBULJ=4|UhiRLf}cPECZJvN$V(gK$$@D>D<6yg=}mD%x0BDeI^71nZIGKn<|}Al zsaDEA8N!H1ZX6|i}k1%*6WNEVdw zK)VAvtP41<%?f73d){g#U3r2^*9~@ zZfNY~GB!cBu{A2>f?JaGK+-Brjq6>NW+E|gb-1Leh3dd-)JzK zeB_FhH^7!Nj9T?Lr5Mn;J8IPmf@QggAIc)Uu1MQDCiJij`f|<(!=}2MpVJs|Pe&K1 zpIo0dQiK*q%hJKngx&k&eT4^zBc7e#DtLj#=^|qb$SvXgd#fVGTXoKJiF1B)_cT%r z`r7Wx+p+Q6kkf?_?pehi5!+P>Gwa!53aQ-%|GV}@A1If=q$J$5G1KH!-)0+p3*uJ` z1skySITC7U4@9Sw{-mL9XsvpnYOF6jY66p0d-~$tC8+8ZUQII_JwwK;k5l|BVXLrm zTPYzh@6cNCuIJWD8r{AiUxkmn>+hu8-vOXR3>4}h)xE9>mKC6HEli}8<TIxQrg{rnCR*CPhP% z;o;%kaWSQpG)$b&uF~#VLpksKz3U%~X|eI>nH7 zVe)UD4{+I3%zq#hvRfcKMob9iMRm1Wbrd2F;!ARO(_b`$28ZSPcj*z71pT z!2mknFJv(oVG#v#Nj#t3UR~jm4?azLC!Y;ia>oJWV&1;Xy>bXR7YO?RI_oUt0VnVj zS9lN~RlGYe(3_WA?Uw4%B}E^7th2&c2~t4Wiw7d8x<_RP*y0^UI1PCVwJK6upA};L zoaG}_kD3L(>RZtk!1ZMGocr(lZBbD2x_~L+y~M_=XSXl_)UMfyw^AO+BWw$YJ-jJw z6Jm)_(w!9Myf2IIVAwg-6@qDCpUlN!kO&I;!)<)8e0WsVv|13b;hm3s3-*_OA zO6zVnU+$=hwutVXF}AMP#n-bU=8z1+i!;!reGPywuTF_jROdkcc@g3y!!S9J)@6Qh z*@kQ-q~Iyt754k?b=+{!8m~A`7Nh#s8#-$@MizVc$s)cf2rtfI zcet{GILu_=?NzE@a@L8zuNx*4bIZ%9h7UX>C|`#|r`JYy1(o zoB0z^h2&F{Yy2wwLy0f`o+T3<=4Zf`24P9&#;AxWZOH1!tb_3+Te}DN4AM;JM1VxB z>1@f?W{&jkpy^yS1wUA^y-0MSkjg+A0jj=FkEb)NK}PJb6hIRWV>ke3&Z#IEMm@X@ zQU~~6Zn|zIJJ}b9x9j6}R%FP9`X%eth6cW9lgegPVBW>VV)RCgB4=hO1g!W47ZHe8gAXUe~L`(78>#C{> zuM9XqV1CzHX=kuI9pBVni#x%);tdWTv3ujj$fAQF#|J=Z9XAK%=?AlVx))2phXM>k z!LA^AfX)Evm!kh^{L499HM(9X5;|!EZ1v83Rw~v<;Xr9Z=HtVbz560hzFRwe9qRGI zqPSR6qmTWFsh@4%BT3_k2Z|`dpi}0)60PG{B^?l2b=xAfzPDtXD1&C-zTMOsF7jSx-j7kR37! zeJlXajHeQN26!Au!3t@{{@a=OvQxoz?e zWDTXux!G>NUvoSxqmDFA@!_A%P7R{Wyr#SMLvOb1F zuP;=wmn75iB~GZg0`&!o7PrVk$f?muNGgBF$NP4UOw2Vs@_3RXG{|;+=RdHww2<4R za8qD3xX*U7Z_ZxE&QW0-iLps8A*p6_jI)tp&zr8i?yXIH4k!KlcO1-4Jy)Irq&=Mj zgBb08ZuI3G32O|#nJ5X;K_2lpksj)c9fvYavh23^JP_M5Fsf;?g9;q5VVqQFwttaq zkSoy9lZ-k}Er1^#(1}vn|N8HeXBfDeC4)6hOfaru_bX3%A!2K0b)hy5eYycHCBj1u z!`}$Oi;MirT&PO0%gTW-2;u=t(Z13IY>g=Pqf4aBEKh~jMsj)~j#Q$!BrmZv}c*%BTB zB2!j&&B?PW3g?zpnu5i>M%10NsqJOH_YA2m*=*Je4OA3mX{2%GFDcgBbAvPx+Je!= z#7^E2yZhn`juhb6-1Y&JjDL?PL3BW7;Fa(Ip5e*l)3^NbVDy@G&W^|8L{t7uu~1TH z+fXSyEg>SI6=@Ct>R%=mKxvn~EMPkuRRn2I6*PF--y51!O0PcO?V( z6QIckdY%ffIUzp)wI6Z!$qnf(E+4H=5V0RN)tniRgr5MT$8ui%Np*1HKZx+52W_bi zo|t~79vUV8YJ>7*b{spTeNTilsc-+;-t>r{?E-ZDYqojU9P+tH59e>`{K0(Gq!x(Q zU>?CUx&aw|N*M%&#nLR1Thyknm1HA~6wANm=)iPuA!NyB4Qd|G%iiR;}#n2W(Sc5TU1Ro4(_xQxt?1=&z@D5b%g0SWwV*mIXEYLhWhDs5d7a|dqLRs2v&Hc9{Tl(he zZOt@$8{kQU69enh>?B5LD!}`7pvev-28*P5go3+BLfn_qKt5EwJA&AW+0DZ!y=aif z{@hgW5M6oaLL~-_^4y{dTXpaK?=l`W*bGhIXn?3OXaO^{UJlR|0pZ^|3%Sl;+Z8 zT%cG%taiEo86R#i^$plL-9q1(KeG<()Z#~RKY7dW^!1toD}zX2Ib>75Jp6RM@d|)v zi~Lp*F~gCD7KdjtJ+*Sd-4=%T8S#9XpS0B zSpY#BjGL^{;=IduK0$*erF$7>dh(fh0L={js{?hGE+UX(LhHoDtjj&IBxOzAHISmK z4*KvudBzYIBx1qpz*E){PwmzNaE$m>!(qBT2<;YnYGk9H6C$lxk(6giQtG+tQg4&5 zsys!0NK!f-LV%O{{oA@l_A`EdAfKDxYI%Vg1 z%e?Uhr?+Ob%&PSkls8ViuG%NH!N1a=vLFXKuRh3@2EJBE@uiUW@65X+H~!{a$^awe zO#g5QjZC#AKu{Hm8;BXyNpV^~lv+cW?lppAbJI_w(`ENX)(r6F5vT(x7{Q>(9}mV0ykxx}6 zU1YX}IH8k~^|2xwTp`@(i;!Bkw^}53*r;>h^>hf^yl7O3o0%e0nGVv{12tGWQAO@e zF|9wq)Ch^iC;d1G5^@zJ3nou<9W{C9nVbev)6~WHfO-sN@xq)Tcwmhw5kzo7Odfn# z(fmM}Pbq;EPhFUIwB*z_o;l*g3l)S297;W1qk_Ee7@5Q!qG0oBBP0# zOh(9mciSNY#{V90FT<`-7pV146ZZy3^w;JR5hcd}e^dHVy5Q$(ZTdV8(y2l!a*c=jpokpzGdiRV+SU-+34Q-R~V3 zNr7d1Y@xHEQ?>$1J(OeAJ|OPd>nf;ui#^b?ap+#4Tc`kSfT!QdS%l2ec-8QYwUX;u^8B`M2i5Djtjy8Z?rz<9@iN(5aM`agP{0CX39CA$gi-? zpG0)06GC|g<<4C@0pwk$Y^?<3#{M31;vyAifG+Xuykdr<=k*tuDU8*hw}8~q)MkO~ zJe*OC3L{lgWAJx`SWr`bVc)YuEe9ZwCelej_|Dut%=retV;eq~wHT@W{W2n1Td1`M zwl2AjeZMee0lNzzX{<46kr3eN$E&^Nv_mz2La1_5)(0=rxZH56F>YtjuDdD0@158b zKM7|a*X%07v*F`MOQm3R8vGYL^1v5J9f(sU>UTsaaUllDS@Nl5u3P^oBHyD!9J%e` zP%*llFVp7h{a5!KbeX=jX8Ffz@X(RL96^E5yD3Tnvm9HP9erDT?+u+h@_Kk3+1p&N zon(>2CRBDYf2ovTrT`k^eW^IBMD(GD-ry}Tp>t=DwikL{qS3vnN(^8c_CqFjY=qi3 zkXiyji@)FvIh9Ty8A$B!fqolC;?tERJD)NjdV?xvLnDIPgvuU@c8De&o7GU`sF|>U zR2jOfYmWImLPfwe81ejl|GjF2%r0sSQ!pDZsUF6Q7~;xd0K^UBL*Dz%S6=njO6#T= z^-UPkBHcF+DuQ!_`Pf52Bv++gXx{*fn6rv?K#Gf*!JDEb`@jS&15&y$DU1a3NP-S85_qNZ;5<>3>cGr&*~YME94!c$~>8n-urr(AW{sr7mTG zxtV=g*Qv<+r8ybw-Gb(lCn5offJR85H{5ij^UGk7=hkE;qPLKbIY`n}4?iNSdkZt4 z9hu*hTZxb&(^4ag(_HHTsg*DSK_s^*ZJ zjL2P+)8H{45X@o80mi2-f$-iWJ*1KH??dYS1*1v4dc`|vSRKBPO$>POiU5HG9=9O= z+K5Hh>2L1tQfb%tz7>E`i@^%MNTwqD^ zfNBfyDWG)P(5DONl(+4F%wt)k4|-t<>O12{8@D%XespC(SsngEnY)(-oYAylCJVh5 z)QqjD?17k3ooYSZ(+_8loK6H0a!N3Gq^0z}X^%%*D%g0pfIlj>3ZZFM51i8C&9z>P ze43C_)c;cf%1x!Xqas<77=!-LpQ$>|BV&%AuYrFM!sfy2 zx*-Bt{(M=#+<&s{!Qbo=-s{iTJHDUC2>E;QKFRCgCyz^ieaOK|{93Mqb)fpSyM8Xj zuXFNqA%2CGp9}FT3jbV)U-{(cLi|dft>X<^ZQC`hC9_gJ1piL=_zGgETUhquXTH}w>x z4cJG%d%&@Hh^vSC#knBGBcHBro(g8kTQIp&a5Z?jeeaRq*3eTA=hv^xuZ0Bu=dwg1 zNX0uZeg1b%GpU{a9mmhgPyNi#3Xhjmy{92n|`{vhS_!TG)Li?{E`YSX1%9_6_iUWl1SDEyyg#1-r|LPh3r=Q|89#&R} z3(QYVO%-uQwYSEXC)n7{-%?gqCj9)B_=~DyTnWyqR&7##_X3U`7ZetIGd`aDDmbwN zdw}OZ7Har|#RJ_Yo*?e{d2sC|dmsV+6A~8Y+V|oB74vi&OSpl7f%DS#N$06)Sq4UC zYmWrGt9KeLmVvJyxhpbKjx_6fq}b-PCRE<{MU4)Cqs>|IXv0 zq2i|+aE#&Rv| z*gOu#qBU{K{~J5ARGZ_eot>S<#l^xQ2SYZoW)&0{H7#vgZf>>5sr}2X(RUcQG274_ z*gbMJ`~UH8)3wLp-;}iucBuQy7&moB7M8}(^hxdU^R~xL6d*AuD^>V|!Xw+-2oD z%{p>#^C$2pPIVM*CFmFI6xsxSuzdPdp~#suIXhVSGo+b>!A%hw%6ggKFpW%x_dzoO3*g5$TsNU2L4Vm|ZwCOesG2DFRkmW@y{x-9on)&a5iV zFKwsh7!cHA_v89X7DT&ff6wz+p_ftd6$y&1=LPg562CPcg#Cv`(Al(HLB0!I-OMU? z+5L_D57pHv2k)K#H`i}9r-~NdzwGY3TIu24#k*8WJbE&xeH;P|C2mUaM2>ua}P^tAFWs<%yD# zQj4}Bk5)5yfFd<5wTiS!2d?|0lczAoa?%SqmyiGGrco-gY`0UeQv5X3>(XyE zOlB@FE^CjxNI3x!-6^XgOrs}}pWlvlyU57GfP4htP2` z^QJ3I+FirYBfk=j{KmueTM!<_jv5#YUSh-Ut|kAkZ>rA7%Fy;U%G4Itm@_l?!UVH~ z`@4J7>(<__b;SK8ql02#e?4i~Jg+`ItT3$k2bJpE(#_4r`r|t!Mjh3Vy1obA`_3*` zl2fz2W~~XQsoV)w%TJST!BWqxGTr}qGeq~J*xewyf#f4pRjY}l*|wQZrWqH^dE|Pn z_>rv@J>34Fd2Fl`HixK~yKrSvf3GA`dOX%W9Zq#O0Csvy+hT}lh+28VAX|G!$K8V( z!{mngp~nmvj>~xq!vq(?R(O9Ues+Cv7P#si=s}B@YR;y)6>lad>f+9E83@R6@$_6KKnCbIo1I_OvlLc09xq zzgf@XjD)+I`F~*dGQ6*@W>*%d5ToZezPk8P7`{ULo{N>6Oc9@~?9l? zNN+XA$@3T~Lc8G`*bV8`K02Z1*4BRuV7<8lX;>Tl$(Y?qOtnfolJ6eVasE#hwY6tO zE(=&3?Ek+9)qma^Ol0q7zy8Oarg+%mE-jUR^X1yNMp3EAD+CB_b)EEh)MqCCaqsP` zQ!MtFgX+QZ109VLA<8H;kw_*$f=w&&(IE(~FvTArpJJAJLvrxqic8Cj zAC9~X@!c@i)cDa2PYIM@cJ)1bbFsru*~QlhKIO}83Qmr-XN2lyyf;p^vhOco^wqX^ zEJ_LGtCFU zEENq)eOFQCMdRgGsd*9QlWAxxpFVdzCWbbG%wQ6q8gaV7=Wtk5wV90?zeoL(oT@AR zYn6UJn-r5ab+>~^aV0E);YOyN{ZjuYL0-c!=8-5BQ(v*_y;l0_h;{SHI$r0+_XW}G z8ly|zsc7Y;!EP4@t5XLN<#EhHi`d#yQ(i%I__*ZuUYX}@rEA_Z)%Ewg+SsPM{WyL$ zijMI@>|6;#kFX!!o~n|xn9jArRd{=A*MFZR50~#PEzZVy?r%z}8g5rhW)I>#Thq+O zNzmNN6o~gxw&!ngDM#Y#XidWHA_W@0h}*}ABz7GOW^P=vr@W!;MxhGL)+VO6pN>}k zGe5)WanI3VTP?(Ve##(Bd%lYlV==4PMck_GU!f;4o<})v1PU}`-PO>Mt;=GYv4OOf z^Jar^g5mKdD0G%I5Z z!KZzCB%jR>-(B~b$!YOD2MKMlmk^SM*X3`{Wiv=DF`i6%WG*JKvD{vUA?RI`@^xSH zkm~c}s@CPZB^b}iwq?4Vi;Xeze8%KJHg_-@iDX`EPug3OEl1xU{|{52sbk_Q=YVli zT110SrQadcpxN4@ROiqhPRs{|bIolx%LR2$b3eCJuceIKXT1oVeH*up{rLK- zWJbwD=gP;s^448gU!GPJv}=pcsq(g8A}8<`x;a(wj&9GsEeNE)+)%Hp!e8}ez}1++ zOZ0frh9jeyhML3PqG>aIzJ7cJp5>*YqF^>*D8~@XPu$%c$U$9846xY7oy5PAl3c%! zan|^4YGTN>KCp20>C5zEksLP54PP!cJI!NBx2~N}l2Yt7rX&v$@^!3+-v_jRABgmL z=~*~#x@WMoa~%)%PsGLc;I|w$%~=sqs%@sTT{)ExcW+^(yh6Txpd+(#x>(j?_w<%9 zz8<~&w&Wc9n6eC%oq%ohL2;uQ4;z=F(7#SRF7$?Dw&5b1UXmlFt-=WVVYWio`Ez`W zoyswjA8-fuZaHl4jpR_g9j^`Tr94<-Qw?pI(~o4z7vBs&jxF=^CaPQMi3FfGsB{h}~dGvj^%frrh4wVsZ9Sd!#d}wivJUC6K z78@Wp8zZy%Bh`nrHrDfvt|#W1bLc6SwlKrQ27C0V%qH2FGXB7GV&(gC#omySbL%&i zcDpN;s4|n1zl(NAPL9iHlkHs>^Tm9VL_5x;m6YAZB0B7{T=^rXk*V63g|=>?%#C3o z_A=M}SVql4r(#{pZCc}7l-9b6P{BU*Nm!XOZ+pXQ%i^K3>?-XC%X10NHFslg{+_D? z$-ys?jq;M>AqBUFFR8_+>^A%5*QdBv#e8zLY6^^KPDzGp-N%OJnb{bSz)%BuBN(LOVe*H28AW$9v=c)gDf}!k;6St8(t$0}6Bwi%jQ7$95sj zW#f$sO#KHt)%x(H{k#5{&RL1>n@xe0f&(SR(Y)65L3`xho&GjV8w#5%soEiKJ>8pp^nm<+Nbi}#eg+-; zWKdBN6?~001Ro`{alAE!;oh{i?v#vCwIpx-GGBNtsc^QLHk~Mja`e=>+$Ivm-zMPd zYB}vuB)GiW*0OBfoU^%lGI*gmZID&fvDVpvyhz>u@Ls5_Px&748e64f=ME7+h#g7r z5RY&5^YnA@^7O{cHLd$vKB|lBDy}LfC~W^2?iTy9w6;;cY`STFFBG3N5IRp8T!LI+ zq>JvwlD<@7y-Y^P!EG%`C{8JSv1RW<>FUX2MH_LW4SaDSQWleS)a7n>LiZ{Q$!!WD ze76fE(8G`Pd)vg@@OE3wsB6TT?$LO?A1!sS`=$rv8f%IMoi^BH9xaGdt8RHM7$j7Z zhb{C`tD_AKciol;!boi2BS&7OkVniZD>wn+ko|?aTPH?v;}Yp#tyUn+jt<6;Sk zUET1a(xu*Wa(4X9#+22t*}a8`^3^Ph@*QJi{a$B>^KA?vrzcT1!B*XLduAxt*x>5H zk26nNltam6Z?k`(iyuS@cJir~*BQ$=8oK=3ITzvSM6k+le#$hOgp#fh)HNLmQ}%3Y zEhI?P-Pc$}iQ9zOw@*%fc5Es@8`I-M|da$7CKUy5A= zZ!LHu@iXlvQI3vANpXF`G6!MG+NJJz>Q$fq^?I4cFok+wgHJAx@)GstDlQX;K1$dL zSS!m15WGh*4@L_4Om~|3$`_YNMPx#Cf3e*};_dp?>!yj)vA3l*mnL@SLJLT13tNqJ znjs}9$_@>x3P5kkrr4{RAaV_UZOHQx( z7#{LpeWlZ~kD(K(n%1fo?^Q&mEHo>(s-Y}4r^zq8ghiOW$~J3<^&>d~H}QsH#1D+< z!cjBI(t`>T*~kZ9+h1&9>K20Ct&)@Ke^{$fc;^83daUajc_fviKXO=x&F}k2EQG>* z^7pQs2Wo;&VzlIKO4Ekl`a=iO0A2ZV!EnE%?@<5l=3L)LKEFz<%UYL>DjrQF34T^oRn%YIHE+xL=ikkD_21Us0OmY|p@sD5(&oXh08nhf@)qXDH|Zbn+9~XLfI(cDq$zBxz8T7~-S5h@Tk<$!#8(+4nJ}ddy*^tv=VX@Ip-} zw7_gG)QDUvZbrL9oZd{Y3)H;ap7o55cv1p0Qolp+7Ia$}_6lVhg^U+^O0rTkE;dia zSEBr72L8)SRgVupDjm_KtPEa~j^AR3T>D?~gG1`E5XGA}v|68J(%-HRYFiz1yJAQw z%(r>eQlcy|+VL)&Y~R0AR1nBJT7FF`TgO~QH$sZn4dw>Rh<2a#sJ=Yx}u~UPb-r+64z%nsh~SE z*BteF&l#7xJ<-}X?Pt`N;Grn!a)nmX`|`TvX2|5AN85a3U3*?@iEJ}qM$HG;}(P8uq{CVG36SG`RdimH$dkgNCr0;LNR8(iYhPIYvBE1|8;+RH;hfj~3pBk7SO&Z-wE%3TW zy}qcnJhlJS3#yKc17vB%-FI8luR~J8K9AB;?>wJXfZYv^*q31U4V(UPEsK}%Erh?Vu)mczU$_H{GTqtZn)$u6TwHY9~4#n(SuboQ1BGue3;KV8VV#szf9SW2e02GjTmpM$i@CQuCI*zuH;%+VEaqJ_@o&@Um3s z=dDXFu6;BirTpexg_1@eR6+LJxYi^_E z7u25=C9oXK&|1u`zS10ECsZSn|3`9T06$J)uWe-?JG)}F5BENffvJwoN-BMY(yK@Fk^CS+B&Fb_@}BikAFk z6}=+y`V_Uf`wIPbKM@lPKRXySP(5Zy|9K)yic8zup5Sz60I5nh8#!uYV{eB!Vl5~k z82>rSqyf2J|HY8+G5b(d4a21C(f_%Z5~@EM;8`}vYsB+^=a_Vd!sS^${vDO5i(HGM Noa)WIfA2o|zW_dEKK=jz literal 0 HcmV?d00001 diff --git a/tutorials/index.html b/tutorials/index.html new file mode 100644 index 00000000..e4a28342 --- /dev/null +++ b/tutorials/index.html @@ -0,0 +1,894 @@ + + + + + + + + + +Tutorials – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +

+
+ + +
+ +
+ + +
+ + + +
+ +
+
+

Tutorials

+
+ + + +
+ + + + +
+ + + +
+ + +

Click through to any of these tutorials to get started with FastHTML’s features.

+ + + + +
+
  • + + + + + + + + + + + + + + + + + + + + + + + + +
    +Title + +Description +
    +FastHTML By Example + +An introduction to FastHTML from the ground up, with four complete examples +
    +Web Devs Quickstart + +A fast introduction to FastHTML for experienced web developers. +
    +JS App Walkthrough + +How to build a website with custom JavaScript in FastHTML step-by-step +
    +Using Jupyter to write FastHTML + +Writing FastHTML applications in Jupyter notebooks requires a slightly different process than normal Python applications. +
    +
    +No matching items +
    +
    + + +
    + + + + + + + \ No newline at end of file diff --git a/tutorials/index.md b/tutorials/index.md new file mode 100644 index 00000000..b84e32e2 --- /dev/null +++ b/tutorials/index.md @@ -0,0 +1,5 @@ +# Tutorials + + +Click through to any of these tutorials to get started with FastHTML’s +features. diff --git a/tutorials/jupyter_and_fasthtml.html b/tutorials/jupyter_and_fasthtml.html new file mode 100644 index 00000000..2489f70d --- /dev/null +++ b/tutorials/jupyter_and_fasthtml.html @@ -0,0 +1,902 @@ + + + + + + + + + + +Using Jupyter to write FastHTML – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    +
    + + +
    + +
    + + +
    + + + +
    + +
    +
    +

    Using Jupyter to write FastHTML

    +
    + +
    +
    + Writing FastHTML applications in Jupyter notebooks requires a slightly different process than normal Python applications. +
    +
    + + +
    + + + + +
    + + + +
    + + + +

    Writing FastHTML applications in Jupyter notebooks requires a slightly different process than normal Python applications.

    +
    +
    +
    + +
    +
    +Download this notebook and try it yourself +
    +
    +
    +

    The source code for this page is a Jupyter notebook. That makes it easy to directly experiment with it. However, as this is working code that means we have to comment out a few things in order for the documentation to build.

    +
    +
    +

    The first step is to import necessary libraries. As using FastHTML inside a Jupyter notebook is a special case, it remains a special import.

    +
    +
    from fasthtml.common import *
    +from fasthtml.jupyter import JupyUvi, HTMX
    +
    +

    Let’s create an app with fast_app.

    +
    +
    app, rt = fast_app(pico=True)
    +
    +

    Define a route to test the application.

    +
    +
    @rt
    +def index():
    +    return Titled('Hello, Jupyter',
    +           P('Welcome to the FastHTML + Jupyter example'),
    +           Button('Click', hx_get='/click', hx_target='#dest'),
    +           Div(id='dest')
    +    )
    +
    +

    Create a server object using JupyUvi, which also starts Uvicorn. The server runs in a separate thread from Jupyter, so it can use normal HTTP client functions in a notebook.

    +
    +
    server = JupyUvi(app)
    +
    + + +
    +
    +

    The HTMX callable displays the server’s HTMX application in an iframe which can be displayed by Jupyter notebook. Pass in the same port variable used in the JupyUvi callable above or leave it blank to use the default (8000).

    +
    +
    # This doesn't display in the docs - uncomment and run it to see it in action
    +# HTMX()
    +
    +

    We didn’t define the /click route, but that’s fine - we can define (or change) it any time, and it’s dynamically inserted into the running app. No need to restart or reload anything!

    +
    +
    @rt
    +def click(): return P('You clicked me!')
    +
    +
    +

    Full screen view

    +

    You can view your app outside of Jupyter by going to localhost:PORT, where PORT is usually the default 8000, so in most cases just click this link.

    +
    +
    +

    Graceful shutdowns

    +

    Use the server.stop() function displayed below. If you restart Jupyter without calling this line the thread may not be released and the HTMX callable above may throw errors. If that happens, a quick temporary fix is to specify a different port number in JupyUvi and HTMX with the port parameter.

    +

    Cleaner solutions to the dangling thread are to kill the dangling thread (dependant on each operating system) or restart the computer.

    +
    +
    server.stop()
    +
    + + +
    + +
    + +
    + + + + + \ No newline at end of file diff --git a/tutorials/jupyter_and_fasthtml.html.md b/tutorials/jupyter_and_fasthtml.html.md new file mode 100644 index 00000000..78ca5e49 --- /dev/null +++ b/tutorials/jupyter_and_fasthtml.html.md @@ -0,0 +1,107 @@ +# Using Jupyter to write FastHTML + + + + +Writing FastHTML applications in Jupyter notebooks requires a slightly +different process than normal Python applications. + +
    + +> **Download this notebook and try it yourself** +> +> The source code for this page is a [Jupyter +> notebook](https://github.com/AnswerDotAI/fasthtml/blob/main/nbs/tutorials/jupyter_and_fasthtml.ipynb). +> That makes it easy to directly experiment with it. However, as this is +> working code that means we have to comment out a few things in order +> for the documentation to build. + +
    + +The first step is to import necessary libraries. As using FastHTML +inside a Jupyter notebook is a special case, it remains a special +import. + +``` python +from fasthtml.common import * +from fasthtml.jupyter import JupyUvi, HTMX +``` + +Let’s create an app with `fast_app`. + +``` python +app, rt = fast_app(pico=True) +``` + +Define a route to test the application. + +``` python +@rt +def index(): + return Titled('Hello, Jupyter', + P('Welcome to the FastHTML + Jupyter example'), + Button('Click', hx_get='/click', hx_target='#dest'), + Div(id='dest') + ) +``` + +Create a `server` object using +[`JupyUvi`](https://AnswerDotAI.github.io/fasthtml/api/jupyter.html#jupyuvi), +which also starts Uvicorn. The `server` runs in a separate thread from +Jupyter, so it can use normal HTTP client functions in a notebook. + +``` python +server = JupyUvi(app) +``` + + + +The +[`HTMX`](https://AnswerDotAI.github.io/fasthtml/api/jupyter.html#htmx) +callable displays the server’s HTMX application in an iframe which can +be displayed by Jupyter notebook. Pass in the same `port` variable used +in the +[`JupyUvi`](https://AnswerDotAI.github.io/fasthtml/api/jupyter.html#jupyuvi) +callable above or leave it blank to use the default (8000). + +``` python +# This doesn't display in the docs - uncomment and run it to see it in action +# HTMX() +``` + +We didn’t define the `/click` route, but that’s fine - we can define (or +change) it any time, and it’s dynamically inserted into the running app. +No need to restart or reload anything! + +``` python +@rt +def click(): return P('You clicked me!') +``` + +## Full screen view + +You can view your app outside of Jupyter by going to `localhost:PORT`, +where `PORT` is usually the default 8000, so in most cases just click +[this link](localhost:8000/). + +## Graceful shutdowns + +Use the `server.stop()` function displayed below. If you restart Jupyter +without calling this line the thread may not be released and the +[`HTMX`](https://AnswerDotAI.github.io/fasthtml/api/jupyter.html#htmx) +callable above may throw errors. If that happens, a quick temporary fix +is to specify a different port number in JupyUvi and HTMX with the +`port` parameter. + +Cleaner solutions to the dangling thread are to kill the dangling thread +(dependant on each operating system) or restart the computer. + +``` python +server.stop() +``` diff --git a/tutorials/quickstart-web-dev/quickstart-fasthtml.png b/tutorials/quickstart-web-dev/quickstart-fasthtml.png new file mode 100644 index 0000000000000000000000000000000000000000..4f4e82b55bf7fddc3ae7d34c30aa20540b3f9853 GIT binary patch literal 13843 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 + + + + + + + + + +Web Devs Quickstart – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    +
    + + +
    + +
    + + +
    + + + +
    + +
    +
    +

    Web Devs Quickstart

    +
    + +
    +
    + A fast introduction to FastHTML for experienced web developers. +
    +
    + + +
    + + + + +
    + + + +
    + + + +
    +

    Installation

    +
    pip install python-fasthtml
    +
    +
    +

    A Minimal Application

    +

    A minimal FastHTML application looks something like this:

    +
    +
    +
    main.py
    +
    +
    1from fasthtml.common import *
    +
    +2app, rt = fast_app()
    +
    +3@rt("/")
    +4def get():
    +5    return Titled("FastHTML", P("Let's do this!"))
    +
    +6serve()
    +
    +
    +
    1
    +
    +We import what we need for rapid development! A carefully-curated set of FastHTML functions and other Python objects is brought into our global namespace for convenience. +
    +
    2
    +
    +We instantiate a FastHTML app with the fast_app() utility function. This provides a number of really useful defaults that we’ll take advantage of later in the tutorial. +
    +
    3
    +
    +We use the rt() decorator to tell FastHTML what to return when a user visits / in their browser. +
    +
    4
    +
    +We connect this route to HTTP GET requests by defining a view function called get(). +
    +
    5
    +
    +A tree of Python function calls that return all the HTML required to write a properly formed web page. You’ll soon see the power of this approach. +
    +
    6
    +
    +The serve() utility configures and runs FastHTML using a library called uvicorn. +
    +
    +

    Run the code:

    +
    python main.py
    +

    The terminal will look like this:

    +
    INFO:     Uvicorn running on http://0.0.0.0:5001 (Press CTRL+C to quit)
    +INFO:     Started reloader process [58058] using WatchFiles
    +INFO:     Started server process [58060]
    +INFO:     Waiting for application startup.
    +INFO:     Application startup complete.
    +

    Confirm FastHTML is running by opening your web browser to 127.0.0.1:5001. You should see something like the image below:

    +

    +
    +
    +
    + +
    +
    +Note +
    +
    +
    +

    While some linters and developers will complain about the wildcard import, it is by design here and perfectly safe. FastHTML is very deliberate about the objects it exports in fasthtml.common. If it bothers you, you can import the objects you need individually, though it will make the code more verbose and less readable.

    +

    If you want to learn more about how FastHTML handles imports, we cover that here.

    +
    +
    +
    +
    +

    A Minimal Charting Application

    +

    The Script function allows you to include JavaScript. You can use Python to generate parts of your JS or JSON like this:

    +
    import json
    +from fasthtml.common import * 
    +
    +app, rt = fast_app(hdrs=(Script(src="https://cdn.plot.ly/plotly-2.32.0.min.js"),))
    +
    +data = json.dumps({
    +    "data": [{"x": [1, 2, 3, 4],"type": "scatter"},
    +            {"x": [1, 2, 3, 4],"y": [16, 5, 11, 9],"type": "scatter"}],
    +    "title": "Plotly chart in FastHTML ",
    +    "description": "This is a demo dashboard",
    +    "type": "scatter"
    +})
    +
    +
    +@rt("/")
    +def get():
    +  return Titled("Chart Demo", Div(id="myDiv"),
    +    Script(f"var data = {data}; Plotly.newPlot('myDiv', data);"))
    +
    +serve()
    +
    +
    +

    Debug Mode

    +

    When we can’t figure out a bug in FastHTML, we can run it in DEBUG mode. When an error is thrown, the error screen is displayed in the browser. This error setting should never be used in a deployed app.

    +
    from fasthtml.common import *
    +
    +1app, rt = fast_app(debug=True)
    +
    +@rt("/")
    +def get():
    +2    1/0
    +    return Titled("FastHTML Error!", P("Let's error!"))
    +
    +serve()
    +
    +
    1
    +
    +debug=True sets debug mode on. +
    +
    2
    +
    +Python throws an error when it tries to divide an integer by zero. +
    +
    +
    +
    +

    Routing

    +

    FastHTML builds upon FastAPI’s friendly decorator pattern for specifying URLs, with extra features:

    +
    +
    +
    main.py
    +
    +
    from fasthtml.common import * 
    +
    +app, rt = fast_app()
    +
    +1@rt("/")
    +def get():
    +  return Titled("FastHTML", P("Let's do this!"))
    +
    +2@rt("/hello")
    +def get():
    +  return Titled("Hello, world!")
    +
    +serve()
    +
    +
    +
    1
    +
    +The “/” URL on line 5 is the home of a project. This would be accessed at 127.0.0.1:5001. +
    +
    2
    +
    +“/hello” URL on line 9 will be found by the project if the user visits 127.0.0.1:5001/hello. +
    +
    +
    +
    +
    + +
    +
    +Tip +
    +
    +
    +

    It looks like get() is being defined twice, but that’s not the case. Each function decorated with rt is totally separate, and is injected into the router. We’re not calling them in the module’s namespace (locals()). Rather, we’re loading them into the routing mechanism using the rt decorator.

    +
    +
    +

    You can do more! Read on to learn what we can do to make parts of the URL dynamic.

    +
    +
    +

    Variables in URLs

    +

    You can add variable sections to a URL by marking them with {variable_name}. Your function then receives the {variable_name} as a keyword argument, but only if it is the correct type. Here’s an example:

    +
    +
    +
    main.py
    +
    +
    from fasthtml.common import * 
    +
    +app, rt = fast_app()
    +
    +1@rt("/{name}/{age}")
    +2def get(name: str, age: int):
    +3  return Titled(f"Hello {name.title()}, age {age}")
    +
    +serve()
    +
    +
    +
    1
    +
    +We specify two variable names, name and age. +
    +
    2
    +
    +We define two function arguments named identically to the variables. You will note that we specify the Python types to be passed. +
    +
    3
    +
    +We use these functions in our project. +
    +
    +

    Try it out by going to this address: 127.0.0.1:5001/uma/5. You should get a page that says,

    +
    +

    “Hello Uma, age 5”.

    +
    +
    +

    What happens if we enter incorrect data?

    +

    The 127.0.0.1:5001/uma/5 URL works because 5 is an integer. If we enter something that is not, such as 127.0.0.1:5001/uma/five, then FastHTML will return an error instead of a web page.

    +
    +
    +
    + +
    +
    +FastHTML URL routing supports more complex types +
    +
    +
    +

    The two examples we provide here use Python’s built-in str and int types, but you can use your own types, including more complex ones such as those defined by libraries like attrs, pydantic, and even sqlmodel.

    +
    +
    +
    +
    +
    +

    HTTP Methods

    +

    FastHTML matches function names to HTTP methods. So far the URL routes we’ve defined have been for HTTP GET methods, the most common method for web pages.

    +

    Form submissions often are sent as HTTP POST. When dealing with more dynamic web page designs, also known as Single Page Apps (SPA for short), the need can arise for other methods such as HTTP PUT and HTTP DELETE. The way FastHTML handles this is by changing the function name.

    +
    +
    +
    main.py
    +
    +
    from fasthtml.common import * 
    +
    +app, rt = fast_app()
    +
    +@rt("/")  
    +1def get():
    +  return Titled("HTTP GET", P("Handle GET"))
    +
    +@rt("/")  
    +2def post():
    +  return Titled("HTTP POST", P("Handle POST"))
    +
    +serve()
    +
    +
    +
    1
    +
    +On line 6 because the get() function name is used, this will handle HTTP GETs going to the / URI. +
    +
    2
    +
    +On line 10 because the post() function name is used, this will handle HTTP POSTs going to the / URI. +
    +
    +
    +
    +

    CSS Files and Inline Styles

    +

    Here we modify default headers to demonstrate how to use the Sakura CSS microframework instead of FastHTML’s default of Pico CSS.

    +
    +
    +
    main.py
    +
    +
    from fasthtml.common import * 
    +
    +app, rt = fast_app(
    +1    pico=False,
    +    hdrs=(
    +        Link(rel='stylesheet', href='assets/normalize.min.css', type='text/css'),
    +2        Link(rel='stylesheet', href='assets/sakura.css', type='text/css'),
    +3        Style("p {color: red;}")
    +))
    +
    +@app.get("/")
    +def home():
    +    return Titled("FastHTML",
    +        P("Let's do this!"),
    +    )
    +
    +serve()
    +
    +
    +
    1
    +
    +By setting pico to False, FastHTML will not include pico.min.css. +
    +
    2
    +
    +This will generate an HTML <link> tag for sourcing the css for Sakura. +
    +
    3
    +
    +If you want an inline styles, the Style() function will put the result into the HTML. +
    +
    +
    +
    +

    Other Static Media File Locations

    +

    As you saw, Script and Link are specific to the most common static media use cases in web apps: including JavaScript, CSS, and images. But it also works with videos and other static media files. The default behavior is to look for these files in the root directory - typically we don’t do anything special to include them. We can change the default directory that is looked in for files by adding the static_path parameter to the fast_app function.

    +
    app, rt = fast_app(static_path='public')
    +

    FastHTML also allows us to define a route that uses FileResponse to serve the file at a specified path. This is useful for serving images, videos, and other media files from a different directory without having to change the paths of many files. So if we move the directory containing the media files, we only need to change the path in one place. In the example below, we call images from a directory called public.

    +
    @rt("/{fname:path}.{ext:static}")
    +async def get(fname:str, ext:str): 
    +    return FileResponse(f'public/{fname}.{ext}')
    +
    +
    +

    Rendering Markdown

    +
    from fasthtml.common import *
    +
    +hdrs = (MarkdownJS(), HighlightJS(langs=['python', 'javascript', 'html', 'css']), )
    +
    +app, rt = fast_app(hdrs=hdrs)
    +
    +content = """
    +Here are some _markdown_ elements.
    +
    +- This is a list item
    +- This is another list item
    +- And this is a third list item
    +
    +**Fenced code blocks work here.**
    +"""
    +
    +@rt('/')
    +def get(req):
    +    return Titled("Markdown rendering example", Div(content,cls="marked"))
    +
    +serve()
    +
    +
    +

    Code highlighting

    +

    Here’s how to highlight code without any markdown configuration.

    +
    from fasthtml.common import *
    +
    +# Add the HighlightJS built-in header
    +hdrs = (HighlightJS(langs=['python', 'javascript', 'html', 'css']),)
    +
    +app, rt = fast_app(hdrs=hdrs)
    +
    +code_example = """
    +import datetime
    +import time
    +
    +for i in range(10):
    +    print(f"{datetime.datetime.now()}")
    +    time.sleep(1)
    +"""
    +
    +@rt('/')
    +def get(req):
    +    return Titled("Markdown rendering example",
    +        Div(
    +            # The code example needs to be surrounded by
    +            # Pre & Code elements
    +            Pre(Code(code_example))
    +    ))
    +
    +serve()
    +
    +
    +

    Defining new ft components

    +

    We can build our own ft components and combine them with other components. The simplest method is defining them as a function.

    +
    +
    from fasthtml.common import *
    +
    +
    +
    def hero(title, statement):
    +    return Div(H1(title),P(statement), cls="hero")
    +
    +# usage example
    +Main(
    +    hero("Hello World", "This is a hero statement")
    +)
    +
    +
    <main>  <div class="hero">
    +    <h1>Hello World</h1>
    +    <p>This is a hero statement</p>
    +  </div>
    +</main>
    +
    +
    +
    +

    Pass through components

    +

    For when we need to define a new component that allows zero-to-many components to be nested within them, we lean on Python’s *args and **kwargs mechanism. Useful for creating page layout controls.

    +
    +
    def layout(*args, **kwargs):
    +    """Dashboard layout for all our dashboard views"""
    +    return Main(
    +        H1("Dashboard"),
    +        Div(*args, **kwargs),
    +        cls="dashboard",
    +    )
    +
    +# usage example
    +layout(
    +    Ul(*[Li(o) for o in range(3)]),
    +    P("Some content", cls="description"),
    +)
    +
    +
    <main class="dashboard">  <h1>Dashboard</h1>
    +  <div>
    +    <ul>
    +      <li>0</li>
    +      <li>1</li>
    +      <li>2</li>
    +    </ul>
    +    <p class="description">Some content</p>
    +  </div>
    +</main>
    +
    +
    +
    +
    +

    Dataclasses as ft components

    +

    While functions are easy to read, for more complex components some might find it easier to use a dataclass.

    +
    +
    from dataclasses import dataclass
    +
    +@dataclass
    +class Hero:
    +    title: str
    +    statement: str
    +    
    +    def __ft__(self):
    +        """ The __ft__ method renders the dataclass at runtime."""
    +        return Div(H1(self.title),P(self.statement), cls="hero")
    +    
    +# usage example
    +Main(
    +    Hero("Hello World", "This is a hero statement")
    +)
    +
    +
    <main>  <div class="hero">
    +    <h1>Hello World</h1>
    +    <p>This is a hero statement</p>
    +  </div>
    +</main>
    +
    +
    +
    +
    +
    +

    Testing views in notebooks

    +

    Because of the ASGI event loop it is currently impossible to run FastHTML inside a notebook. However, we can still test the output of our views. To do this, we leverage Starlette, an ASGI toolkit that FastHTML uses.

    +
    +
    # First we instantiate our app, in this case we remove the
    +# default headers to reduce the size of the output.
    +app, rt = fast_app(default_hdrs=False)
    +
    +# Setting up the Starlette test client
    +from starlette.testclient import TestClient
    +client = TestClient(app)
    +
    +# Usage example
    +@rt("/")
    +def get():
    +    return Titled("FastHTML is awesome", 
    +        P("The fastest way to create web apps in Python"))
    +
    +print(client.get("/").text)
    +
    +
     <!doctype html>
    + <html>
    +   <head>
    +<title>FastHTML is awesome</title>   </head>
    +   <body>
    +<main class="container">       <h1>FastHTML is awesome</h1>
    +       <p>The fastest way to create web apps in Python</p>
    +</main>   </body>
    + </html>
    +
    +
    +
    +
    +
    +

    Forms

    +

    To validate data coming from users, first define a dataclass representing the data you want to check. Here’s an example representing a signup form.

    +
    +
    from dataclasses import dataclass
    +
    +@dataclass
    +class Profile: email:str; phone:str; age:int
    +
    +

    Create an FT component representing an empty version of that form. Don’t pass in any value to fill the form, that gets handled later.

    +
    +
    profile_form = Form(method="post", action="/profile")(
    +        Fieldset(
    +            Label('Email', Input(name="email")),
    +            Label("Phone", Input(name="phone")),
    +            Label("Age", Input(name="age")),
    +        ),
    +        Button("Save", type="submit"),
    +    )
    +profile_form
    +
    +
    <form enctype="multipart/form-data" method="post" action="/profile"><fieldset><label>Email      <input name="email">
    +</label><label>Phone      <input name="phone">
    +</label><label>Age      <input name="age">
    +</label></fieldset><button type="submit">Save</button></form>
    +
    +
    +

    Once the dataclass and form function are completed, we can add data to the form. To do that, instantiate the profile dataclass:

    +
    +
    profile = Profile(email='john@example.com', phone='123456789', age=5)
    +profile
    +
    +
    Profile(email='john@example.com', phone='123456789', age=5)
    +
    +
    +

    Then add that data to the profile_form using FastHTML’s fill_form class:

    +
    +
    fill_form(profile_form, profile)
    +
    +
    <form enctype="multipart/form-data" method="post" action="/profile"><fieldset><label>Email      <input name="email" value="john@example.com">
    +</label><label>Phone      <input name="phone" value="123456789">
    +</label><label>Age      <input name="age" value="5">
    +</label></fieldset><button type="submit">Save</button></form>
    +
    +
    +
    +

    Forms with views

    +

    The usefulness of FastHTML forms becomes more apparent when they are combined with FastHTML views. We’ll show how this works by using the test client from above. First, let’s create a SQlite database:

    +
    +
    db = database("profiles.db")
    +profiles = db.create(Profile, pk="email")
    +
    +

    Now we insert a record into the database:

    +
    +
    profiles.insert(profile)
    +
    +
    Profile(email='john@example.com', phone='123456789', age=5)
    +
    +
    +

    And we can then demonstrate in the code that form is filled and displayed to the user.

    +
    +
    @rt("/profile/{email}")
    +def profile(email:str):
    +1    profile = profiles[email]
    +2    filled_profile_form = fill_form(profile_form, profile)
    +    return Titled(f'Profile for {profile.email}', filled_profile_form)
    +
    +print(client.get(f"/profile/john@example.com").text)
    +
    +
    +
    1
    +
    +Fetch the profile using the profile table’s email primary key +
    +
    2
    +
    +Fill the form for display. +
    +
    +
    +
    +
     <!doctype html>
    + <html>
    +   <head>
    +<title>Profile for john@example.com</title>   </head>
    +   <body>
    +<main class="container">       <h1>Profile for john@example.com</h1>
    +<form enctype="multipart/form-data" method="post" action="/profile"><fieldset><label>Email             <input name="email" value="john@example.com">
    +</label><label>Phone             <input name="phone" value="123456789">
    +</label><label>Age             <input name="age" value="5">
    +</label></fieldset><button type="submit">Save</button></form></main>   </body>
    + </html>
    +
    +
    +
    +

    And now let’s demonstrate making a change to the data.

    +
    +
    @rt("/profile")
    +1def post(profile: Profile):
    +2    profiles.update(profile)
    +3    return RedirectResponse(url=f"/profile/{profile.email}")
    +
    +new_data = dict(email='john@example.com', phone='7654321', age=25)
    +4print(client.post("/profile", data=new_data).text)
    +
    +
    +
    1
    +
    +We use the Profile dataclass definition to set the type for the incoming profile content. This validates the field types for the incoming data +
    +
    2
    +
    +Taking our validated data, we updated the profiles table +
    +
    3
    +
    +We redirect the user back to their profile view +
    +
    4
    +
    +The display is of the profile form view showing the changes in data. +
    +
    +
    +
    +
     <!doctype html>
    + <html>
    +   <head>
    +<title>Profile for john@example.com</title>   </head>
    +   <body>
    +<main class="container">       <h1>Profile for john@example.com</h1>
    +<form enctype="multipart/form-data" method="post" action="/profile"><fieldset><label>Email             <input name="email" value="john@example.com">
    +</label><label>Phone             <input name="phone" value="7654321">
    +</label><label>Age             <input name="age" value="25">
    +</label></fieldset><button type="submit">Save</button></form></main>   </body>
    + </html>
    +
    +
    +
    +
    +
    +
    +

    Strings and conversion order

    +

    The general rules for rendering are: - __ft__ method will be called (for default components like P, H2, etc. or if you define your own components) - If you pass a string, it will be escaped - On other python objects, str() will be called

    +

    As a consequence, if you want to include plain HTML tags directly into e.g. a Div() they will get escaped by default (as a security measure to avoid code injections). This can be avoided by using NotStr(), a convenient way to reuse python code that returns already HTML. If you use pandas, you can use pandas.DataFrame.to_html() to get a nice table. To include the output a FastHTML, wrap it in NotStr(), like Div(NotStr(df.to_html())).

    +

    Above we saw how a dataclass behaves with the __ft__ method defined. On a plain dataclass, str() will be called (but not escaped).

    +
    +
    from dataclasses import dataclass
    +
    +@dataclass
    +class Hero:
    +    title: str
    +    statement: str
    +        
    +# rendering the dataclass with the default method
    +Main(
    +    Hero("<h1>Hello World</h1>", "This is a hero statement")
    +)
    +
    +
    <main>Hero(title='<h1>Hello World</h1>', statement='This is a hero statement')</main>
    +
    +
    +
    +
    # This will display the HTML as text on your page
    +Div("Let's include some HTML here: <div>Some HTML</div>")
    +
    +
    <div>Let&#x27;s include some HTML here: &lt;div&gt;Some HTML&lt;/div&gt;</div>
    +
    +
    +
    +
    # Keep the string untouched, will be rendered on the page
    +Div(NotStr("<div><h1>Some HTML</h1></div>"))
    +
    +
    <div><div><h1>Some HTML</h1></div></div>
    +
    +
    +
    +
    +

    Custom exception handlers

    +

    FastHTML allows customization of exception handlers, but does so gracefully. What this means is by default it includes all the <html> tags needed to display attractive content. Try it out!

    +
    from fasthtml.common import *
    +
    +def not_found(req, exc): return Titled("404: I don't exist!")
    +
    +exception_handlers = {404: not_found}
    +
    +app, rt = fast_app(exception_handlers=exception_handlers)
    +
    +@rt('/')
    +def get():
    +    return (Titled("Home page", P(A(href="/oops")("Click to generate 404 error"))))
    +
    +serve()
    +

    We can also use lambda to make things more terse:

    +
    from fasthtml.common import *
    +
    +exception_handlers={
    +    404: lambda req, exc: Titled("404: I don't exist!"),
    +    418: lambda req, exc: Titled("418: I'm a teapot!")
    +}
    +
    +app, rt = fast_app(exception_handlers=exception_handlers)
    +
    +@rt('/')
    +def get():
    +    return (Titled("Home page", P(A(href="/oops")("Click to generate 404 error"))))
    +
    +serve()
    +
    +
    +

    Cookies

    +

    We can set cookies using the cookie() function. In our example, we’ll create a timestamp cookie.

    +
    +
    from datetime import datetime
    +from IPython.display import HTML
    +
    +
    +
    @rt("/settimestamp")
    +def get(req):
    +    now = datetime.now()
    +    return P(f'Set to {now}'), cookie('now', datetime.now())
    +
    +HTML(client.get('/settimestamp').text)
    +
    + + + +FastHTML page + +

    Set to 2024-09-26 15:33:48.141869

    + + +
    +
    +

    Now let’s get it back using the same name for our parameter as the cookie name.

    +
    +
    @rt('/gettimestamp')
    +def get(now:parsed_date): return f'Cookie was set at time {now.time()}'
    +
    +client.get('/gettimestamp').text
    +
    +
    'Cookie was set at time 15:33:48.141903'
    +
    +
    +
    +
    +

    Sessions

    +

    For convenience and security, FastHTML has a mechanism for storing small amounts of data in the user’s browser. We can do this by adding a session argument to routes. FastHTML sessions are Python dictionaries, and we can leverage to our benefit. The example below shows how to concisely set and get sessions.

    +
    +
    @rt('/adder/{num}')
    +def get(session, num: int):
    +    session.setdefault('sum', 0)
    +    session['sum'] = session.get('sum') + num
    +    return Response(f'The sum is {session["sum"]}.')
    +
    +
    +
    +

    Toasts (also known as Messages)

    +

    Toasts, sometimes called “Messages” are small notifications usually in colored boxes used to notify users that something has happened. Toasts can be of four types:

    +
      +
    • info
    • +
    • success
    • +
    • warning
    • +
    • error
    • +
    +

    Examples toasts might include:

    +
      +
    • “Payment accepted”
    • +
    • “Data submitted”
    • +
    • “Request approved”
    • +
    +

    Toasts require the use of the setup_toasts() function plus every view needs these two features:

    +
      +
    • The session argument
    • +
    • Must return FT components
    • +
    +
    1setup_toasts(app)
    +
    +@rt('/toasting')
    +2def get(session):
    +    # Normally one toast is enough, this allows us to see
    +    # different toast types in action.
    +    add_toast(session, f"Toast is being cooked", "info")
    +    add_toast(session, f"Toast is ready", "success")
    +    add_toast(session, f"Toast is getting a bit crispy", "warning")
    +    add_toast(session, f"Toast is burning!", "error")
    +3    return Titled("I like toast")
    +
    +
    1
    +
    +setup_toasts is a helper function that adds toast dependencies. Usually this would be declared right after fast_app() +
    +
    2
    +
    +Toasts require sessions +
    +
    3
    +
    +Views with Toasts must return FT or FtResponse components. +
    +
    +

    💡 setup_toasts takes a duration input that allows you to specify how long a toast will be visible before disappearing. For example setup_toasts(duration=5) sets the toasts duration to 5 seconds. By default toasts disappear after 10 seconds.

    +
    +
    +

    Authentication and authorization

    +

    In FastHTML the tasks of authentication and authorization are handled with Beforeware. Beforeware are functions that run before the route handler is called. They are useful for global tasks like ensuring users are authenticated or have permissions to access a view.

    +

    First, we write a function that accepts a request and session arguments:

    +
    +
    # Status code 303 is a redirect that can change POST to GET,
    +# so it's appropriate for a login page.
    +login_redir = RedirectResponse('/login', status_code=303)
    +
    +def user_auth_before(req, sess):
    +    # The `auth` key in the request scope is automatically provided
    +    # to any handler which requests it, and can not be injected
    +    # by the user using query params, cookies, etc, so it should
    +    # be secure to use.    
    +    auth = req.scope['auth'] = sess.get('auth', None)
    +    # If the session key is not there, it redirects to the login page.
    +    if not auth: return login_redir
    +
    +

    Now we pass our user_auth_before function as the first argument into a Beforeware class. We also pass a list of regular expressions to the skip argument, designed to allow users to still get to the home and login pages.

    +
    +
    beforeware = Beforeware(
    +    user_auth_before,
    +    skip=[r'/favicon\.ico', r'/static/.*', r'.*\.css', r'.*\.js', '/login', '/']
    +)
    +
    +app, rt = fast_app(before=beforeware)
    +
    +
    +
    +

    Server-sent events (SSE)

    +

    With server-sent events, it’s possible for a server to send new data to a web page at any time, by pushing messages to the web page. Unlike WebSockets, SSE can only go in one direction: server to client. SSE is also part of the HTTP specification unlike WebSockets which uses its own specification.

    +

    FastHTML introduces several tools for working with SSE which are covered in the example below. While concise, there’s a lot going on in this function so we’ve annotated it quite a bit.

    +
    +
    import random
    +from asyncio import sleep
    +from fasthtml.common import *
    +
    +1hdrs=(Script(src="https://unpkg.com/htmx-ext-sse@2.2.1/sse.js"),)
    +app,rt = fast_app(hdrs=hdrs)
    +
    +@rt
    +def index():
    +    return Titled("SSE Random Number Generator",
    +        P("Generate pairs of random numbers, as the list grows scroll downwards."),
    +2        Div(hx_ext="sse",
    +3            sse_connect="/number-stream",
    +4            hx_swap="beforeend show:bottom",
    +5            sse_swap="message"))
    +
    +6shutdown_event = signal_shutdown()
    +
    +7async def number_generator():
    +8    while not shutdown_event.is_set():
    +        data = Article(random.randint(1, 100))
    +9        yield sse_message(data)
    +        await sleep(1)
    +
    +@rt("/number-stream")
    +10async def get(): return EventStream(number_generator())
    +
    +
    +
    1
    +
    +Import the HTMX SSE extension +
    +
    2
    +
    +Tell HTMX to load the SSE extension +
    +
    3
    +
    +Look at the /number-stream endpoint for SSE content +
    +
    4
    +
    +When new items come in from the SSE endpoint, add them at the end of the current content within the div. If they go beyond the screen, scroll downwards +
    +
    5
    +
    +Specify the name of the event. FastHTML’s default event name is “message”. Only change if you have more than one call to SSE endpoints within a view +
    +
    6
    +
    +Set up the asyncio event loop +
    +
    7
    +
    +Don’t forget to make this an async function! +
    +
    8
    +
    +Iterate through the asyncio event loop +
    +
    9
    +
    +We yield the data. Data ideally should be comprised of FT components as that plugs nicely into HTMX in the browser +
    +
    10
    +
    +The endpoint view needs to be an async function that returns a EventStream +
    +
    +
    +
    +
    +
    +

    Websockets

    +

    With websockets we can have bi-directional communications between a browser and client. Websockets are useful for things like chat and certain types of games. While websockets can be used for single direction messages from the server (i.e. telling users that a process is finished), that task is arguably better suited for SSE.

    +

    FastHTML provides useful tools for adding websockets to your pages.

    +
    +
    from fasthtml.common import *
    +from asyncio import sleep
    +
    +1app, rt = fast_app(exts='ws')
    +
    +2def mk_inp(): return Input(id='msg', autofocus=True)
    +
    +@rt('/')
    +async def get(request):
    +    cts = Div(
    +        Div(id='notifications'),
    +3        Form(mk_inp(), id='form', ws_send=True),
    +4        hx_ext='ws', ws_connect='/ws')
    +    return Titled('Websocket Test', cts)
    +
    +5async def on_connect(send):
    +    print('Connected!')
    +6    await send(Div('Hello, you have connected', id="notifications"))
    +
    +7async def on_disconnect(ws):
    +    print('Disconnected!')
    +
    +8@app.ws('/ws', conn=on_connect, disconn=on_disconnect)
    +9async def ws(msg:str, send):
    +10    await send(Div('Hello ' + msg, id="notifications"))
    +    await sleep(2)
    +11    return Div('Goodbye ' + msg, id="notifications"), mk_inp()
    +
    +
    +
    1
    +
    +To use websockets in FastHTML, you must instantiate the app with exts set to ‘ws’ +
    +
    2
    +
    +As we want to use websockets to reset the form, we define the mk_input function that can be called from multiple locations +
    +
    3
    +
    +We create the form and mark it with the ws_send attribute, which is documented here in the HTMX websocket specification. This tells HTMX to send a message to the nearest websocket based on the trigger for the form element, which for forms is pressing the enter key, an action considered to be a form submission +
    +
    4
    +
    +This is where the HTMX extension is loaded (hx_ext='ws') and the nearest websocket is defined (ws_connect='/ws') +
    +
    5
    +
    +When a websocket first connects we can optionally have it call a function that accepts a send argument. The send argument will push a message to the browser. +
    +
    6
    +
    +Here we use the send function that was passed into the on_connect function to send a Div with an id of notifications that HTMX assigns to the element in the page that already has an id of notifications +
    +
    7
    +
    +When a websocket disconnects we can call a function which takes no arguments. Typically the role of this function is to notify the server to take an action. In this case, we print a simple message to the console +
    +
    8
    +
    +We use the app.ws decorator to mark that /ws is the route for our websocket. We also pass in the two optional conn and disconn parameters to this decorator. As a fun experiment, remove the conn and disconn arguments and see what happens +
    +
    9
    +
    +Define the ws function as async. This is necessary for ASGI to be able to serve websockets. The function accepts two arguments, a msg that is user input from the browser, and a send function for pushing data back to the browser +
    +
    10
    +
    +The send function is used here to send HTML back to the page. As the HTML has an id of notifications, HTMX will overwrite what is already on the page with the same ID +
    +
    11
    +
    +The websocket function can also be used to return a value. In this case, it is a tuple of two HTML elements. HTMX will take the elements and replace them where appropriate. As both have id specified (notifications and msg respectively), they will replace their predecessor on the page. +
    +
    +
    +
    +
    +
    +

    File Uploads

    +

    A common task in web development is uploading files. The examples below are for uploading files to the hosting server, with information about the uploaded file presented to the user.

    +
    +
    +
    + +
    +
    +File uploads in production can be dangerous +
    +
    +
    +

    File uploads can be the target of abuse, accidental or intentional. That means users may attempt to upload files that are too large or present a security risk. This is especially of concern for public facing apps. File upload security is outside the scope of this tutorial, for now we suggest reading the OWASP File Upload Cheat Sheet.

    +
    +
    +
    +

    Single File Uploads

    +
    from fasthtml.common import *
    +from pathlib import Path
    +
    +app, rt = fast_app()
    +
    +upload_dir = Path("filez")
    +upload_dir.mkdir(exist_ok=True)
    +
    +@rt('/')
    +def get():
    +    return Titled("File Upload Demo",
    +        Article(
    +1            Form(hx_post=upload, hx_target="#result-one")(
    +2                Input(type="file", name="file"),
    +                Button("Upload", type="submit", cls='secondary'),
    +            ),
    +            Div(id="result-one")
    +        )
    +    )
    +
    +def FileMetaDataCard(file):
    +    return Article(
    +        Header(H3(file.filename)),
    +        Ul(
    +            Li('Size: ', file.size),            
    +            Li('Content Type: ', file.content_type),
    +            Li('Headers: ', file.headers),
    +        )
    +    )    
    +
    +@rt
    +3async def upload(file: UploadFile):
    +4    card = FileMetaDataCard(file)
    +5    filebuffer = await file.read()
    +6    (upload_dir / file.filename).write_bytes(filebuffer)
    +    return card
    +
    +serve()
    +
    +
    1
    +
    +Every form rendered with the Form FT component defaults to enctype="multipart/form-data" +
    +
    2
    +
    +Don’t forget to set the Input FT Component’s type to file +
    +
    3
    +
    +The upload view should receive a Starlette UploadFile type. You can add other form variables +
    +
    4
    +
    +We can access the metadata of the card (filename, size, content_type, headers), a quick and safe process. We set that to the card variable +
    +
    5
    +
    +In order to access the contents contained within a file we use the await method to read() it. As files may be quite large or contain bad data, this is a seperate step from accessing metadata +
    +
    6
    +
    +This step shows how to use Python’s built-in pathlib.Path library to write the file to disk. +
    +
    +
    +
    +

    Multiple File Uploads

    +
    from fasthtml.common import *
    +from pathlib import Path
    +
    +app, rt = fast_app()
    +
    +upload_dir = Path("filez")
    +upload_dir.mkdir(exist_ok=True)
    +
    +@rt('/')
    +def get():
    +    return Titled("Multiple File Upload Demo",
    +        Article(
    +1            Form(hx_post=upload_many, hx_target="#result-many")(
    +2                Input(type="file", name="files", multiple=True),
    +                Button("Upload", type="submit", cls='secondary'),
    +            ),
    +            Div(id="result-many")
    +        )
    +    )
    +
    +def FileMetaDataCard(file):
    +    return Article(
    +        Header(H3(file.filename)),
    +        Ul(
    +            Li('Size: ', file.size),            
    +            Li('Content Type: ', file.content_type),
    +            Li('Headers: ', file.headers),
    +        )
    +    )    
    +
    +@rt
    +3async def upload_many(files: list[UploadFile]):
    +    cards = []
    +4    for file in files:
    +5        cards.append(FileMetaDataCard(file))
    +6        filebuffer = await file.read()
    +7        (upload_dir / file.filename).write_bytes(filebuffer)
    +    return cards
    +
    +serve()
    +
    +
    1
    +
    +Every form rendered with the Form FT component defaults to enctype="multipart/form-data" +
    +
    2
    +
    +Don’t forget to set the Input FT Component’s type to file and assign the multiple attribute to True +
    +
    3
    +
    +The upload view should receive a list containing the Starlette UploadFile type. You can add other form variables +
    +
    4
    +
    +Iterate through the files +
    +
    5
    +
    +We can access the metadata of the card (filename, size, content_type, headers), a quick and safe process. We add that to the cards variable +
    +
    6
    +
    +In order to access the contents contained within a file we use the await method to read() it. As files may be quite large or contain bad data, this is a seperate step from accessing metadata +
    +
    7
    +
    +This step shows how to use Python’s built-in pathlib.Path library to write the file to disk. +
    +
    + + +
    +
    + +
    + +
    + + + + + \ No newline at end of file diff --git a/tutorials/quickstart_for_web_devs.html.md b/tutorials/quickstart_for_web_devs.html.md new file mode 100644 index 00000000..a719bc88 --- /dev/null +++ b/tutorials/quickstart_for_web_devs.html.md @@ -0,0 +1,1301 @@ +# Web Devs Quickstart + + + + +## Installation + +``` bash +pip install python-fasthtml +``` + +## A Minimal Application + +A minimal FastHTML application looks something like this: + +
    + +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/") +def get(): + return Titled("FastHTML", P("Let's do this!")) + +serve() +``` + +
    + +Line 1 +We import what we need for rapid development! A carefully-curated set of +FastHTML functions and other Python objects is brought into our global +namespace for convenience. + +Line 3 +We instantiate a FastHTML app with the `fast_app()` utility function. +This provides a number of really useful defaults that we’ll take +advantage of later in the tutorial. + +Line 5 +We use the `rt()` decorator to tell FastHTML what to return when a user +visits `/` in their browser. + +Line 6 +We connect this route to HTTP GET requests by defining a view function +called `get()`. + +Line 7 +A tree of Python function calls that return all the HTML required to +write a properly formed web page. You’ll soon see the power of this +approach. + +Line 9 +The +[`serve()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#serve) +utility configures and runs FastHTML using a library called `uvicorn`. + +Run the code: + +``` bash +python main.py +``` + +The terminal will look like this: + +``` bash +INFO: Uvicorn running on http://0.0.0.0:5001 (Press CTRL+C to quit) +INFO: Started reloader process [58058] using WatchFiles +INFO: Started server process [58060] +INFO: Waiting for application startup. +INFO: Application startup complete. +``` + +Confirm FastHTML is running by opening your web browser to +[127.0.0.1:5001](http://127.0.0.1:5001). You should see something like +the image below: + +![](quickstart-web-dev/quickstart-fasthtml.png) + +
    + +> **Note** +> +> While some linters and developers will complain about the wildcard +> import, it is by design here and perfectly safe. FastHTML is very +> deliberate about the objects it exports in `fasthtml.common`. If it +> bothers you, you can import the objects you need individually, though +> it will make the code more verbose and less readable. +> +> If you want to learn more about how FastHTML handles imports, we cover +> that [here](https://docs.fastht.ml/explains/faq.html#why-use-import). + +
    + +## A Minimal Charting Application + +The +[`Script`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#script) +function allows you to include JavaScript. You can use Python to +generate parts of your JS or JSON like this: + +``` python +import json +from fasthtml.common import * + +app, rt = fast_app(hdrs=(Script(src="https://cdn.plot.ly/plotly-2.32.0.min.js"),)) + +data = json.dumps({ + "data": [{"x": [1, 2, 3, 4],"type": "scatter"}, + {"x": [1, 2, 3, 4],"y": [16, 5, 11, 9],"type": "scatter"}], + "title": "Plotly chart in FastHTML ", + "description": "This is a demo dashboard", + "type": "scatter" +}) + + +@rt("/") +def get(): + return Titled("Chart Demo", Div(id="myDiv"), + Script(f"var data = {data}; Plotly.newPlot('myDiv', data);")) + +serve() +``` + +## Debug Mode + +When we can’t figure out a bug in FastHTML, we can run it in `DEBUG` +mode. When an error is thrown, the error screen is displayed in the +browser. This error setting should never be used in a deployed app. + +``` python +from fasthtml.common import * + +app, rt = fast_app(debug=True) + +@rt("/") +def get(): + 1/0 + return Titled("FastHTML Error!", P("Let's error!")) + +serve() +``` + +Line 3 +`debug=True` sets debug mode on. + +Line 7 +Python throws an error when it tries to divide an integer by zero. + +## Routing + +FastHTML builds upon FastAPI’s friendly decorator pattern for specifying +URLs, with extra features: + +
    + +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/") +def get(): + return Titled("FastHTML", P("Let's do this!")) + +@rt("/hello") +def get(): + return Titled("Hello, world!") + +serve() +``` + +
    + +Line 5 +The “/” URL on line 5 is the home of a project. This would be accessed +at [127.0.0.1:5001](http://127.0.0.1:5001). + +Line 9 +“/hello” URL on line 9 will be found by the project if the user visits +[127.0.0.1:5001/hello](http://127.0.0.1:5001/hello). + +
    + +> **Tip** +> +> It looks like `get()` is being defined twice, but that’s not the case. +> Each function decorated with `rt` is totally separate, and is injected +> into the router. We’re not calling them in the module’s namespace +> (`locals()`). Rather, we’re loading them into the routing mechanism +> using the `rt` decorator. + +
    + +You can do more! Read on to learn what we can do to make parts of the +URL dynamic. + +## Variables in URLs + +You can add variable sections to a URL by marking them with +`{variable_name}`. Your function then receives the `{variable_name}` as +a keyword argument, but only if it is the correct type. Here’s an +example: + +
    + +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/{name}/{age}") +def get(name: str, age: int): + return Titled(f"Hello {name.title()}, age {age}") + +serve() +``` + +
    + +Line 5 +We specify two variable names, `name` and `age`. + +Line 6 +We define two function arguments named identically to the variables. You +will note that we specify the Python types to be passed. + +Line 7 +We use these functions in our project. + +Try it out by going to this address: +[127.0.0.1:5001/uma/5](http://127.0.0.1:5001/uma/5). You should get a +page that says, + +> “Hello Uma, age 5”. + +### What happens if we enter incorrect data? + +The [127.0.0.1:5001/uma/5](http://127.0.0.1:5001/uma/5) URL works +because `5` is an integer. If we enter something that is not, such as +[127.0.0.1:5001/uma/five](http://127.0.0.1:5001/uma/five), then FastHTML +will return an error instead of a web page. + +
    + +> **FastHTML URL routing supports more complex types** +> +> The two examples we provide here use Python’s built-in `str` and `int` +> types, but you can use your own types, including more complex ones +> such as those defined by libraries like +> [attrs](https://pypi.org/project/attrs/), +> [pydantic](https://pypi.org/project/pydantic/), and even +> [sqlmodel](https://pypi.org/project/sqlmodel/). + +
    + +## HTTP Methods + +FastHTML matches function names to HTTP methods. So far the URL routes +we’ve defined have been for HTTP GET methods, the most common method for +web pages. + +Form submissions often are sent as HTTP POST. When dealing with more +dynamic web page designs, also known as Single Page Apps (SPA for +short), the need can arise for other methods such as HTTP PUT and HTTP +DELETE. The way FastHTML handles this is by changing the function name. + +
    + +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/") +def get(): + return Titled("HTTP GET", P("Handle GET")) + +@rt("/") +def post(): + return Titled("HTTP POST", P("Handle POST")) + +serve() +``` + +
    + +Line 6 +On line 6 because the `get()` function name is used, this will handle +HTTP GETs going to the `/` URI. + +Line 10 +On line 10 because the `post()` function name is used, this will handle +HTTP POSTs going to the `/` URI. + +## CSS Files and Inline Styles + +Here we modify default headers to demonstrate how to use the [Sakura CSS +microframework](https://github.com/oxalorg/sakura) instead of FastHTML’s +default of Pico CSS. + +
    + +**main.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app( + pico=False, + hdrs=( + Link(rel='stylesheet', href='assets/normalize.min.css', type='text/css'), + Link(rel='stylesheet', href='assets/sakura.css', type='text/css'), + Style("p {color: red;}") +)) + +@app.get("/") +def home(): + return Titled("FastHTML", + P("Let's do this!"), + ) + +serve() +``` + +
    + +Line 4 +By setting `pico` to `False`, FastHTML will not include `pico.min.css`. + +Line 7 +This will generate an HTML `` tag for sourcing the css for Sakura. + +Line 8 +If you want an inline styles, the +[`Style()`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#style) +function will put the result into the HTML. + +## Other Static Media File Locations + +As you saw, +[`Script`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#script) +and `Link` are specific to the most common static media use cases in web +apps: including JavaScript, CSS, and images. But it also works with +videos and other static media files. The default behavior is to look for +these files in the root directory - typically we don’t do anything +special to include them. We can change the default directory that is +looked in for files by adding the `static_path` parameter to the +`fast_app` function. + +``` python +app, rt = fast_app(static_path='public') +``` + +FastHTML also allows us to define a route that uses `FileResponse` to +serve the file at a specified path. This is useful for serving images, +videos, and other media files from a different directory without having +to change the paths of many files. So if we move the directory +containing the media files, we only need to change the path in one +place. In the example below, we call images from a directory called +`public`. + +``` python +@rt("/{fname:path}.{ext:static}") +async def get(fname:str, ext:str): + return FileResponse(f'public/{fname}.{ext}') +``` + +## Rendering Markdown + +``` python +from fasthtml.common import * + +hdrs = (MarkdownJS(), HighlightJS(langs=['python', 'javascript', 'html', 'css']), ) + +app, rt = fast_app(hdrs=hdrs) + +content = """ +Here are some _markdown_ elements. + +- This is a list item +- This is another list item +- And this is a third list item + +**Fenced code blocks work here.** +""" + +@rt('/') +def get(req): + return Titled("Markdown rendering example", Div(content,cls="marked")) + +serve() +``` + +## Code highlighting + +Here’s how to highlight code without any markdown configuration. + +``` python +from fasthtml.common import * + +# Add the HighlightJS built-in header +hdrs = (HighlightJS(langs=['python', 'javascript', 'html', 'css']),) + +app, rt = fast_app(hdrs=hdrs) + +code_example = """ +import datetime +import time + +for i in range(10): + print(f"{datetime.datetime.now()}") + time.sleep(1) +""" + +@rt('/') +def get(req): + return Titled("Markdown rendering example", + Div( + # The code example needs to be surrounded by + # Pre & Code elements + Pre(Code(code_example)) + )) + +serve() +``` + +## Defining new `ft` components + +We can build our own `ft` components and combine them with other +components. The simplest method is defining them as a function. + +``` python +from fasthtml.common import * +``` + +``` python +def hero(title, statement): + return Div(H1(title),P(statement), cls="hero") + +# usage example +Main( + hero("Hello World", "This is a hero statement") +) +``` + +``` html +
    +

    Hello World

    +

    This is a hero statement

    +
    +
    +``` + +### Pass through components + +For when we need to define a new component that allows zero-to-many +components to be nested within them, we lean on Python’s `*args` and +`**kwargs` mechanism. Useful for creating page layout controls. + +``` python +def layout(*args, **kwargs): + """Dashboard layout for all our dashboard views""" + return Main( + H1("Dashboard"), + Div(*args, **kwargs), + cls="dashboard", + ) + +# usage example +layout( + Ul(*[Li(o) for o in range(3)]), + P("Some content", cls="description"), +) +``` + +``` html +

    Dashboard

    +
    +
      +
    • 0
    • +
    • 1
    • +
    • 2
    • +
    +

    Some content

    +
    +
    +``` + +### Dataclasses as ft components + +While functions are easy to read, for more complex components some might +find it easier to use a dataclass. + +``` python +from dataclasses import dataclass + +@dataclass +class Hero: + title: str + statement: str + + def __ft__(self): + """ The __ft__ method renders the dataclass at runtime.""" + return Div(H1(self.title),P(self.statement), cls="hero") + +# usage example +Main( + Hero("Hello World", "This is a hero statement") +) +``` + +``` html +
    +

    Hello World

    +

    This is a hero statement

    +
    +
    +``` + +## Testing views in notebooks + +Because of the ASGI event loop it is currently impossible to run +FastHTML inside a notebook. However, we can still test the output of our +views. To do this, we leverage Starlette, an ASGI toolkit that FastHTML +uses. + +``` python +# First we instantiate our app, in this case we remove the +# default headers to reduce the size of the output. +app, rt = fast_app(default_hdrs=False) + +# Setting up the Starlette test client +from starlette.testclient import TestClient +client = TestClient(app) + +# Usage example +@rt("/") +def get(): + return Titled("FastHTML is awesome", + P("The fastest way to create web apps in Python")) + +print(client.get("/").text) +``` + + + + + FastHTML is awesome + +

    FastHTML is awesome

    +

    The fastest way to create web apps in Python

    +
    + + +## Forms + +To validate data coming from users, first define a dataclass +representing the data you want to check. Here’s an example representing +a signup form. + +``` python +from dataclasses import dataclass + +@dataclass +class Profile: email:str; phone:str; age:int +``` + +Create an FT component representing an empty version of that form. Don’t +pass in any value to fill the form, that gets handled later. + +``` python +profile_form = Form(method="post", action="/profile")( + Fieldset( + Label('Email', Input(name="email")), + Label("Phone", Input(name="phone")), + Label("Age", Input(name="age")), + ), + Button("Save", type="submit"), + ) +profile_form +``` + +``` html +
    +``` + +Once the dataclass and form function are completed, we can add data to +the form. To do that, instantiate the profile dataclass: + +``` python +profile = Profile(email='john@example.com', phone='123456789', age=5) +profile +``` + + Profile(email='john@example.com', phone='123456789', age=5) + +Then add that data to the `profile_form` using FastHTML’s +[`fill_form`](https://AnswerDotAI.github.io/fasthtml/api/components.html#fill_form) +class: + +``` python +fill_form(profile_form, profile) +``` + +``` html +
    +``` + +### Forms with views + +The usefulness of FastHTML forms becomes more apparent when they are +combined with FastHTML views. We’ll show how this works by using the +test client from above. First, let’s create a SQlite database: + +``` python +db = database("profiles.db") +profiles = db.create(Profile, pk="email") +``` + +Now we insert a record into the database: + +``` python +profiles.insert(profile) +``` + + Profile(email='john@example.com', phone='123456789', age=5) + +And we can then demonstrate in the code that form is filled and +displayed to the user. + +``` python +@rt("/profile/{email}") +def profile(email:str): + profile = profiles[email] + filled_profile_form = fill_form(profile_form, profile) + return Titled(f'Profile for {profile.email}', filled_profile_form) + +print(client.get(f"/profile/john@example.com").text) +``` + +Line 3 +Fetch the profile using the profile table’s `email` primary key + +Line 4 +Fill the form for display. + + + + + + + Profile for john@example.com + +

    Profile for john@example.com

    +
    + + +And now let’s demonstrate making a change to the data. + +``` python +@rt("/profile") +def post(profile: Profile): + profiles.update(profile) + return RedirectResponse(url=f"/profile/{profile.email}") + +new_data = dict(email='john@example.com', phone='7654321', age=25) +print(client.post("/profile", data=new_data).text) +``` + +Line 2 +We use the `Profile` dataclass definition to set the type for the +incoming `profile` content. This validates the field types for the +incoming data + +Line 3 +Taking our validated data, we updated the profiles table + +Line 4 +We redirect the user back to their profile view + +Line 7 +The display is of the profile form view showing the changes in data. + + + + + + + Profile for john@example.com + +

    Profile for john@example.com

    +
    + + +## Strings and conversion order + +The general rules for rendering are: - `__ft__` method will be called +(for default components like `P`, `H2`, etc. or if you define your own +components) - If you pass a string, it will be escaped - On other python +objects, `str()` will be called + +As a consequence, if you want to include plain HTML tags directly into +e.g. a `Div()` they will get escaped by default (as a security measure +to avoid code injections). This can be avoided by using `NotStr()`, a +convenient way to reuse python code that returns already HTML. If you +use pandas, you can use `pandas.DataFrame.to_html()` to get a nice +table. To include the output a FastHTML, wrap it in `NotStr()`, like +`Div(NotStr(df.to_html()))`. + +Above we saw how a dataclass behaves with the `__ft__` method defined. +On a plain dataclass, `str()` will be called (but not escaped). + +``` python +from dataclasses import dataclass + +@dataclass +class Hero: + title: str + statement: str + +# rendering the dataclass with the default method +Main( + Hero("

    Hello World

    ", "This is a hero statement") +) +``` + +``` html +
    Hero(title='

    Hello World

    ', statement='This is a hero statement')
    +``` + +``` python +# This will display the HTML as text on your page +Div("Let's include some HTML here:
    Some HTML
    ") +``` + +``` html +
    Let's include some HTML here: <div>Some HTML</div>
    +``` + +``` python +# Keep the string untouched, will be rendered on the page +Div(NotStr("

    Some HTML

    ")) +``` + +``` html +

    Some HTML

    +``` + +## Custom exception handlers + +FastHTML allows customization of exception handlers, but does so +gracefully. What this means is by default it includes all the `` +tags needed to display attractive content. Try it out! + +``` python +from fasthtml.common import * + +def not_found(req, exc): return Titled("404: I don't exist!") + +exception_handlers = {404: not_found} + +app, rt = fast_app(exception_handlers=exception_handlers) + +@rt('/') +def get(): + return (Titled("Home page", P(A(href="/oops")("Click to generate 404 error")))) + +serve() +``` + +We can also use lambda to make things more terse: + +``` python +from fasthtml.common import * + +exception_handlers={ + 404: lambda req, exc: Titled("404: I don't exist!"), + 418: lambda req, exc: Titled("418: I'm a teapot!") +} + +app, rt = fast_app(exception_handlers=exception_handlers) + +@rt('/') +def get(): + return (Titled("Home page", P(A(href="/oops")("Click to generate 404 error")))) + +serve() +``` + +## Cookies + +We can set cookies using the +[`cookie()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#cookie) +function. In our example, we’ll create a `timestamp` cookie. + +``` python +from datetime import datetime +from IPython.display import HTML +``` + +``` python +@rt("/settimestamp") +def get(req): + now = datetime.now() + return P(f'Set to {now}'), cookie('now', datetime.now()) + +HTML(client.get('/settimestamp').text) +``` + + + + +FastHTML page + +

    Set to 2024-09-26 15:33:48.141869

    + + + +Now let’s get it back using the same name for our parameter as the +cookie name. + +``` python +@rt('/gettimestamp') +def get(now:parsed_date): return f'Cookie was set at time {now.time()}' + +client.get('/gettimestamp').text +``` + + 'Cookie was set at time 15:33:48.141903' + +## Sessions + +For convenience and security, FastHTML has a mechanism for storing small +amounts of data in the user’s browser. We can do this by adding a +`session` argument to routes. FastHTML sessions are Python dictionaries, +and we can leverage to our benefit. The example below shows how to +concisely set and get sessions. + +``` python +@rt('/adder/{num}') +def get(session, num: int): + session.setdefault('sum', 0) + session['sum'] = session.get('sum') + num + return Response(f'The sum is {session["sum"]}.') +``` + +## Toasts (also known as Messages) + +Toasts, sometimes called “Messages” are small notifications usually in +colored boxes used to notify users that something has happened. Toasts +can be of four types: + +- info +- success +- warning +- error + +Examples toasts might include: + +- “Payment accepted” +- “Data submitted” +- “Request approved” + +Toasts require the use of the `setup_toasts()` function plus every view +needs these two features: + +- The session argument +- Must return FT components + +``` python +setup_toasts(app) + +@rt('/toasting') +def get(session): + # Normally one toast is enough, this allows us to see + # different toast types in action. + add_toast(session, f"Toast is being cooked", "info") + add_toast(session, f"Toast is ready", "success") + add_toast(session, f"Toast is getting a bit crispy", "warning") + add_toast(session, f"Toast is burning!", "error") + return Titled("I like toast") +``` + +Line 1 +`setup_toasts` is a helper function that adds toast dependencies. +Usually this would be declared right after `fast_app()` + +Line 4 +Toasts require sessions + +Line 11 +Views with Toasts must return FT or FtResponse components. + +💡 `setup_toasts` takes a `duration` input that allows you to specify +how long a toast will be visible before disappearing. For example +`setup_toasts(duration=5)` sets the toasts duration to 5 seconds. By +default toasts disappear after 10 seconds. + +## Authentication and authorization + +In FastHTML the tasks of authentication and authorization are handled +with Beforeware. Beforeware are functions that run before the route +handler is called. They are useful for global tasks like ensuring users +are authenticated or have permissions to access a view. + +First, we write a function that accepts a request and session arguments: + +``` python +# Status code 303 is a redirect that can change POST to GET, +# so it's appropriate for a login page. +login_redir = RedirectResponse('/login', status_code=303) + +def user_auth_before(req, sess): + # The `auth` key in the request scope is automatically provided + # to any handler which requests it, and can not be injected + # by the user using query params, cookies, etc, so it should + # be secure to use. + auth = req.scope['auth'] = sess.get('auth', None) + # If the session key is not there, it redirects to the login page. + if not auth: return login_redir +``` + +Now we pass our `user_auth_before` function as the first argument into a +[`Beforeware`](https://AnswerDotAI.github.io/fasthtml/api/core.html#beforeware) +class. We also pass a list of regular expressions to the `skip` +argument, designed to allow users to still get to the home and login +pages. + +``` python +beforeware = Beforeware( + user_auth_before, + skip=[r'/favicon\.ico', r'/static/.*', r'.*\.css', r'.*\.js', '/login', '/'] +) + +app, rt = fast_app(before=beforeware) +``` + +## Server-sent events (SSE) + +With [server-sent +events](https://developer.mozilla.org/en-US/docs/Web/API/Server-sent_events), +it’s possible for a server to send new data to a web page at any time, +by pushing messages to the web page. Unlike WebSockets, SSE can only go +in one direction: server to client. SSE is also part of the HTTP +specification unlike WebSockets which uses its own specification. + +FastHTML introduces several tools for working with SSE which are covered +in the example below. While concise, there’s a lot going on in this +function so we’ve annotated it quite a bit. + +``` python +import random +from asyncio import sleep +from fasthtml.common import * + +hdrs=(Script(src="https://unpkg.com/htmx-ext-sse@2.2.1/sse.js"),) +app,rt = fast_app(hdrs=hdrs) + +@rt +def index(): + return Titled("SSE Random Number Generator", + P("Generate pairs of random numbers, as the list grows scroll downwards."), + Div(hx_ext="sse", + sse_connect="/number-stream", + hx_swap="beforeend show:bottom", + sse_swap="message")) + +shutdown_event = signal_shutdown() + +async def number_generator(): + while not shutdown_event.is_set(): + data = Article(random.randint(1, 100)) + yield sse_message(data) + await sleep(1) + +@rt("/number-stream") +async def get(): return EventStream(number_generator()) +``` + +Line 5 +Import the HTMX SSE extension + +Line 12 +Tell HTMX to load the SSE extension + +Line 13 +Look at the `/number-stream` endpoint for SSE content + +Line 14 +When new items come in from the SSE endpoint, add them at the end of the +current content within the div. If they go beyond the screen, scroll +downwards + +Line 15 +Specify the name of the event. FastHTML’s default event name is +“message”. Only change if you have more than one call to SSE endpoints +within a view + +Line 17 +Set up the asyncio event loop + +Line 19 +Don’t forget to make this an `async` function! + +Line 20 +Iterate through the asyncio event loop + +Line 22 +We yield the data. Data ideally should be comprised of FT components as +that plugs nicely into HTMX in the browser + +Line 26 +The endpoint view needs to be an async function that returns a +[`EventStream`](https://AnswerDotAI.github.io/fasthtml/api/core.html#eventstream) + +## Websockets + +With websockets we can have bi-directional communications between a +browser and client. Websockets are useful for things like chat and +certain types of games. While websockets can be used for single +direction messages from the server (i.e. telling users that a process is +finished), that task is arguably better suited for SSE. + +FastHTML provides useful tools for adding websockets to your pages. + +``` python +from fasthtml.common import * +from asyncio import sleep + +app, rt = fast_app(exts='ws') + +def mk_inp(): return Input(id='msg', autofocus=True) + +@rt('/') +async def get(request): + cts = Div( + Div(id='notifications'), + Form(mk_inp(), id='form', ws_send=True), + hx_ext='ws', ws_connect='/ws') + return Titled('Websocket Test', cts) + +async def on_connect(send): + print('Connected!') + await send(Div('Hello, you have connected', id="notifications")) + +async def on_disconnect(ws): + print('Disconnected!') + +@app.ws('/ws', conn=on_connect, disconn=on_disconnect) +async def ws(msg:str, send): + await send(Div('Hello ' + msg, id="notifications")) + await sleep(2) + return Div('Goodbye ' + msg, id="notifications"), mk_inp() +``` + +Line 4 +To use websockets in FastHTML, you must instantiate the app with `exts` +set to ‘ws’ + +Line 6 +As we want to use websockets to reset the form, we define the `mk_input` +function that can be called from multiple locations + +Line 12 +We create the form and mark it with the `ws_send` attribute, which is +documented here in the [HTMX websocket +specification](https://v1.htmx.org/extensions/web-sockets/). This tells +HTMX to send a message to the nearest websocket based on the trigger for +the form element, which for forms is pressing the `enter` key, an action +considered to be a form submission + +Line 13 +This is where the HTMX extension is loaded (`hx_ext='ws'`) and the +nearest websocket is defined (`ws_connect='/ws'`) + +Line 16 +When a websocket first connects we can optionally have it call a +function that accepts a `send` argument. The `send` argument will push a +message to the browser. + +Line 18 +Here we use the `send` function that was passed into the `on_connect` +function to send a `Div` with an `id` of `notifications` that HTMX +assigns to the element in the page that already has an `id` of +`notifications` + +Line 20 +When a websocket disconnects we can call a function which takes no +arguments. Typically the role of this function is to notify the server +to take an action. In this case, we print a simple message to the +console + +Line 23 +We use the `app.ws` decorator to mark that `/ws` is the route for our +websocket. We also pass in the two optional `conn` and `disconn` +parameters to this decorator. As a fun experiment, remove the `conn` and +`disconn` arguments and see what happens + +Line 24 +Define the `ws` function as async. This is necessary for ASGI to be able +to serve websockets. The function accepts two arguments, a `msg` that is +user input from the browser, and a `send` function for pushing data back +to the browser + +Line 25 +The `send` function is used here to send HTML back to the page. As the +HTML has an `id` of `notifications`, HTMX will overwrite what is already +on the page with the same ID + +Line 27 +The websocket function can also be used to return a value. In this case, +it is a tuple of two HTML elements. HTMX will take the elements and +replace them where appropriate. As both have `id` specified +(`notifications` and `msg` respectively), they will replace their +predecessor on the page. + +## File Uploads + +A common task in web development is uploading files. The examples below +are for uploading files to the hosting server, with information about +the uploaded file presented to the user. + +
    + +> **File uploads in production can be dangerous** +> +> File uploads can be the target of abuse, accidental or intentional. +> That means users may attempt to upload files that are too large or +> present a security risk. This is especially of concern for public +> facing apps. File upload security is outside the scope of this +> tutorial, for now we suggest reading the [OWASP File Upload Cheat +> Sheet](https://cheatsheetseries.owasp.org/cheatsheets/File_Upload_Cheat_Sheet.html). + +
    + +### Single File Uploads + +``` python +from fasthtml.common import * +from pathlib import Path + +app, rt = fast_app() + +upload_dir = Path("filez") +upload_dir.mkdir(exist_ok=True) + +@rt('/') +def get(): + return Titled("File Upload Demo", + Article( + Form(hx_post=upload, hx_target="#result-one")( + Input(type="file", name="file"), + Button("Upload", type="submit", cls='secondary'), + ), + Div(id="result-one") + ) + ) + +def FileMetaDataCard(file): + return Article( + Header(H3(file.filename)), + Ul( + Li('Size: ', file.size), + Li('Content Type: ', file.content_type), + Li('Headers: ', file.headers), + ) + ) + +@rt +async def upload(file: UploadFile): + card = FileMetaDataCard(file) + filebuffer = await file.read() + (upload_dir / file.filename).write_bytes(filebuffer) + return card + +serve() +``` + +Line 13 +Every form rendered with the +[`Form`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#form) FT +component defaults to `enctype="multipart/form-data"` + +Line 14 +Don’t forget to set the `Input` FT Component’s type to `file` + +Line 32 +The upload view should receive a [Starlette +UploadFile](https://www.starlette.io/requests/#request-files) type. You +can add other form variables + +Line 33 +We can access the metadata of the card (filename, size, content_type, +headers), a quick and safe process. We set that to the card variable + +Line 34 +In order to access the contents contained within a file we use the +`await` method to read() it. As files may be quite large or contain bad +data, this is a seperate step from accessing metadata + +Line 35 +This step shows how to use Python’s built-in `pathlib.Path` library to +write the file to disk. + +### Multiple File Uploads + +``` python +from fasthtml.common import * +from pathlib import Path + +app, rt = fast_app() + +upload_dir = Path("filez") +upload_dir.mkdir(exist_ok=True) + +@rt('/') +def get(): + return Titled("Multiple File Upload Demo", + Article( + Form(hx_post=upload_many, hx_target="#result-many")( + Input(type="file", name="files", multiple=True), + Button("Upload", type="submit", cls='secondary'), + ), + Div(id="result-many") + ) + ) + +def FileMetaDataCard(file): + return Article( + Header(H3(file.filename)), + Ul( + Li('Size: ', file.size), + Li('Content Type: ', file.content_type), + Li('Headers: ', file.headers), + ) + ) + +@rt +async def upload_many(files: list[UploadFile]): + cards = [] + for file in files: + cards.append(FileMetaDataCard(file)) + filebuffer = await file.read() + (upload_dir / file.filename).write_bytes(filebuffer) + return cards + +serve() +``` + +Line 13 +Every form rendered with the +[`Form`](https://AnswerDotAI.github.io/fasthtml/api/xtend.html#form) FT +component defaults to `enctype="multipart/form-data"` + +Line 14 +Don’t forget to set the `Input` FT Component’s type to `file` and assign +the multiple attribute to `True` + +Line 32 +The upload view should receive a `list` containing the [Starlette +UploadFile](https://www.starlette.io/requests/#request-files) type. You +can add other form variables + +Line 34 +Iterate through the files + +Line 35 +We can access the metadata of the card (filename, size, content_type, +headers), a quick and safe process. We add that to the cards variable + +Line 36 +In order to access the contents contained within a file we use the +`await` method to read() it. As files may be quite large or contain bad +data, this is a seperate step from accessing metadata + +Line 37 +This step shows how to use Python’s built-in `pathlib.Path` library to +write the file to disk. diff --git a/unpublished/tutorial_for_web_devs.html b/unpublished/tutorial_for_web_devs.html new file mode 100644 index 00000000..4f2c7cce --- /dev/null +++ b/unpublished/tutorial_for_web_devs.html @@ -0,0 +1,1077 @@ + + + + + + + + + + +BYO Blog – fasthtml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    +
    + + +
    + +
    + + +
    + + + +
    + +
    +
    +

    BYO Blog

    +
    + +
    +
    + Learn the foundations of FastHTML by creating your own blogging system from scratch. +
    +
    + + +
    + + + + +
    + + + +
    + + + +
    +
    +
    + +
    +
    +Caution +
    +
    +
    +

    This document is a work in progress.

    +
    +
    +

    In this tutorial we’re going to write a blog by example. Blogs are a good way to learn a web framework as they start simple yet can get surprisingly sophistated. The wikipedia definition of a blog is “an informational website consisting of discrete, often informal diary-style text entries (posts) informal diary-style text entries (posts)”, which means we need to provide these basic features:

    +
      +
    • A list of articles
    • +
    • A means to create/edit/delete the articles
    • +
    • An attractive but accessible layout
    • +
    +

    We’ll also add in these features, so the blog can become a working site:

    +
      +
    • RSS feed
    • +
    • Pages independent of the list of articles (about and contact come to mind)
    • +
    • Import and Export of articles
    • +
    • Tagging and categorization of data
    • +
    • Deployment
    • +
    • Ability to scale for large volumes of readers
    • +
    +
    +

    How to best use this tutorial

    +

    We could copy/paste every code example in sequence and have a finished blog at the end. However, it’s debatable how much we will learn through the copy/paste method. We’re not saying its impossible to learn through copy/paste, we’re just saying it’s not that of an efficient way to learn. It’s analogous to learning how to play a musical instrument or sport or video game by watching other people do it - you can learn some but its not the same as doing.

    +

    A better approach is to type out every line of code in this tutorial. This forces us to run the code through our brains, giving us actual practice in how to write FastHTML and Pythoncode and forcing us to debug our own mistakes. In some cases we’ll repeat similar tasks - a key component in achieving mastery in anything. Coming back to the instrument/sport/video game analogy, it’s exactly like actually practicing an instrument, sport, or video game. Through practice and repetition we eventually achieve mastery.

    +
    +
    +

    Installing FastHTML

    +

    FastHTML is just Python. Installation is often done with pip:

    +
    pip install python-fasthtml
    +
    +
    +

    A minimal FastHTML app

    +

    First, create the directory for our project using Python’s pathlib module:

    +
    import pathlib
    +pathlib.Path('blog-system').mkdir()
    +

    Now that we have our directory, let’s create a minimal FastHTML site in it.

    +
    +
    +
    blog-system/minimal.py
    +
    +
    from fasthtml.common import * 
    +
    +app, rt = fast_app()  
    +
    +@rt("/") 
    +def get():
    +    return Titled("FastHTML", P("Let's do this!")) 
    +
    +serve()
    +
    +

    Run that with python minimal.py and you should get something like this:

    +
    python minimal.py 
    +Link: http://localhost:5001
    +INFO:     Will watch for changes in these directories: ['/Users/pydanny/projects/blog-system']
    +INFO:     Uvicorn running on http://0.0.0.0:5001 (Press CTRL+C to quit)
    +INFO:     Started reloader process [46572] using WatchFiles
    +INFO:     Started server process [46576]
    +INFO:     Waiting for application startup.
    +INFO:     Application startup complete.
    +

    Confirm FastHTML is running by opening your web browser to 127.0.0.1:5001. You should see something like the image below:

    +

    +
    +
    +
    + +
    +
    +What about the import *? +
    +
    +
    +

    For those worried about the use of import * rather than a PEP8-style declared namespace, understand that __all__ is defined in FastHTML’s common module. That means that only the symbols (functions, classes, and other things) the framework wants us to have will be brought into our own code via import *. Read importing from a package) for more information.

    +

    Nevertheless, if we want to use a defined namespace we can do so. Here’s an example:

    +
    from fasthtml import common as fh
    +
    +
    +app, rt = fh.fast_app()  
    +
    +@rt("/") 
    +def get():
    +    return fh.Titled("FastHTML", fh.P("Let's do this!")) 
    +
    +fh.serve()
    +
    +
    +
    +
    +

    Looking more closely at our app

    +

    Let’s look more closely at our application. Every line is packed with powerful features of FastHTML:

    +
    +
    +
    blog-system/minimal.py
    +
    +
    1from fasthtml.common import *
    +
    +2app, rt = fast_app()
    +
    +3@rt("/")
    +4def get():
    +5    return Titled("FastHTML", P("Let's do this!"))
    +
    +6serve()
    +
    +
    +
    1
    +
    +The top level namespace of Fast HTML (fasthtml.common) contains everything we need from FastHTML to build applications. A carefully-curated set of FastHTML functions and other Python objects is brought into our global namespace for convenience. +
    +
    2
    +
    +We instantiate a FastHTML app with the fast_app() utility function. This provides a number of really useful defaults that we’ll modify or take advantage of later in the tutorial. +
    +
    3
    +
    +We use the rt() decorator to tell FastHTML what to return when a user visits / in their browser. +
    +
    4
    +
    +We connect this route to HTTP GET requests by defining a view function called get(). +
    +
    5
    +
    +A tree of Python function calls that return all the HTML required to write a properly formed web page. You’ll soon see the power of this approach. +
    +
    6
    +
    +The serve() utility configures and runs FastHTML using a library called uvicorn. Any changes to this module will be reloaded into the browser. +
    +
    +
    +
    +

    Adding dynamic content to our minimal app

    +

    Our page is great, but we’ll make it better. Let’s add a randomized list of letters to the page. Every time the page reloads, a new list of varying length will be generated.

    +
    +
    +
    blog-system/random_letters.py
    +
    +
    from fasthtml.common import *
    +1import string, random
    +
    +app, rt = fast_app()
    +
    +@rt("/")
    +def get():
    +2    letters = random.choices(string.ascii_uppercase, k=random.randint(5, 20))
    +3    items = [Li(c) for c in letters]
    +    return Titled("Random lists of letters",
    +4        Ul(*items)
    +    ) 
    +
    +serve()
    +
    +
    +
    1
    +
    +The string and random libraries are part of Python’s standard library +
    +
    2
    +
    +We use these libraries to generate a random length list of random letters called letters +
    +
    3
    +
    +Using letters as the base we use list comprehension to generate a list of Li ft display components, each with their own letter and save that to the variable items +
    +
    4
    +
    +Inside a call to the Ul() ft component we use Python’s *args special syntax on the items variable. Therefore *list is treated not as one argument but rather a set of them. +
    +
    +

    When this is run, it will generate something like this with a different random list of letters for each page load:

    +

    +
    +
    +

    Storing the articles

    +

    The most basic component of a blog is a series of articles sorted by date authored. Rather than a database we’re going to use our computer’s harddrive to store a set of markdown files in a directory within our blog called posts. First, let’s create the directory and some test files we can use to search for:

    +
    +
    from fastcore.utils import *
    +
    +
    +
    # Create some dummy posts
    +posts = Path("posts")
    +posts.mkdir(exist_ok=True)
    +for i in range(10): (posts/f"article_{i}.md").write_text(f"This is article {i}")
    +
    +

    Searching for these files can be done with pathlib.

    +
    +
    import pathlib
    +posts.ls()
    +
    +
    (#10) [Path('posts/article_5.md'),Path('posts/article_1.md'),Path('posts/article_0.md'),Path('posts/article_4.md'),Path('posts/article_3.md'),Path('posts/article_7.md'),Path('posts/article_6.md'),Path('posts/article_2.md'),Path('posts/article_9.md'),Path('posts/article_8.md')]
    +
    +
    +
    +
    +
    + +
    +
    +Tip +
    +
    +
    +

    Python’s pathlib library is quite useful and makes file search and manipulation much easier. There’s many uses for it and is compatible across operating systems.

    +
    +
    +
    +
    +

    Creating the blog home page

    +

    We now have enough tools that we can create the home page. Let’s create a new Python file and write out our simple view to list the articles in our blog.

    +
    +
    +
    blog-system/main.py
    +
    +
    from fasthtml.common import *
    +import pathlib
    +
    +app, rt = fast_app()
    +
    +@rt("/")
    +def get():
    +    fnames = pathlib.Path("posts").rglob("*.md")
    +    items = [Li(A(fname, href=fname)) for fname in fnames]    
    +    return Titled("My Blog",
    +        Ul(*items)
    +    ) 
    +
    +serve()
    +
    +
    +
    for p in posts.ls(): p.unlink()
    +
    + + +
    + +
    + +
    + + + + + \ No newline at end of file diff --git a/unpublished/tutorial_for_web_devs.html.md b/unpublished/tutorial_for_web_devs.html.md new file mode 100644 index 00000000..76634852 --- /dev/null +++ b/unpublished/tutorial_for_web_devs.html.md @@ -0,0 +1,319 @@ +# BYO Blog + + + + +
    + +> **Caution** +> +> This document is a work in progress. + +
    + +In this tutorial we’re going to write a blog by example. Blogs are a +good way to learn a web framework as they start simple yet can get +surprisingly sophistated. The [wikipedia definition of a +blog](https://en.wikipedia.org/wiki/Blog) is “an informational website +consisting of discrete, often informal diary-style text entries (posts) +informal diary-style text entries (posts)”, which means we need to +provide these basic features: + +- A list of articles +- A means to create/edit/delete the articles +- An attractive but accessible layout + +We’ll also add in these features, so the blog can become a working site: + +- RSS feed +- Pages independent of the list of articles (about and contact come to + mind) +- Import and Export of articles +- Tagging and categorization of data +- Deployment +- Ability to scale for large volumes of readers + +## How to best use this tutorial + +We could copy/paste every code example in sequence and have a finished +blog at the end. However, it’s debatable how much we will learn through +the copy/paste method. We’re not saying its impossible to learn through +copy/paste, we’re just saying it’s not that of an efficient way to +learn. It’s analogous to learning how to play a musical instrument or +sport or video game by watching other people do it - you can learn some +but its not the same as doing. + +A better approach is to type out every line of code in this tutorial. +This forces us to run the code through our brains, giving us actual +practice in how to write FastHTML and Pythoncode and forcing us to debug +our own mistakes. In some cases we’ll repeat similar tasks - a key +component in achieving mastery in anything. Coming back to the +instrument/sport/video game analogy, it’s exactly like actually +practicing an instrument, sport, or video game. Through practice and +repetition we eventually achieve mastery. + +## Installing FastHTML + +FastHTML is *just Python*. Installation is often done with pip: + +``` shellscript +pip install python-fasthtml +``` + +## A minimal FastHTML app + +First, create the directory for our project using Python’s +[pathlib](https://docs.python.org/3/library/pathlib.html) module: + +``` python +import pathlib +pathlib.Path('blog-system').mkdir() +``` + +Now that we have our directory, let’s create a minimal FastHTML site in +it. + +
    + +**blog-system/minimal.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/") +def get(): + return Titled("FastHTML", P("Let's do this!")) + +serve() +``` + +
    + +Run that with `python minimal.py` and you should get something like +this: + +``` shellscript +python minimal.py +Link: http://localhost:5001 +INFO: Will watch for changes in these directories: ['/Users/pydanny/projects/blog-system'] +INFO: Uvicorn running on http://0.0.0.0:5001 (Press CTRL+C to quit) +INFO: Started reloader process [46572] using WatchFiles +INFO: Started server process [46576] +INFO: Waiting for application startup. +INFO: Application startup complete. +``` + +Confirm FastHTML is running by opening your web browser to +[127.0.0.1:5001](http://127.0.0.1:5001). You should see something like +the image below: + +![](quickstart-web-dev/quickstart-fasthtml.png) + +
    + +> **What about the `import *`?** +> +> For those worried about the use of `import *` rather than a PEP8-style +> declared namespace, understand that `__all__` is defined in FastHTML’s +> common module. That means that only the symbols (functions, classes, +> and other things) the framework wants us to have will be brought into +> our own code via `import *`. Read [importing from a +> package](https://docs.python.org/3/tutorial/modules.html#importing-from-a-package)) +> for more information. +> +> Nevertheless, if we want to use a defined namespace we can do so. +> Here’s an example: +> +> ``` python +> from fasthtml import common as fh +> +> +> app, rt = fh.fast_app() +> +> @rt("/") +> def get(): +> return fh.Titled("FastHTML", fh.P("Let's do this!")) +> +> fh.serve() +> ``` + +
    + +## Looking more closely at our app + +Let’s look more closely at our application. Every line is packed with +powerful features of FastHTML: + +
    + +**blog-system/minimal.py** + +``` python +from fasthtml.common import * + +app, rt = fast_app() + +@rt("/") +def get(): + return Titled("FastHTML", P("Let's do this!")) + +serve() +``` + +
    + +Line 1 +The top level namespace of Fast HTML (fasthtml.common) contains +everything we need from FastHTML to build applications. A +carefully-curated set of FastHTML functions and other Python objects is +brought into our global namespace for convenience. + +Line 3 +We instantiate a FastHTML app with the `fast_app()` utility function. +This provides a number of really useful defaults that we’ll modify or +take advantage of later in the tutorial. + +Line 5 +We use the `rt()` decorator to tell FastHTML what to return when a user +visits `/` in their browser. + +Line 6 +We connect this route to HTTP GET requests by defining a view function +called `get()`. + +Line 7 +A tree of Python function calls that return all the HTML required to +write a properly formed web page. You’ll soon see the power of this +approach. + +Line 9 +The +[`serve()`](https://AnswerDotAI.github.io/fasthtml/api/core.html#serve) +utility configures and runs FastHTML using a library called `uvicorn`. +Any changes to this module will be reloaded into the browser. + +## Adding dynamic content to our minimal app + +Our page is great, but we’ll make it better. Let’s add a randomized list +of letters to the page. Every time the page reloads, a new list of +varying length will be generated. + +
    + +**blog-system/random_letters.py** + +``` python +from fasthtml.common import * +import string, random + +app, rt = fast_app() + +@rt("/") +def get(): + letters = random.choices(string.ascii_uppercase, k=random.randint(5, 20)) + items = [Li(c) for c in letters] + return Titled("Random lists of letters", + Ul(*items) + ) + +serve() +``` + +
    + +Line 2 +The `string` and `random` libraries are part of Python’s standard +library + +Line 8 +We use these libraries to generate a random length list of random +letters called `letters` + +Line 9 +Using `letters` as the base we use list comprehension to generate a list +of `Li` ft display components, each with their own letter and save that +to the variable `items` + +Line 11 +Inside a call to the `Ul()` ft component we use Python’s `*args` special +syntax on the `items` variable. Therefore `*list` is treated not as one +argument but rather a set of them. + +When this is run, it will generate something like this with a different +random list of letters for each page load: + +![](web-dev-tut/random-list-letters.png) + +## Storing the articles + +The most basic component of a blog is a series of articles sorted by +date authored. Rather than a database we’re going to use our computer’s +harddrive to store a set of markdown files in a directory within our +blog called `posts`. First, let’s create the directory and some test +files we can use to search for: + +``` python +from fastcore.utils import * +``` + +``` python +# Create some dummy posts +posts = Path("posts") +posts.mkdir(exist_ok=True) +for i in range(10): (posts/f"article_{i}.md").write_text(f"This is article {i}") +``` + +Searching for these files can be done with pathlib. + +``` python +import pathlib +posts.ls() +``` + + (#10) [Path('posts/article_5.md'),Path('posts/article_1.md'),Path('posts/article_0.md'),Path('posts/article_4.md'),Path('posts/article_3.md'),Path('posts/article_7.md'),Path('posts/article_6.md'),Path('posts/article_2.md'),Path('posts/article_9.md'),Path('posts/article_8.md')] + +
    + +> **Tip** +> +> Python’s [pathlib](https://docs.python.org/3/library/pathlib.html) +> library is quite useful and makes file search and manipulation much +> easier. There’s many uses for it and is compatible across operating +> systems. + +
    + +## Creating the blog home page + +We now have enough tools that we can create the home page. Let’s create +a new Python file and write out our simple view to list the articles in +our blog. + +
    + +**blog-system/main.py** + +``` python +from fasthtml.common import * +import pathlib + +app, rt = fast_app() + +@rt("/") +def get(): + fnames = pathlib.Path("posts").rglob("*.md") + items = [Li(A(fname, href=fname)) for fname in fnames] + return Titled("My Blog", + Ul(*items) + ) + +serve() +``` + +
    + +``` python +for p in posts.ls(): p.unlink() +``` diff --git a/unpublished/web-dev-tut/random-list-letters.png b/unpublished/web-dev-tut/random-list-letters.png new file mode 100644 index 0000000000000000000000000000000000000000..a63a7d59ae40916799bd92af434086fa4237c27f GIT binary patch literal 18818 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 OCHY=fv|JeE_kRFLW;KKW literal 0 HcmV?d00001