rumtk_web/pages/mod.rs
1/*
2 * rumtk attempts to implement HL7 and medical protocols for interoperability in medicine.
3 * This toolkit aims to be reliable, simple, performant, and standards compliant.
4 * Copyright (C) 2025 Luis M. Santos, M.D.
5 * Copyright (C) 2025 Nick Stephenson
6 * Copyright (C) 2025 Ethan Dixon
7 * Copyright (C) 2025 MedicalMasses L.L.C.
8 *
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
13 *
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
18 *
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22 */
23pub mod index;
24
25use crate::utils::PageFunction;
26use rumtk_core::cache::{new_cache, LazyRUMCache, LazyRUMCacheValue};
27use rumtk_core::strings::RUMString;
28use rumtk_core::{rumtk_cache_get, rumtk_cache_push};
29use std::ops::Deref;
30
31pub type PageCache = LazyRUMCache<RUMString, PageFunction>;
32pub type PageItem<'a> = (&'a str, PageFunction);
33pub type UserPages<'a> = Vec<PageItem<'a>>;
34pub type PageCacheItem = LazyRUMCacheValue<PageFunction>;
35
36static mut PAGE_CACHE: PageCache = new_cache();
37static mut DEFAULT_PAGE: PageFunction = index::index;
38pub const DEFAULT_PAGE_NAME: &str = "default";
39
40pub fn register_page(name: &str, component_fxn: PageFunction) -> PageCacheItem {
41 let key = RUMString::from(name);
42 let r = rumtk_cache_push!(&raw mut PAGE_CACHE, &key, &component_fxn);
43 unsafe {
44 DEFAULT_PAGE = component_fxn;
45 }
46 println!(
47 " ➡ Registered page {} => page function [{:?}]",
48 name, &component_fxn
49 );
50 r
51}
52
53pub fn get_page(name: &str) -> PageCacheItem {
54 rumtk_cache_get!(
55 &raw mut PAGE_CACHE,
56 &RUMString::from(name),
57 get_default_page()
58 )
59}
60
61pub fn get_default_page() -> &'static PageFunction {
62 unsafe { &DEFAULT_PAGE }
63}
64
65pub fn init_pages(user_components: &UserPages) {
66 println!("🗎 Registering user pages! 🗎");
67 /* Init any user prescribed components */
68 for itm in user_components {
69 let (name, value) = itm;
70 register_page(name, *value);
71 }
72 println!("🗎 ~~~~~~~~~~~~~~~~~~~~~~ 🗎");
73}
74
75#[macro_export]
76macro_rules! rumtk_web_register_page {
77 ( $key:expr, $fxn:expr ) => {{
78 use $crate::pages::register_page;
79 register_page($key, $fxn)
80 }};
81}
82
83///
84/// Helper function for retrieving pages registered in the global cache using a string key!
85///
86/// ## Example
87///
88/// ### With Named Page
89///
90/// ```
91/// use rumtk_core::strings::rumtk_format;
92/// use rumtk_web::utils::{SharedAppConf, RenderedPageComponents};
93/// use rumtk_web::{rumtk_web_render_component, rumtk_web_register_page, rumtk_web_get_page};
94///
95/// pub fn index(app_state: SharedAppConf) -> RenderedPageComponents {
96/// let title_welcome = rumtk_web_render_component!("title", [("type", "welcome")], app_state.clone());
97///
98/// vec![
99/// title_welcome,
100/// ]
101/// }
102///
103/// let r = rumtk_format!("{:?}", &rumtk_web_register_page!("index", index));
104/// let p = rumtk_format!("{:?}", &rumtk_web_get_page!("index"));
105///
106/// assert_eq!(&r, &p, "{}", rumtk_format!("The registered page does not match the retrieved page!\nGot: {:?}\nExpected: {:?}", &r, &p));
107///
108/// ```
109///
110/// ### With Default Page
111///
112/// ```
113/// use rumtk_core::strings::rumtk_format;
114/// use rumtk_web::utils::{SharedAppConf, RenderedPageComponents};
115/// use rumtk_web::{rumtk_web_render_component, rumtk_web_register_page, rumtk_web_get_page, rumtk_web_get_default_page};
116///
117/// pub fn index(app_state: SharedAppConf) -> RenderedPageComponents {
118/// let title_welcome = rumtk_web_render_component!("title", [("type", "welcome")], app_state.clone());
119///
120/// vec![
121/// title_welcome,
122/// ]
123/// }
124///
125/// let default = rumtk_format!("{:?}", rumtk_web_get_default_page!());
126/// let r = rumtk_format!("{:?}", &rumtk_web_register_page!("index", index));
127/// let p = rumtk_format!("{:?}", &rumtk_web_get_page!(""));
128///
129/// assert_ne!(&default, &p, "{}", rumtk_format!("The default page matches the retrieved page!\nGot: {:?}\nExpected: {:?}", &r, &p));
130/// assert_eq!(&r, &p, "{}", rumtk_format!("The registered page does not match the retrieved page!\nGot: {:?}\nExpected: {:?}", &r, &p));
131///
132/// ```
133///
134#[macro_export]
135macro_rules! rumtk_web_get_page {
136 ( $key:expr ) => {{
137 use $crate::pages::get_page;
138 get_page($key)
139 }};
140}
141
142///
143/// Returns the default page function that can be used for rendering that page
144///
145#[macro_export]
146macro_rules! rumtk_web_get_default_page {
147 ( ) => {{
148 use $crate::pages::get_default_page;
149 get_default_page()
150 }};
151}
152
153///
154/// Registers a set of pages provided by the user.
155///
156/// ## Example
157///
158///```
159/// use std::ops::Deref;
160/// use rumtk_core::strings::rumtk_format;
161/// use rumtk_web::utils::{SharedAppConf, RenderedPageComponents};
162/// use rumtk_web::{rumtk_web_render_component, rumtk_web_init_pages, rumtk_web_get_page};
163///
164/// fn my_page(app_state: SharedAppConf) -> RenderedPageComponents {
165/// let title_welcome = rumtk_web_render_component!("title", [("type", "welcome")], app_state.clone());
166///
167/// vec![
168/// title_welcome,
169/// ]
170/// }
171///
172/// let my_page_name = "my_page";
173///
174/// rumtk_web_init_pages!(&vec![(my_page_name, my_page)]);
175///```
176///
177///
178#[macro_export]
179macro_rules! rumtk_web_init_pages {
180 ( $pages:expr ) => {{
181 use $crate::pages::init_pages;
182 init_pages($pages)
183 }};
184}
185
186#[macro_export]
187macro_rules! rumtk_web_collect_page {
188 ( $page:expr, $app_state:expr ) => {{
189 use $crate::rumtk_web_get_page;
190
191 let page = rumtk_web_get_page!(&$page);
192
193 page($app_state.clone())
194 }};
195}