|
@@ -1,24 +1,5 @@
|
|
-use scroll::ScrollPosition;
|
|
|
|
-use wasm_bindgen::JsCast;
|
|
|
|
-use wasm_bindgen::{prelude::Closure, JsValue};
|
|
|
|
-use web_sys::{window, Window};
|
|
|
|
-use web_sys::{Event, History, ScrollRestoration};
|
|
|
|
-
|
|
|
|
-mod scroll;
|
|
|
|
-
|
|
|
|
-fn base_path() -> Option<String> {
|
|
|
|
- let base_path = dioxus_cli_config::web_base_path();
|
|
|
|
- tracing::trace!("Using base_path from the CLI: {:?}", base_path);
|
|
|
|
- base_path
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-#[allow(clippy::extra_unused_type_parameters)]
|
|
|
|
-fn update_scroll(window: &Window, history: &History) {
|
|
|
|
- let scroll = ScrollPosition::of_window(window);
|
|
|
|
- if let Err(err) = replace_state_with_url(history, &[scroll.x, scroll.y], None) {
|
|
|
|
- web_sys::console::error_1(&err);
|
|
|
|
- }
|
|
|
|
-}
|
|
|
|
|
|
+use wasm_bindgen::{prelude::Closure, JsCast, JsValue};
|
|
|
|
+use web_sys::{window, Event, History, ScrollRestoration, Window};
|
|
|
|
|
|
/// A [`dioxus_history::History`] provider that integrates with a browser via the [History API](https://developer.mozilla.org/en-US/docs/Web/API/History_API).
|
|
/// A [`dioxus_history::History`] provider that integrates with a browser via the [History API](https://developer.mozilla.org/en-US/docs/Web/API/History_API).
|
|
///
|
|
///
|
|
@@ -75,7 +56,7 @@ impl WebHistory {
|
|
|
|
|
|
let prefix = prefix
|
|
let prefix = prefix
|
|
// If there isn't a base path, try to grab one from the CLI
|
|
// If there isn't a base path, try to grab one from the CLI
|
|
- .or_else(base_path)
|
|
|
|
|
|
+ .or_else(dioxus_cli_config::web_base_path)
|
|
// Normalize the prefix to start and end with no slashes
|
|
// Normalize the prefix to start and end with no slashes
|
|
.as_ref()
|
|
.as_ref()
|
|
.map(|prefix| prefix.trim_matches('/'))
|
|
.map(|prefix| prefix.trim_matches('/'))
|
|
@@ -102,9 +83,13 @@ impl WebHistory {
|
|
let scroll = self.scroll_pos();
|
|
let scroll = self.scroll_pos();
|
|
[scroll.x, scroll.y]
|
|
[scroll.x, scroll.y]
|
|
}
|
|
}
|
|
-}
|
|
|
|
|
|
|
|
-impl WebHistory {
|
|
|
|
|
|
+ fn handle_nav(&self) {
|
|
|
|
+ if self.do_scroll_restoration {
|
|
|
|
+ self.window.scroll_to_with_x_and_y(0.0, 0.0)
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
fn route_from_location(&self) -> String {
|
|
fn route_from_location(&self) -> String {
|
|
let location = self.window.location();
|
|
let location = self.window.location();
|
|
let path = location.pathname().unwrap_or_else(|_| "/".into())
|
|
let path = location.pathname().unwrap_or_else(|_| "/".into())
|
|
@@ -127,55 +112,177 @@ impl WebHistory {
|
|
Some(prefix) => format!("{prefix}{state}"),
|
|
Some(prefix) => format!("{prefix}{state}"),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl dioxus_history::History for WebHistory {
|
|
|
|
+ fn current_route(&self) -> String {
|
|
|
|
+ self.route_from_location()
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn current_prefix(&self) -> Option<String> {
|
|
|
|
+ self.prefix.clone()
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn go_back(&self) {
|
|
|
|
+ let _ = self.history.back();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn go_forward(&self) {
|
|
|
|
+ let _ = self.history.forward();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn push(&self, state: String) {
|
|
|
|
+ if state == self.current_route() {
|
|
|
|
+ // don't push the same state twice
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ let w = window().expect("access to `window`");
|
|
|
|
+ let h = w.history().expect("`window` has access to `history`");
|
|
|
|
+
|
|
|
|
+ // update the scroll position before pushing the new state
|
|
|
|
+ update_scroll(&w, &h);
|
|
|
|
+
|
|
|
|
+ if push_state_and_url(&self.history, &self.create_state(), self.full_path(&state)).is_ok() {
|
|
|
|
+ self.handle_nav();
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
|
|
- fn handle_nav(&self, result: Result<(), JsValue>) {
|
|
|
|
- match result {
|
|
|
|
- Ok(_) => {
|
|
|
|
- if self.do_scroll_restoration {
|
|
|
|
- self.window.scroll_to_with_x_and_y(0.0, 0.0)
|
|
|
|
|
|
+ fn replace(&self, state: String) {
|
|
|
|
+ if replace_state_with_url(
|
|
|
|
+ &self.history,
|
|
|
|
+ &self.create_state(),
|
|
|
|
+ Some(&self.full_path(&state)),
|
|
|
|
+ )
|
|
|
|
+ .is_ok()
|
|
|
|
+ {
|
|
|
|
+ self.handle_nav();
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn external(&self, url: String) -> bool {
|
|
|
|
+ self.window.location().set_href(&url).is_ok()
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn updater(&self, callback: std::sync::Arc<dyn Fn() + Send + Sync>) {
|
|
|
|
+ let w = self.window.clone();
|
|
|
|
+ let h = self.history.clone();
|
|
|
|
+ let d = self.do_scroll_restoration;
|
|
|
|
+
|
|
|
|
+ let function = Closure::wrap(Box::new(move |_| {
|
|
|
|
+ (*callback)();
|
|
|
|
+ if d {
|
|
|
|
+ if let Some([x, y]) = get_current(&h) {
|
|
|
|
+ ScrollPosition { x, y }.scroll_to(w.clone())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- Err(e) => {
|
|
|
|
- web_sys::console::error_2(&JsValue::from_str("failed to change state: "), &e);
|
|
|
|
- }
|
|
|
|
|
|
+ }) as Box<dyn FnMut(Event)>);
|
|
|
|
+ self.window
|
|
|
|
+ .add_event_listener_with_callback(
|
|
|
|
+ "popstate",
|
|
|
|
+ &function.into_js_value().unchecked_into(),
|
|
|
|
+ )
|
|
|
|
+ .unwrap();
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/// A [`dioxus_history::History`] provider that integrates with a browser via the [History API](https://developer.mozilla.org/en-US/docs/Web/API/History_API)
|
|
|
|
+/// but uses the url fragment for the route. This allows serving as a single html file or on a single url path.
|
|
|
|
+pub struct HashHistory {
|
|
|
|
+ do_scroll_restoration: bool,
|
|
|
|
+ history: History,
|
|
|
|
+ pathname: String,
|
|
|
|
+ window: Window,
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl Default for HashHistory {
|
|
|
|
+ fn default() -> Self {
|
|
|
|
+ Self::new(true)
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl HashHistory {
|
|
|
|
+ /// Create a new [`HashHistory`].
|
|
|
|
+ ///
|
|
|
|
+ /// If `do_scroll_restoration` is [`true`], [`HashHistory`] will take control of the history
|
|
|
|
+ /// state. It'll also set the browsers scroll restoration to `manual`.
|
|
|
|
+ pub fn new(do_scroll_restoration: bool) -> Self {
|
|
|
|
+ let myself = Self::new_inner(do_scroll_restoration);
|
|
|
|
+
|
|
|
|
+ let current_route = dioxus_history::History::current_route(&myself);
|
|
|
|
+ let current_route_str = current_route.to_string();
|
|
|
|
+ let pathname_str = &myself.pathname;
|
|
|
|
+ let current_url = format!("{pathname_str}#{current_route_str}");
|
|
|
|
+ let state = myself.create_state();
|
|
|
|
+ let _ = replace_state_with_url(&myself.history, &state, Some(¤t_url));
|
|
|
|
+
|
|
|
|
+ myself
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn new_inner(do_scroll_restoration: bool) -> Self {
|
|
|
|
+ let window = window().expect("access to `window`");
|
|
|
|
+ let history = window.history().expect("`window` has access to `history`");
|
|
|
|
+ let pathname = window.location().pathname().unwrap();
|
|
|
|
+
|
|
|
|
+ if do_scroll_restoration {
|
|
|
|
+ history
|
|
|
|
+ .set_scroll_restoration(ScrollRestoration::Manual)
|
|
|
|
+ .expect("`history` can set scroll restoration");
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Self {
|
|
|
|
+ do_scroll_restoration,
|
|
|
|
+ history,
|
|
|
|
+ pathname,
|
|
|
|
+ window,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- fn navigate_external(&self, url: String) -> bool {
|
|
|
|
- match self.window.location().set_href(&url) {
|
|
|
|
- Ok(_) => true,
|
|
|
|
- Err(e) => {
|
|
|
|
- web_sys::console::error_4(
|
|
|
|
- &JsValue::from_str("failed to navigate to external url ("),
|
|
|
|
- &JsValue::from_str(&url),
|
|
|
|
- &JsValue::from_str("): "),
|
|
|
|
- &e,
|
|
|
|
- );
|
|
|
|
- false
|
|
|
|
- }
|
|
|
|
|
|
+ fn scroll_pos(&self) -> ScrollPosition {
|
|
|
|
+ self.do_scroll_restoration
|
|
|
|
+ .then(|| ScrollPosition::of_window(&self.window))
|
|
|
|
+ .unwrap_or_default()
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn create_state(&self) -> [f64; 2] {
|
|
|
|
+ let scroll = self.scroll_pos();
|
|
|
|
+ [scroll.x, scroll.y]
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn full_path(&self, state: &String) -> String {
|
|
|
|
+ format!("{}#{state}", self.pathname)
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn handle_nav(&self) {
|
|
|
|
+ if self.do_scroll_restoration {
|
|
|
|
+ self.window.scroll_to_with_x_and_y(0.0, 0.0)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-impl dioxus_history::History for WebHistory {
|
|
|
|
|
|
+impl dioxus_history::History for HashHistory {
|
|
fn current_route(&self) -> String {
|
|
fn current_route(&self) -> String {
|
|
- self.route_from_location()
|
|
|
|
|
|
+ let location = self.window.location();
|
|
|
|
+
|
|
|
|
+ let hash = location.hash().unwrap();
|
|
|
|
+ if hash.is_empty() {
|
|
|
|
+ // If the path is empty, parse the root route instead
|
|
|
|
+ "/".to_owned()
|
|
|
|
+ } else {
|
|
|
|
+ hash.trim_start_matches("#").to_owned()
|
|
|
|
+ }
|
|
}
|
|
}
|
|
|
|
|
|
fn current_prefix(&self) -> Option<String> {
|
|
fn current_prefix(&self) -> Option<String> {
|
|
- self.prefix.clone()
|
|
|
|
|
|
+ Some(format!("{}#", self.pathname))
|
|
}
|
|
}
|
|
|
|
|
|
fn go_back(&self) {
|
|
fn go_back(&self) {
|
|
- if let Err(e) = self.history.back() {
|
|
|
|
- web_sys::console::error_2(&JsValue::from_str("failed to go back: "), &e);
|
|
|
|
- }
|
|
|
|
|
|
+ let _ = self.history.back();
|
|
}
|
|
}
|
|
|
|
|
|
fn go_forward(&self) {
|
|
fn go_forward(&self) {
|
|
- if let Err(e) = self.history.forward() {
|
|
|
|
- web_sys::console::error_2(&JsValue::from_str("failed to go forward: "), &e);
|
|
|
|
- }
|
|
|
|
|
|
+ let _ = self.history.forward();
|
|
}
|
|
}
|
|
|
|
|
|
fn push(&self, state: String) {
|
|
fn push(&self, state: String) {
|
|
@@ -190,21 +297,25 @@ impl dioxus_history::History for WebHistory {
|
|
// update the scroll position before pushing the new state
|
|
// update the scroll position before pushing the new state
|
|
update_scroll(&w, &h);
|
|
update_scroll(&w, &h);
|
|
|
|
|
|
- let path = self.full_path(&state);
|
|
|
|
-
|
|
|
|
- let state: [f64; 2] = self.create_state();
|
|
|
|
- self.handle_nav(push_state_and_url(&self.history, &state, path));
|
|
|
|
|
|
+ if push_state_and_url(&self.history, &self.create_state(), self.full_path(&state)).is_ok() {
|
|
|
|
+ self.handle_nav();
|
|
|
|
+ }
|
|
}
|
|
}
|
|
|
|
|
|
fn replace(&self, state: String) {
|
|
fn replace(&self, state: String) {
|
|
- let path = self.full_path(&state);
|
|
|
|
-
|
|
|
|
- let state = self.create_state();
|
|
|
|
- self.handle_nav(replace_state_with_url(&self.history, &state, Some(&path)));
|
|
|
|
|
|
+ if replace_state_with_url(
|
|
|
|
+ &self.history,
|
|
|
|
+ &self.create_state(),
|
|
|
|
+ Some(&self.full_path(&state)),
|
|
|
|
+ )
|
|
|
|
+ .is_ok()
|
|
|
|
+ {
|
|
|
|
+ self.handle_nav();
|
|
|
|
+ }
|
|
}
|
|
}
|
|
|
|
|
|
fn external(&self, url: String) -> bool {
|
|
fn external(&self, url: String) -> bool {
|
|
- self.navigate_external(url)
|
|
|
|
|
|
+ self.window.location().set_href(&url).is_ok()
|
|
}
|
|
}
|
|
|
|
|
|
fn updater(&self, callback: std::sync::Arc<dyn Fn() + Send + Sync>) {
|
|
fn updater(&self, callback: std::sync::Arc<dyn Fn() + Send + Sync>) {
|
|
@@ -229,6 +340,32 @@ impl dioxus_history::History for WebHistory {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+#[derive(Clone, Copy, Debug, Default)]
|
|
|
|
+pub(crate) struct ScrollPosition {
|
|
|
|
+ pub x: f64,
|
|
|
|
+ pub y: f64,
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl ScrollPosition {
|
|
|
|
+ pub(crate) fn of_window(window: &Window) -> Self {
|
|
|
|
+ Self {
|
|
|
|
+ x: window.scroll_x().unwrap_or_default(),
|
|
|
|
+ y: window.scroll_y().unwrap_or_default(),
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ pub(crate) fn scroll_to(&self, window: Window) {
|
|
|
|
+ let Self { x, y } = *self;
|
|
|
|
+ let f = Closure::wrap(
|
|
|
|
+ Box::new(move || window.scroll_to_with_x_and_y(x, y)) as Box<dyn FnMut()>
|
|
|
|
+ );
|
|
|
|
+ web_sys::window()
|
|
|
|
+ .expect("should be run in a context with a `Window` object (dioxus cannot be run from a web worker)")
|
|
|
|
+ .request_animation_frame(&f.into_js_value().unchecked_into())
|
|
|
|
+ .expect("should register `requestAnimationFrame` OK");
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
pub(crate) fn replace_state_with_url(
|
|
pub(crate) fn replace_state_with_url(
|
|
history: &History,
|
|
history: &History,
|
|
value: &[f64; 2],
|
|
value: &[f64; 2],
|
|
@@ -237,7 +374,6 @@ pub(crate) fn replace_state_with_url(
|
|
let position = js_sys::Array::new();
|
|
let position = js_sys::Array::new();
|
|
position.push(&JsValue::from(value[0]));
|
|
position.push(&JsValue::from(value[0]));
|
|
position.push(&JsValue::from(value[1]));
|
|
position.push(&JsValue::from(value[1]));
|
|
-
|
|
|
|
history.replace_state_with_url(&position, "", url)
|
|
history.replace_state_with_url(&position, "", url)
|
|
}
|
|
}
|
|
|
|
|
|
@@ -249,21 +385,20 @@ pub(crate) fn push_state_and_url(
|
|
let position = js_sys::Array::new();
|
|
let position = js_sys::Array::new();
|
|
position.push(&JsValue::from(value[0]));
|
|
position.push(&JsValue::from(value[0]));
|
|
position.push(&JsValue::from(value[1]));
|
|
position.push(&JsValue::from(value[1]));
|
|
-
|
|
|
|
history.push_state_with_url(&position, "", Some(&url))
|
|
history.push_state_with_url(&position, "", Some(&url))
|
|
}
|
|
}
|
|
|
|
|
|
pub(crate) fn get_current(history: &History) -> Option<[f64; 2]> {
|
|
pub(crate) fn get_current(history: &History) -> Option<[f64; 2]> {
|
|
use wasm_bindgen::JsCast;
|
|
use wasm_bindgen::JsCast;
|
|
-
|
|
|
|
- let state = history.state();
|
|
|
|
- if let Err(err) = &state {
|
|
|
|
- web_sys::console::error_1(err);
|
|
|
|
- }
|
|
|
|
- state.ok().and_then(|state| {
|
|
|
|
|
|
+ history.state().ok().and_then(|state| {
|
|
let state = state.dyn_into::<js_sys::Array>().ok()?;
|
|
let state = state.dyn_into::<js_sys::Array>().ok()?;
|
|
let x = state.get(0).as_f64()?;
|
|
let x = state.get(0).as_f64()?;
|
|
let y = state.get(1).as_f64()?;
|
|
let y = state.get(1).as_f64()?;
|
|
Some([x, y])
|
|
Some([x, y])
|
|
})
|
|
})
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+fn update_scroll(window: &Window, history: &History) {
|
|
|
|
+ let scroll = ScrollPosition::of_window(window);
|
|
|
|
+ let _ = replace_state_with_url(history, &[scroll.x, scroll.y], None);
|
|
|
|
+}
|