leptos_use/use_window_size.rs
1use crate::core::Size;
2use crate::{
3 use_event_listener_with_options, use_media_query, use_window, UseEventListenerOptions,
4};
5use default_struct_builder::DefaultBuilder;
6use leptos::ev::resize;
7use leptos::prelude::*;
8
9/// Reactive window size.
10///
11/// ## Demo
12///
13/// [Link to Demo](https://github.com/Synphonyte/leptos-use/tree/main/examples/use_window_size)
14///
15/// ## Usage
16///
17/// ```
18/// # use leptos::*;
19/// # use leptos_use::{use_window_size, UseWindowSizeReturn};
20/// #
21/// # #[component]
22/// # fn Demo() -> impl IntoView {
23/// let UseWindowSizeReturn { width, height } = use_window_size();
24/// #
25/// # view! { }
26/// # }
27/// ```
28///
29/// ## Server-Side Rendering
30///
31/// On the server the width and height are always `initial_size` which defaults to
32/// `Size { width: INFINITY, height: INFINITY }`.
33// #[doc(cfg(feature = "use_window_size"))]
34pub fn use_window_size() -> UseWindowSizeReturn {
35 use_window_size_with_options(UseWindowSizeOptions::default())
36}
37
38/// Version of [`fn@crate::use_window_size`] that takes a `UseWindowSizeOptions`. See [`fn@crate::use_window_size`] for how to use.
39// #[doc(cfg(feature = "use_window_size"))]
40pub fn use_window_size_with_options(options: UseWindowSizeOptions) -> UseWindowSizeReturn {
41 let UseWindowSizeOptions {
42 initial_size,
43 listen_orientation,
44 include_scrollbar,
45 measure_type,
46 } = options;
47
48 let (width, set_width) = signal(initial_size.width);
49 let (height, set_height) = signal(initial_size.height);
50
51 let update;
52
53 #[cfg(not(feature = "ssr"))]
54 {
55 update = move || match measure_type {
56 MeasureType::Outer => {
57 set_width.set(
58 window()
59 .outer_width()
60 .expect("failed to get window width")
61 .as_f64()
62 .expect("width is not a f64"),
63 );
64 set_height.set(
65 window()
66 .outer_height()
67 .expect("failed to get window height")
68 .as_f64()
69 .expect("height is not a f64"),
70 );
71 }
72 MeasureType::Inner => {
73 if include_scrollbar {
74 set_width.set(
75 window()
76 .inner_width()
77 .expect("failed to get window width")
78 .as_f64()
79 .expect("width is not a f64"),
80 );
81 set_height.set(
82 window()
83 .inner_height()
84 .expect("failed to get window height")
85 .as_f64()
86 .expect("height is not a f64"),
87 );
88 } else {
89 set_width.set(
90 document()
91 .document_element()
92 .expect("no document element")
93 .client_width() as f64,
94 );
95 set_height.set(
96 document()
97 .document_element()
98 .expect("no document element")
99 .client_height() as f64,
100 );
101 }
102 }
103 };
104 }
105
106 #[cfg(feature = "ssr")]
107 {
108 update = || {};
109
110 let _ = initial_size;
111 let _ = include_scrollbar;
112 let _ = measure_type;
113
114 let _ = set_width;
115 let _ = set_height;
116 }
117
118 update();
119 let _ = use_event_listener_with_options(
120 use_window(),
121 resize,
122 move |_| update(),
123 UseEventListenerOptions::default().passive(true),
124 );
125
126 if listen_orientation {
127 let matches = use_media_query("(orientation: portrait)");
128
129 Effect::new(move |_| {
130 let _ = matches.get();
131
132 update();
133 });
134 }
135
136 UseWindowSizeReturn {
137 width: width.into(),
138 height: height.into(),
139 }
140}
141
142/// Options for [`fn@crate::use_window_size_with_options`].
143// #[doc(cfg(feature = "use_window_size"))]
144#[derive(DefaultBuilder)]
145pub struct UseWindowSizeOptions {
146 /// The initial size before anything is measured (like on the server side).
147 /// Defaults to `Size { width: INFINITY, height: INFINITY }`.
148 initial_size: Size,
149
150 /// Listen to the window ` orientationchange ` event. Defaults to `true`.
151 listen_orientation: bool,
152
153 /// Whether the scrollbar should be included in the width and height
154 /// Only effective when `measure_type` is `MeasureType::Inner`.
155 /// Defaults to `true`.
156 include_scrollbar: bool,
157
158 /// Use `window.innerWidth` or `window.outerWidth`.
159 /// Defaults to `MeasureType::Inner`.
160 measure_type: MeasureType,
161}
162
163/// Type of the `measure_type` option.
164#[derive(Default, Clone, Copy, PartialEq, Eq, Debug)]
165pub enum MeasureType {
166 /// Use `window.innerWidth`
167 #[default]
168 Inner,
169 /// Use `window.outerWidth`
170 Outer,
171}
172
173impl Default for UseWindowSizeOptions {
174 fn default() -> Self {
175 Self {
176 initial_size: Size {
177 width: f64::INFINITY,
178 height: f64::INFINITY,
179 },
180 listen_orientation: true,
181 include_scrollbar: true,
182 measure_type: MeasureType::default(),
183 }
184 }
185}
186
187/// Return type of [`fn@crate::use_window_size`].
188// #[doc(cfg(feature = "use_window_size"))]
189pub struct UseWindowSizeReturn {
190 /// The width of the window.
191 pub width: Signal<f64>,
192 /// The height of the window.
193 pub height: Signal<f64>,
194}