Added DataForm component for easier edit dialogs.

main
Josef Rokos 2 years ago
parent 0f8166913d
commit a7188e8153

@ -1,5 +1,7 @@
use leptos::*; use leptos::*;
use validator::Validate;
use crate::backend::data::{ApiResponse, Company}; use crate::backend::data::{ApiResponse, Company};
use crate::components::data_form::ForValidation;
#[server(GetCompany, "/api", "Url", "get_company")] #[server(GetCompany, "/api", "Url", "get_company")]
pub async fn get_company() -> Result<ApiResponse<Company>, ServerFnError> { pub async fn get_company() -> Result<ApiResponse<Company>, ServerFnError> {
@ -44,3 +46,9 @@ pub async fn update_company(company: Company) -> Result<ApiResponse<()>, ServerF
Ok(ApiResponse::Data(())) Ok(ApiResponse::Data(()))
} }
impl ForValidation for UpdateCompany {
fn entity(&self) -> &dyn Validate {
&self.company
}
}

@ -1,6 +1,8 @@
use cfg_if::cfg_if; use cfg_if::cfg_if;
use leptos::*; use leptos::*;
use validator::Validate;
use crate::backend::data::{ApiResponse, PwdChange, User, UserProfile}; use crate::backend::data::{ApiResponse, PwdChange, User, UserProfile};
use crate::components::data_form::ForValidation;
cfg_if! { if #[cfg(feature = "ssr")] { cfg_if! { if #[cfg(feature = "ssr")] {
use sqlx::{query_as, Error, PgPool, query}; use sqlx::{query_as, Error, PgPool, query};
@ -110,6 +112,21 @@ pub async fn get_user() -> Result<Option<User>, ServerFnError> {
Ok(logged_in_user().await) Ok(logged_in_user().await)
} }
#[server(GetUsers, "/api", "Url", "get_users")]
pub async fn get_users() -> Result<ApiResponse<Vec<User>>, ServerFnError> {
use crate::backend::AppData;
use actix_web::web::Data;
use leptos_actix::extract;
use crate::perm_check;
perm_check!(is_admin);
let pool = extract(|data: Data<AppData>| async move { data.db_pool().clone() }).await?;
let users = sqlx::query_as::<_, User>(r#"SELECT * FROM "user""#).fetch_all(&pool).await?;
Ok(ApiResponse::Data(users))
}
#[server] #[server]
pub async fn update_profile(user: UserProfile) -> Result<ApiResponse<()>, ServerFnError> { pub async fn update_profile(user: UserProfile) -> Result<ApiResponse<()>, ServerFnError> {
use crate::backend::AppData; use crate::backend::AppData;
@ -136,6 +153,12 @@ pub async fn update_profile(user: UserProfile) -> Result<ApiResponse<()>, Server
Ok(ApiResponse::Data(())) Ok(ApiResponse::Data(()))
} }
impl ForValidation for UpdateProfile {
fn entity(&self) -> &dyn Validate {
&self.user
}
}
#[server] #[server]
pub async fn change_pwd(new_pw: PwdChange) -> Result<ApiResponse<()>, ServerFnError> { pub async fn change_pwd(new_pw: PwdChange) -> Result<ApiResponse<()>, ServerFnError> {
use crate::backend::AppData; use crate::backend::AppData;
@ -163,3 +186,9 @@ pub async fn change_pwd(new_pw: PwdChange) -> Result<ApiResponse<()>, ServerFnEr
Ok(ApiResponse::Data(())) Ok(ApiResponse::Data(()))
} }
impl ForValidation for ChangePwd {
fn entity(&self) -> &dyn Validate {
&self.new_pw
}
}

@ -0,0 +1,52 @@
use crate::backend::data::ApiResponse;
use crate::components::modal_box::{DialogOpener, ModalBody, ModalDialog, ModalFooter};
use leptos::*;
use leptos_router::*;
use validator::Validate;
use crate::components::server_err::ServerErr;
use crate::components::validation_err::ValidationErr;
use crate::locales::trl;
use crate::validator::Validator;
pub trait ForValidation {
fn entity(&self) -> &dyn Validate;
}
#[component]
pub fn data_form<T: 'static + server_fn::ServerFn<()> + Clone + ForValidation>(
opener: DialogOpener,
action: Action<T, Result<ApiResponse<()>, ServerFnError>>,
title: &'static str,
children: Children
) -> impl IntoView {
let upd_val = action.value();
let validator = Validator::new();
view! {
<ActionForm
on:submit=move |ev| {
let act = T::from_event(&ev);
if !act.is_err() {
validator.check(act.unwrap().entity(), &ev);
}
}
action=action>
<ModalDialog opener=opener title=title>
<ModalBody>
<ServerErr result={upd_val} opener=opener/>
<ValidationErr validator=validator />
{children()}
</ModalBody>
<ModalFooter>
<button type="button" class="btn btn-outline-secondary" data-bs-dismiss="modal"
on:click=move |_| {validator.reset(); opener.hide()}>
{trl("Close")}
</button>
<button type="submit" class="btn btn-primary">
{trl("Save changes")}
</button>
</ModalFooter>
</ModalDialog>
</ActionForm>
}
}

