from fasthtml.common import * from backup import upload, download db = database("data/utodos.db") todos,users = db.t.todos,db.t.users if todos not in db.t: users.create(name=str, pwd=str, pk='name') todos.create(id=int, title=str, done=bool, name=str, pk='id') Todo,User = todos.dataclass(),users.dataclass() id_curr = 'current-todo' def tid(id): return f'todo-{id}' def lookup_user(u,p): try: user = users[u] except NotFoundError: user = users.insert(name=u, pwd=p) return user.pwd==p css = Style(':root { --pico-font-size: 100%; }') authmw = user_pwd_auth(lookup_user, skip=[r'/favicon\.ico', r'/static/.*', r'.*\.css']) def before(auth): todos.xtra(name=auth) app = FastHTML(hdrs=(picolink, css), middleware=authmw, before=before) rt = app.route @rt("/{fname:path}.{ext:static}") async def get(fname:str, ext:str): return FileResponse(f'{fname}.{ext}') @patch def __xt__(self:Todo): show = AX(self.title, f'/todos/{self.id}', id_curr) edit = AX('edit', f'/edit/{self.id}' , id_curr) dt = ' (done)' if self.done else '' return Li(show, dt, ' | ', edit, id=tid(self.id)) def mk_input(**kw): return Input(id="new-title", name="title", placeholder="New Todo", **kw) def clr_details(): return Div(hx_swap_oob='innerHTML', id=id_curr) @rt("/") async def get(request, auth): add = Form(Group(mk_input(), Button("Add")), hx_post="/", target_id='todo-list', hx_swap="beforeend") card = Card(Ul(*todos(), id='todo-list'), header=add, footer=Div(id=id_curr)), title = 'Todo list' top = Grid(H1(f"{auth}'s {title}"), Div(A('logout', href=basic_logout(request), target="_blank"), style='text-align: right')) return Title(title), Main(top, card, cls='container') @rt("/todos/{id}") async def delete(id:int): todos.delete(id) return clr_details() @rt("/") async def post(todo:Todo): return todos.insert(todo), mk_input(hx_swap_oob='true') @rt("/edit/{id}") async def get(id:int): res = Form(Group(Input(id="title"), Button("Save")), Hidden(id="id"), Checkbox(id="done", label='Done'), hx_put="/", target_id=tid(id), id="edit") return fill_form(res, todos[id]) @rt("/") async def put(todo: Todo): return todos.upsert(todo), clr_details() @rt("/todos/{id}") async def get(id:int): todo = todos[id] btn = Button('delete', hx_delete=f'/todos/{todo.id}', target_id=tid(todo.id), hx_swap="outerHTML") return Div(Div(todo.title), btn) app.on_event("startup")(download) app.on_event("shutdown")(upload) run_uv()