2022-12-09 00:15:04 +00:00
|
|
|
use dioxus::prelude::*;
|
2023-05-24 00:09:24 +00:00
|
|
|
use dioxus_router::prelude::*;
|
|
|
|
use std::str::FromStr;
|
2022-12-09 00:15:04 +00:00
|
|
|
|
2023-10-26 19:19:51 +00:00
|
|
|
#[cfg(feature = "liveview")]
|
|
|
|
#[tokio::main]
|
|
|
|
async fn main() {
|
|
|
|
use axum::{extract::ws::WebSocketUpgrade, response::Html, routing::get, Router};
|
|
|
|
|
|
|
|
let listen_address: std::net::SocketAddr = ([127, 0, 0, 1], 3030).into();
|
|
|
|
let view = dioxus_liveview::LiveViewPool::new();
|
|
|
|
let app = Router::new()
|
|
|
|
.fallback(get(move || async move {
|
|
|
|
Html(format!(
|
|
|
|
r#"
|
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head></head>
|
|
|
|
<body><div id="main"></div></body>
|
|
|
|
{glue}
|
|
|
|
</html>
|
|
|
|
"#,
|
|
|
|
glue = dioxus_liveview::interpreter_glue(&format!("ws://{listen_address}/ws"))
|
|
|
|
))
|
|
|
|
}))
|
|
|
|
.route(
|
|
|
|
"/ws",
|
|
|
|
get(move |ws: WebSocketUpgrade| async move {
|
|
|
|
ws.on_upgrade(move |socket| async move {
|
|
|
|
_ = view
|
|
|
|
.launch(dioxus_liveview::axum_socket(socket), Root)
|
|
|
|
.await;
|
|
|
|
})
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
println!("Listening on http://{listen_address}");
|
|
|
|
|
|
|
|
axum::Server::bind(&listen_address.to_string().parse().unwrap())
|
|
|
|
.serve(app.into_make_service())
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "liveview"))]
|
2022-12-09 00:15:04 +00:00
|
|
|
fn main() {
|
2023-05-24 00:09:24 +00:00
|
|
|
#[cfg(not(target_arch = "wasm32"))]
|
2023-09-15 14:13:36 +00:00
|
|
|
dioxus_desktop::launch(Root);
|
2022-12-09 00:15:04 +00:00
|
|
|
|
2023-05-24 00:09:24 +00:00
|
|
|
#[cfg(target_arch = "wasm32")]
|
|
|
|
dioxus_web::launch(root);
|
|
|
|
}
|
2022-12-15 19:09:53 +00:00
|
|
|
|
2023-10-26 19:19:51 +00:00
|
|
|
#[cfg(feature = "liveview")]
|
2023-09-15 14:13:36 +00:00
|
|
|
#[component]
|
|
|
|
fn Root(cx: Scope) -> Element {
|
2023-10-26 19:19:51 +00:00
|
|
|
let history = LiveviewHistory::new(cx);
|
|
|
|
render! { Router::<Route> {
|
|
|
|
config: || RouterConfig::default().history(history),
|
|
|
|
} }
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "liveview"))]
|
|
|
|
#[component]
|
|
|
|
fn Root(cx: Scope) -> Element {
|
|
|
|
render! { Router::<Route> {} }
|
2023-05-24 00:09:24 +00:00
|
|
|
}
|
2022-12-09 00:15:04 +00:00
|
|
|
|
2023-09-15 14:13:36 +00:00
|
|
|
#[component]
|
2023-05-24 00:09:24 +00:00
|
|
|
fn UserFrame(cx: Scope, user_id: usize) -> Element {
|
2022-12-09 00:15:04 +00:00
|
|
|
render! {
|
2023-10-26 19:19:51 +00:00
|
|
|
pre { "UserFrame{{\n\tuser_id:{user_id}\n}}" }
|
|
|
|
div { background_color: "rgba(0,0,0,50%)",
|
2023-05-24 00:09:24 +00:00
|
|
|
"children:"
|
2023-07-26 01:14:48 +00:00
|
|
|
Outlet::<Route> {}
|
2022-12-09 03:11:31 +00:00
|
|
|
}
|
2022-12-09 00:15:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-15 14:13:36 +00:00
|
|
|
#[component]
|
2023-05-31 17:11:11 +00:00
|
|
|
fn Route1(cx: Scope, user_id: usize, dynamic: usize, query: String, extra: String) -> Element {
|
2022-12-09 00:15:04 +00:00
|
|
|
render! {
|
2023-05-24 00:09:24 +00:00
|
|
|
pre {
|
2023-05-31 17:11:11 +00:00
|
|
|
"Route1{{\n\tuser_id:{user_id},\n\tdynamic:{dynamic},\n\tquery:{query},\n\textra:{extra}\n}}"
|
2023-05-24 00:09:24 +00:00
|
|
|
}
|
2023-05-24 00:31:30 +00:00
|
|
|
Link {
|
2023-07-24 18:57:56 +00:00
|
|
|
to: Route::Route1 { user_id: *user_id, dynamic: *dynamic, query: String::new(), extra: extra.clone() + "." },
|
2023-05-24 00:09:24 +00:00
|
|
|
"Route1 with extra+\".\""
|
|
|
|
}
|
|
|
|
p { "Footer" }
|
2023-05-24 00:31:30 +00:00
|
|
|
Link {
|
2023-07-24 18:57:56 +00:00
|
|
|
to: Route::Route3 { dynamic: String::new() },
|
2023-05-24 00:09:24 +00:00
|
|
|
"Home"
|
2022-12-09 03:11:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-15 14:13:36 +00:00
|
|
|
#[component]
|
2023-05-24 00:09:24 +00:00
|
|
|
fn Route2(cx: Scope, user_id: usize) -> Element {
|
2022-12-09 03:11:31 +00:00
|
|
|
render! {
|
2023-05-24 00:09:24 +00:00
|
|
|
pre {
|
|
|
|
"Route2{{\n\tuser_id:{user_id}\n}}"
|
|
|
|
}
|
|
|
|
(0..*user_id).map(|i| rsx!{ p { "{i}" } }),
|
|
|
|
p { "Footer" }
|
2023-05-24 00:31:30 +00:00
|
|
|
Link {
|
2023-07-24 18:57:56 +00:00
|
|
|
to: Route::Route3 { dynamic: String::new() },
|
2023-05-24 00:09:24 +00:00
|
|
|
"Home"
|
2022-12-09 03:11:31 +00:00
|
|
|
}
|
2022-12-09 00:15:04 +00:00
|
|
|
}
|
|
|
|
}
|
2022-12-14 17:07:04 +00:00
|
|
|
|
2023-09-15 14:13:36 +00:00
|
|
|
#[component]
|
2023-05-24 00:09:24 +00:00
|
|
|
fn Route3(cx: Scope, dynamic: String) -> Element {
|
2023-06-01 14:45:31 +00:00
|
|
|
let navigator = use_navigator(cx);
|
|
|
|
let current_route = use_route(cx)?;
|
|
|
|
let current_route_str = use_ref(cx, String::new);
|
|
|
|
let parsed = Route::from_str(¤t_route_str.read());
|
2023-05-24 00:09:24 +00:00
|
|
|
|
|
|
|
let site_map = Route::SITE_MAP
|
|
|
|
.iter()
|
|
|
|
.flat_map(|seg| seg.flatten().into_iter())
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
2022-12-14 17:07:04 +00:00
|
|
|
render! {
|
2023-05-24 00:09:24 +00:00
|
|
|
input {
|
|
|
|
oninput: move |evt| {
|
2023-06-01 14:45:31 +00:00
|
|
|
*current_route_str.write() = evt.value.clone();
|
2023-05-24 00:09:24 +00:00
|
|
|
},
|
2023-06-01 14:45:31 +00:00
|
|
|
value: "{current_route_str.read()}"
|
2023-05-24 00:09:24 +00:00
|
|
|
}
|
|
|
|
"dynamic: {dynamic}"
|
2023-05-24 00:31:30 +00:00
|
|
|
Link {
|
2023-07-24 18:57:56 +00:00
|
|
|
to: Route::Route2 { user_id: 8888 },
|
2023-05-24 00:09:24 +00:00
|
|
|
"hello world link"
|
|
|
|
}
|
2023-10-26 19:19:51 +00:00
|
|
|
button {
|
|
|
|
disabled: !navigator.can_go_back(),
|
|
|
|
onclick: move |_| { navigator.go_back(); },
|
|
|
|
"go back"
|
|
|
|
}
|
|
|
|
button {
|
|
|
|
disabled: !navigator.can_go_forward(),
|
|
|
|
onclick: move |_| { navigator.go_forward(); },
|
|
|
|
"go forward"
|
|
|
|
}
|
2023-05-31 22:12:52 +00:00
|
|
|
button {
|
2023-07-26 18:08:45 +00:00
|
|
|
onclick: move |_| { navigator.push("https://www.google.com"); },
|
2023-05-31 22:12:52 +00:00
|
|
|
"google link"
|
|
|
|
}
|
2023-05-24 00:09:24 +00:00
|
|
|
p { "Site Map" }
|
|
|
|
pre { "{site_map:#?}" }
|
|
|
|
p { "Dynamic link" }
|
2023-07-14 18:40:38 +00:00
|
|
|
match parsed {
|
|
|
|
Ok(route) => {
|
|
|
|
if route != current_route {
|
|
|
|
render! {
|
|
|
|
Link {
|
2023-07-24 18:57:56 +00:00
|
|
|
to: route.clone(),
|
2023-07-14 18:40:38 +00:00
|
|
|
"{route}"
|
|
|
|
}
|
2023-05-24 00:09:24 +00:00
|
|
|
}
|
|
|
|
}
|
2023-07-14 18:40:38 +00:00
|
|
|
else {
|
|
|
|
None
|
|
|
|
}
|
2023-05-24 00:09:24 +00:00
|
|
|
}
|
2023-07-14 18:40:38 +00:00
|
|
|
Err(err) => {
|
|
|
|
render! {
|
|
|
|
pre {
|
|
|
|
color: "red",
|
|
|
|
"Invalid route:\n{err}"
|
|
|
|
}
|
|
|
|
}
|
2023-05-24 00:39:18 +00:00
|
|
|
}
|
|
|
|
}
|
2022-12-14 17:07:04 +00:00
|
|
|
}
|
|
|
|
}
|
2023-05-24 00:09:24 +00:00
|
|
|
|
|
|
|
#[rustfmt::skip]
|
2023-06-01 16:03:44 +00:00
|
|
|
#[derive(Clone, Debug, PartialEq, Routable)]
|
2023-05-24 00:09:24 +00:00
|
|
|
enum Route {
|
2023-07-14 18:40:38 +00:00
|
|
|
#[nest("/test")]
|
|
|
|
// Nests with parameters have types taken from child routes
|
|
|
|
#[nest("/user/:user_id")]
|
|
|
|
// Everything inside the nest has the added parameter `user_id: usize`
|
|
|
|
// UserFrame is a layout component that will receive the `user_id: usize` parameter
|
|
|
|
#[layout(UserFrame)]
|
|
|
|
#[route("/:dynamic?:query")]
|
|
|
|
Route1 {
|
|
|
|
// The type is taken from the first instance of the dynamic parameter
|
|
|
|
user_id: usize,
|
|
|
|
dynamic: usize,
|
|
|
|
query: String,
|
|
|
|
extra: String,
|
|
|
|
},
|
|
|
|
#[route("/hello_world")]
|
|
|
|
// You can opt out of the layout by using the `!` prefix
|
|
|
|
#[layout(!UserFrame)]
|
|
|
|
Route2 { user_id: usize },
|
|
|
|
#[end_layout]
|
|
|
|
#[end_nest]
|
2023-05-24 00:09:24 +00:00
|
|
|
#[end_nest]
|
2023-05-31 17:11:11 +00:00
|
|
|
#[redirect("/:id/user", |id: usize| Route::Route3 { dynamic: id.to_string()})]
|
2023-05-31 18:57:33 +00:00
|
|
|
#[route("/:dynamic")]
|
2023-05-24 00:09:24 +00:00
|
|
|
Route3 { dynamic: String },
|
|
|
|
}
|