@ -4,4 +4,5 @@ pub mod validation_err;
pub mod header; pub mod header;
pub mod admin_portal; pub mod admin_portal;
pub mod user_menu; pub mod user_menu;
pub mod data_form;

@ -1,40 +1,23 @@
use leptos::*; use leptos::*;
use leptos_router::*;
use crate::backend::data::{ApiResponse, User}; use crate::backend::data::{ApiResponse, User};
use crate::backend::user::ChangePwd; use crate::backend::user::ChangePwd;
use crate::components::modal_box::{DialogOpener, ModalBody, ModalDialog, ModalFooter}; use crate::components::data_form::DataForm;
use crate::components::server_err::ServerErr; use crate::components::modal_box::DialogOpener;
use crate::components::validation_err::ValidationErr;
use crate::locales::trl;
use crate::validator::Validator;
#[component] #[component]
pub fn change_password(user: ReadSignal<User>, opener: DialogOpener) -> impl IntoView { pub fn change_password(user: ReadSignal<User>, opener: DialogOpener) -> impl IntoView {
let change_pwd = create_server_action::<ChangePwd>(); let change_pwd = create_server_action::<ChangePwd>();
let upd_val = change_pwd.value();
let validator = Validator::new();
let empty = create_rw_signal("".to_string()); let empty = create_rw_signal("".to_string());
view! { view! {
{move || { {move || {
if let Some(res) = upd_val.get() { if let Some(res) = change_pwd.value().get() {
if let Ok(r) = res { if let Ok(r) = res {
if let ApiResponse::Data(_) = r { empty.update(|e| *e = "".to_string())} if let ApiResponse::Data(_) = r { empty.update(|e| *e = "".to_string())}
} }
} }
view! { view! {
<ActionForm <DataForm opener=opener action=change_pwd title="Change password">
on:submit=move |ev| {
let act = ChangePwd::from_event(&ev);
if !act.is_err() {
validator.check(&act.unwrap().new_pw, &ev);
}
}
action=change_pwd>
<ModalDialog opener=opener title="Change password">
<ModalBody>
<ServerErr result={upd_val} opener=opener/>
<ValidationErr validator=validator />
<input type="hidden" value={move || user.get().login} name="new_pw[login]"/> <input type="hidden" value={move || user.get().login} name="new_pw[login]"/>
<div class="row"> <div class="row">
<div class="col mb-3"> <div class="col mb-3">
@ -72,18 +55,7 @@ pub fn change_password(user: ReadSignal<User>, opener: DialogOpener) -> impl Int
/> />
</div> </div>
</div> </div>
</ModalBody> </DataForm>
<ModalFooter>
<button type="button" class="btn btn-outline-secondary" data-bs-dismiss="modal"
on:click=move |_| {validator.reset(); opener.hide(); empty.update(|e| *e = "".to_string())}>
{trl("Close")}
</button>
<button type="submit" class="btn btn-primary">
{trl("Save changes")}
</button>
</ModalFooter>
</ModalDialog>
</ActionForm>
} }
}} }}
} }

@ -1,14 +1,8 @@
use crate::backend::data::Company; use crate::backend::data::Company;
use crate::backend::company::UpdateCompany; use crate::backend::company::UpdateCompany;
use crate::components::modal_box::{ use crate::components::modal_box::DialogOpener;
DialogOpener, ModalBody, ModalDialog, ModalFooter,
};
use crate::components::server_err::ServerErr;
use crate::locales::trl;
use crate::validator::Validator;
use leptos::*; use leptos::*;
use leptos_router::*; use crate::components::data_form::DataForm;
use crate::components::validation_err::ValidationErr;
#[component] #[component]
pub fn CompanyEdit( pub fn CompanyEdit(
@ -16,95 +10,71 @@ pub fn CompanyEdit(
opener: DialogOpener, opener: DialogOpener,
) -> impl IntoView { ) -> impl IntoView {
let update_company = create_server_action::<UpdateCompany>(); let update_company = create_server_action::<UpdateCompany>();
let upd_val = update_company.value();
let validator = Validator::new();
view! { view! {
<ActionForm <DataForm opener=opener action=update_company title="Edit company">
on:submit=move |ev| { <input type="hidden" value={move || company.get().id()} name="company[id]"/>
let act = UpdateCompany::from_event(&ev); <div class="row">
if !act.is_err() { <div class="col mb-3">
validator.check(&act.unwrap().company, &ev); <label for="nameWithTitle" class="form-label">"Name"</label>
} <input
} type="text"
action=update_company> id="nameWithTitle"
<ModalDialog opener=opener title="Edit company"> class="form-control"
<ModalBody> placeholder="Enter Name"
<ServerErr result={upd_val} opener=opener/> prop:value={move || company.get().name}
<ValidationErr validator=validator /> name="company[name]"
<input type="hidden" value={move || company.get().id()} name="company[id]"/> />
<div class="row">
<div class="col mb-3">
<label for="nameWithTitle" class="form-label">"Name"</label>
<input
type="text"
id="nameWithTitle"
class="form-control"
placeholder="Enter Name"
prop:value={move || company.get().name}
name="company[name]"
/>
</div>
</div> </div>
<div class="row"> </div>
<div class="col mb-3"> <div class="row">
<label for="street" class="form-label">"Street"</label> <div class="col mb-3">
<input <label for="street" class="form-label">"Street"</label>
type="text" <input
id="street" type="text"
class="form-control" id="street"
placeholder="Enter Street" class="form-control"
prop:value={move || company.get().street} placeholder="Enter Street"
name="company[street]" prop:value={move || company.get().street}
/> name="company[street]"
</div> />
<div class="col-4 mb-3">
<label for="houseNumber" class="form-label">"House number"</label>
<input
type="text"
id="houseNumber"
class="form-control"
placeholder="Enter House number"
prop:value={move || company.get().house_number}
name="company[house_number]"
/>
</div>
</div> </div>
<div class="row"> <div class="col-4 mb-3">
<div class="col-4 mb-3"> <label for="houseNumber" class="form-label">"House number"</label>
<label for="zip" class="form-label">"ZIP code"</label> <input
<input type="text"
type="text" id="houseNumber"
id="zip" class="form-control"
class="form-control" placeholder="Enter House number"
placeholder="Enter ZIP code" prop:value={move || company.get().house_number}
prop:value={move || company.get().zip_code} name="company[house_number]"
name="company[zip_code]" />
/>
</div>
<div class="col mb-3">
<label for="city" class="form-label">"City"</label>
<input
type="text"
id="city"
class="form-control"
placeholder="Enter City"
prop:value={move || company.get().city}
name="company[city]"
/>
</div>
</div> </div>
</ModalBody> </div>
<ModalFooter> <div class="row">
<button type="button" class="btn btn-outline-secondary" data-bs-dismiss="modal" <div class="col-4 mb-3">
on:click=move |_| {validator.reset(); opener.hide()}> <label for="zip" class="form-label">"ZIP code"</label>
{trl("Close")} <input
</button> type="text"
<button type="submit" class="btn btn-primary"> id="zip"
{trl("Save changes")} class="form-control"
</button> placeholder="Enter ZIP code"
</ModalFooter> prop:value={move || company.get().zip_code}
</ModalDialog> name="company[zip_code]"
</ActionForm> />
</div>
<div class="col mb-3">
<label for="city" class="form-label">"City"</label>
<input
type="text"
id="city"
class="form-control"
placeholder="Enter City"
prop:value={move || company.get().city}
name="company[city]"
/>
</div>
</div>
</DataForm>
} }
} }

@ -13,7 +13,7 @@ pub fn CompanyInfo() -> impl IntoView {
view! { view! {
<CompanyEdit company={cmp} opener=editor/> <CompanyEdit company={cmp} opener=editor/>
<div class="card"> <div class="card mb-3">
<div class="card-body"> <div class="card-body">
<h5 class="card-title"><i class="bx bx-buildings"></i>" "{trl("Company info")}</h5> <h5 class="card-title"><i class="bx bx-buildings"></i>" "{trl("Company info")}</h5>
<p class="card-text"> <p class="card-text">
@ -41,7 +41,7 @@ pub fn CompanyInfo() -> impl IntoView {
</Transition> </Transition>
</p> </p>
<a href="javascript:void(0)" class="card-link" on:click = move |_| editor.show()> <a href="javascript:void(0)" class="card-link" on:click = move |_| editor.show()>
<i class="bx bx-pencil fs-4 lh-0"></i></a> <i class="bx bx-edit-alt fs-4 lh-0"></i></a>
</div> </div>
</div> </div>
} }

@ -6,3 +6,5 @@ pub mod login;
pub mod public; pub mod public;
pub mod profile_edit; pub mod profile_edit;
pub mod change_pwd; pub mod change_pwd;
pub mod users;
pub mod user_edit;

@ -1,83 +1,55 @@
use leptos::*; use leptos::*;
use leptos_router::*;
use crate::backend::data::User; use crate::backend::data::User;
use crate::backend::user::UpdateProfile; use crate::backend::user::UpdateProfile;
use crate::components::modal_box::{DialogOpener, ModalBody, ModalDialog, ModalFooter}; use crate::components::data_form::DataForm;
use crate::components::server_err::ServerErr; use crate::components::modal_box::DialogOpener;
use crate::components::validation_err::ValidationErr;
use crate::locales::trl;
use crate::validator::Validator;
#[component] #[component]
pub fn ProfileEdit(user: ReadSignal<User>, opener: DialogOpener) -> impl IntoView { pub fn ProfileEdit(user: ReadSignal<User>, opener: DialogOpener) -> impl IntoView {
let update_user = create_server_action::<UpdateProfile>(); let update_user = create_server_action::<UpdateProfile>();
let upd_val = update_user.value();
let validator = Validator::new();
view! { view! {
<ActionForm <DataForm opener=opener action=update_user title="Edit profile">
on:submit=move |ev| { <input type="hidden" value={move || user.get().login} name="user[login]"/>
let act = UpdateProfile::from_event(&ev); <div class="row">
if !act.is_err() { <div class="col mb-3">
validator.check(&act.unwrap().user, &ev); <label for="name" class="form-label">"Full name"</label>
} <input
} type="text"
action=update_user> id="name"
<ModalDialog opener=opener title="Edit profile"> class="form-control"
<ModalBody> placeholder="Enter Full name"
<ServerErr result={upd_val} opener=opener/> prop:value={move || user.get().full_name}
<ValidationErr validator=validator /> name="user[full_name]"
<input type="hidden" value={move || user.get().login} name="user[login]"/> />
<div class="row">
<div class="col mb-3">
<label for="name" class="form-label">"Full name"</label>
<input
type="text"
id="name"
class="form-control"
placeholder="Enter Full name"
prop:value={move || user.get().full_name}
name="user[full_name]"
/>
</div>
</div> </div>
<div class="row"> </div>
<div class="col mb-3"> <div class="row">
<label for="email" class="form-label">"Email"</label> <div class="col mb-3">
<input <label for="email" class="form-label">"Email"</label>
type="text" <input
id="name" type="text"
class="form-control" id="name"
placeholder="Enter email" class="form-control"
prop:value={move || user.get().email.unwrap_or("".to_string())} placeholder="Enter email"
name="user[email]" prop:value={move || user.get().email.unwrap_or("".to_string())}
/> name="user[email]"
</div> />
</div> </div>
<div class="row"> </div>
<div class="col mb-3"> <div class="row">
<input <div class="col mb-3">
class="form-check-input" <input
type="checkbox" class="form-check-input"
id="getMail" type="checkbox"
prop:value={move || if user.get().get_emails {"true"} else {"false"}} id="getMail"
prop:checked={move || user.get().get_emails} prop:value={move || if user.get().get_emails {"true"} else {"false"}}
name="user[get_emails]" prop:checked={move || user.get().get_emails}
/> name="user[get_emails]"
<label class="form-check-label" for="getMail">"Get emails"</label> />
</div> <label class="form-check-label" for="getMail">"Get emails"</label>
</div> </div>
</ModalBody> </div>
<ModalFooter> </DataForm>
<button type="button" class="btn btn-outline-secondary" data-bs-dismiss="modal"
on:click=move |_| {validator.reset(); opener.hide()}>
{trl("Close")}
</button>
<button type="submit" class="btn btn-primary">
{trl("Save changes")}
</button>
</ModalFooter>
</ModalDialog>
</ActionForm>
} }
} }

@ -26,7 +26,7 @@ impl Validator {
} }
} }
pub fn check(&self, entity: &impl Validate, ev: &web_sys::Event) { pub fn check(&self, entity: &(impl Validate + ?Sized), ev: &web_sys::Event) {
if let Err(val_err) = entity.validate() { if let Err(val_err) = entity.validate() {
ev.prevent_default(); ev.prevent_default();
//self.set_message.update(|m| *m = Some(val_err.to_string().clone())); //self.set_message.update(|m| *m = Some(val_err.to_string().clone()));

Loading…
Cancel
Save