1use crate::{use_media_query, use_window};
2use leptos::logging::error;
3use leptos::prelude::*;
4use leptos::reactive::wrappers::read::Signal;
5use paste::paste;
6use std::collections::HashMap;
7use std::fmt::Debug;
8use std::hash::Hash;
9
10pub fn use_breakpoints<K: Eq + Hash + Debug + Clone + Send + Sync>(
115 breakpoints: HashMap<K, u32>,
116) -> UseBreakpointsReturn<K> {
117 UseBreakpointsReturn { breakpoints }
118}
119
120#[derive(Clone)]
122pub struct UseBreakpointsReturn<K: Eq + Hash + Debug + Clone + Send + Sync> {
123 breakpoints: HashMap<K, u32>,
124}
125
126macro_rules! query_suffix {
127 (>) => {
128 ".1"
129 };
130 (<) => {
131 ".9"
132 };
133 (=) => {
134 ""
135 };
136}
137
138macro_rules! value_expr {
139 ($v:ident, >) => {
140 $v
141 };
142 ($v:ident, <) => {
143 $v - 1
144 };
145 ($v:ident, =) => {
146 $v
147 };
148}
149
150macro_rules! format_media_query {
151 ($cmp:tt, $suffix:tt, $v:ident) => {
152 format!(
153 "({}-width: {}{}px)",
154 $cmp,
155 value_expr!($v, $suffix),
156 query_suffix!($suffix)
157 )
158 };
159}
160
161macro_rules! impl_cmp_reactively {
162 ( #[$attr:meta]
163 $fn:ident, $cmp:tt, $suffix:tt) => {
164 paste! {
165 #[$attr]
167 pub fn $fn(&self, key: K) -> Signal<bool> {
168 if let Some(value) = self.breakpoints.get(&key) {
169 use_media_query(format_media_query!($cmp, $suffix, value))
170 } else {
171 self.not_found_signal(key)
172 }
173 }
174
175 #[$attr]
177 pub fn [<is_ $fn>](&self, key: K) -> bool {
178 if let Some(value) = self.breakpoints.get(&key) {
179 Self::match_(&format_media_query!($cmp, $suffix, value))
180 } else {
181 self.not_found(key)
182 }
183 }
184 }
185 };
186}
187
188impl<K> UseBreakpointsReturn<K>
189where
190 K: Eq + Hash + Debug + Clone + Send + Sync + 'static,
191{
192 fn match_(query: &str) -> bool {
193 if let Ok(Some(query_list)) = use_window().match_media(query) {
194 return query_list.matches();
195 }
196
197 false
198 }
199
200 fn not_found_signal(&self, key: K) -> Signal<bool> {
201 error!("Breakpoint \"{:?}\" not found", key);
202 Signal::derive(|| false)
203 }
204
205 fn not_found(&self, key: K) -> bool {
206 error!("Breakpoint \"{:?}\" not found", key);
207 false
208 }
209
210 impl_cmp_reactively!(
211 gt, "min", >
213 );
214 impl_cmp_reactively!(
215 ge, "min", =
217 );
218 impl_cmp_reactively!(
219 lt, "max", <
221 );
222 impl_cmp_reactively!(
223 le, "max", =
225 );
226
227 fn between_media_query(min: &u32, max: &u32) -> String {
228 format!("(min-width: {min}px) and (max-width: {}.9px)", max - 1)
229 }
230
231 pub fn between(&self, min_key: K, max_key: K) -> Signal<bool> {
233 if let Some(min) = self.breakpoints.get(&min_key) {
234 if let Some(max) = self.breakpoints.get(&max_key) {
235 use_media_query(Self::between_media_query(min, max))
236 } else {
237 self.not_found_signal(max_key)
238 }
239 } else {
240 self.not_found_signal(min_key)
241 }
242 }
243
244 pub fn is_between(&self, min_key: K, max_key: K) -> bool {
246 if let Some(min) = self.breakpoints.get(&min_key) {
247 if let Some(max) = self.breakpoints.get(&max_key) {
248 Self::match_(&Self::between_media_query(min, max))
249 } else {
250 self.not_found(max_key)
251 }
252 } else {
253 self.not_found(min_key)
254 }
255 }
256
257 pub fn current(&self) -> Signal<Vec<K>> {
259 let breakpoints = self.breakpoints.clone();
260 let keys: Vec<_> = breakpoints.keys().cloned().collect();
261
262 let ge = move |key: &K| {
263 let value = breakpoints
264 .get(key)
265 .expect("only used with keys() from the HashMap");
266
267 use_media_query(format_media_query!("min", =, value))
268 };
269
270 let signals: Vec<_> = keys.iter().map(ge.clone()).collect();
271
272 Signal::derive(move || {
273 keys.iter()
274 .cloned()
275 .zip(signals.iter().cloned())
276 .filter_map(|(key, signal)| signal.get().then_some(key))
277 .collect::<Vec<_>>()
278 })
279 }
280}
281
282#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
286pub enum BreakpointsTailwind {
287 Sm,
288 Md,
289 Lg,
290 Xl,
291 Xxl,
292}
293
294pub fn breakpoints_tailwind() -> HashMap<BreakpointsTailwind, u32> {
298 HashMap::from([
299 (BreakpointsTailwind::Sm, 640),
300 (BreakpointsTailwind::Md, 768),
301 (BreakpointsTailwind::Lg, 1024),
302 (BreakpointsTailwind::Xl, 1280),
303 (BreakpointsTailwind::Xxl, 1536),
304 ])
305}
306
307#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
311pub enum BreakpointsBootstrapV5 {
312 Sm,
313 Md,
314 Lg,
315 Xl,
316 Xxl,
317}
318
319pub fn breakpoints_bootstrap_v5() -> HashMap<BreakpointsBootstrapV5, u32> {
323 HashMap::from([
324 (BreakpointsBootstrapV5::Sm, 576),
325 (BreakpointsBootstrapV5::Md, 768),
326 (BreakpointsBootstrapV5::Lg, 992),
327 (BreakpointsBootstrapV5::Xl, 1200),
328 (BreakpointsBootstrapV5::Xxl, 1400),
329 ])
330}
331
332#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
336pub enum BreakpointsMaterial {
337 Xs,
338 Sm,
339 Md,
340 Lg,
341 Xl,
342}
343
344pub fn breakpoints_material() -> HashMap<BreakpointsMaterial, u32> {
348 HashMap::from([
349 (BreakpointsMaterial::Xs, 1),
350 (BreakpointsMaterial::Sm, 600),
351 (BreakpointsMaterial::Md, 900),
352 (BreakpointsMaterial::Lg, 1200),
353 (BreakpointsMaterial::Xl, 1536),
354 ])
355}
356
357#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
361pub enum BreakpointsAntDesign {
362 Xs,
363 Sm,
364 Md,
365 Lg,
366 Xl,
367 Xxl,
368}
369
370pub fn breakpoints_ant_design() -> HashMap<BreakpointsAntDesign, u32> {
374 HashMap::from([
375 (BreakpointsAntDesign::Xs, 480),
376 (BreakpointsAntDesign::Sm, 576),
377 (BreakpointsAntDesign::Md, 768),
378 (BreakpointsAntDesign::Lg, 992),
379 (BreakpointsAntDesign::Xl, 1200),
380 (BreakpointsAntDesign::Xxl, 1600),
381 ])
382}
383
384#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
388pub enum BreakpointsQuasar {
389 Xs,
390 Sm,
391 Md,
392 Lg,
393 Xl,
394}
395
396pub fn breakpoints_quasar() -> HashMap<BreakpointsQuasar, u32> {
400 HashMap::from([
401 (BreakpointsQuasar::Xs, 1),
402 (BreakpointsQuasar::Sm, 600),
403 (BreakpointsQuasar::Md, 1024),
404 (BreakpointsQuasar::Lg, 1440),
405 (BreakpointsQuasar::Xl, 1920),
406 ])
407}
408
409#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
413pub enum BreakpointsSemantic {
414 Mobile,
415 Tablet,
416 SmallMonitor,
417 LargeMonitor,
418}
419
420pub fn breakpoints_semantic() -> HashMap<BreakpointsSemantic, u32> {
424 HashMap::from([
425 (BreakpointsSemantic::Mobile, 1),
426 (BreakpointsSemantic::Tablet, 768),
427 (BreakpointsSemantic::SmallMonitor, 992),
428 (BreakpointsSemantic::LargeMonitor, 1200),
429 ])
430}
431
432#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
436pub enum BreakpointsMasterCss {
437 Xxxs,
438 Xxs,
439 Xs,
440 Sm,
441 Md,
442 Lg,
443 Xl,
444 Xxl,
445 Xxxl,
446 Xxxxl,
447}
448
449pub fn breakpoints_master_css() -> HashMap<BreakpointsMasterCss, u32> {
453 HashMap::from([
454 (BreakpointsMasterCss::Xxxs, 360),
455 (BreakpointsMasterCss::Xxs, 480),
456 (BreakpointsMasterCss::Xs, 600),
457 (BreakpointsMasterCss::Sm, 768),
458 (BreakpointsMasterCss::Md, 1024),
459 (BreakpointsMasterCss::Lg, 1280),
460 (BreakpointsMasterCss::Xl, 1440),
461 (BreakpointsMasterCss::Xxl, 1600),
462 (BreakpointsMasterCss::Xxxl, 1920),
463 (BreakpointsMasterCss::Xxxxl, 2560),
464 ])
465}