From 659d6a6fd5a1493943ad0e384aa05fed9b4f7b63 Mon Sep 17 00:00:00 2001
From: griffi-gh <prasol258@gmail.com>
Date: Tue, 11 Mar 2025 15:54:11 +0100
Subject: [PATCH] hui: no_std support

---
 hui/src/element.rs                      |  3 ++-
 hui/src/element/builtin/container.rs    |  3 ++-
 hui/src/element/builtin/frame_view.rs   |  3 ++-
 hui/src/element/builtin/image.rs        |  2 +-
 hui/src/element/builtin/interactable.rs |  1 +
 hui/src/element/builtin/progress_bar.rs |  3 ++-
 hui/src/element/builtin/slider.rs       |  3 ++-
 hui/src/element/builtin/text.rs         |  4 ++--
 hui/src/element/builtin/transformer.rs  |  1 +
 hui/src/event.rs                        |  3 ++-
 hui/src/font.rs                         |  1 +
 hui/src/frame/stack.rs                  |  1 +
 hui/src/input.rs                        |  3 ++-
 hui/src/lib.rs                          |  6 +++---
 hui/src/measure.rs                      |  4 +++-
 hui/src/signal.rs                       |  3 ++-
 hui/src/signal/trigger.rs               |  2 +-
 hui/src/state.rs                        | 17 +++++++++--------
 18 files changed, 39 insertions(+), 24 deletions(-)

diff --git a/hui/src/element.rs b/hui/src/element.rs
index 7cc1a5f..eb7cc61 100644
--- a/hui/src/element.rs
+++ b/hui/src/element.rs
@@ -1,5 +1,7 @@
 //! element API and built-in elements like `Container`, `Button`, `Text`, etc.
 
+use alloc::{boxed::Box, vec::Vec};
+use hui_painter::{paint::command::PaintList, text::FontHandle, PainterInstance};
 use crate::{
   input::InputCtx,
   layout::{LayoutInfo, Size2d},
@@ -12,7 +14,6 @@ use crate::{
 
 mod builtin;
 pub use builtin::*;
-use hui_painter::{paint::command::PaintList, text::FontHandle, PainterInstance};
 
 /// Context for the `Element::measure` function
 pub struct MeasureContext<'a> {
diff --git a/hui/src/element/builtin/container.rs b/hui/src/element/builtin/container.rs
index d6dd13e..df5ccc3 100644
--- a/hui/src/element/builtin/container.rs
+++ b/hui/src/element/builtin/container.rs
@@ -1,5 +1,6 @@
 //! a container element that can hold and layout multiple children elements
 
+use alloc::{boxed::Box, vec::Vec};
 use derive_setters::Setters;
 use glam::{Vec2, vec2};
 use crate::{
@@ -28,7 +29,7 @@ struct ContainerUserData {
 
 /// A container element that can hold and layout multiple children elements
 #[derive(Setters)]
-#[setters(prefix = "with_")]
+#[setters(no_std, prefix = "with_")]
 pub struct Container {
   /// Size of the container
   #[setters(into)]
diff --git a/hui/src/element/builtin/frame_view.rs b/hui/src/element/builtin/frame_view.rs
index fed8191..41465e7 100644
--- a/hui/src/element/builtin/frame_view.rs
+++ b/hui/src/element/builtin/frame_view.rs
@@ -1,5 +1,6 @@
 //! Simple element that displays the specified frame
 
+use alloc::boxed::Box;
 use derive_setters::Setters;
 use crate::{
   element::{MeasureContext, ProcessContext, UiElement},
@@ -11,7 +12,7 @@ use crate::{
 
 /// Simple rectangle that displays the specified frame
 #[derive(Setters)]
-#[setters(prefix = "with_")]
+#[setters(no_std, prefix = "with_")]
 pub struct FrameView {
   /// Size of the rectangle
   #[setters(into)]
diff --git a/hui/src/element/builtin/image.rs b/hui/src/element/builtin/image.rs
index fb7a83a..314b0d5 100644
--- a/hui/src/element/builtin/image.rs
+++ b/hui/src/element/builtin/image.rs
@@ -9,7 +9,7 @@ use crate::{
 };
 
 #[derive(Setters)]
-#[setters(prefix = "with_")]
+#[setters(no_std, prefix = "with_")]
 pub struct Image {
   /// Image handle to draw
   #[setters(skip)]
diff --git a/hui/src/element/builtin/interactable.rs b/hui/src/element/builtin/interactable.rs
index ff4183e..e75af7c 100644
--- a/hui/src/element/builtin/interactable.rs
+++ b/hui/src/element/builtin/interactable.rs
@@ -3,6 +3,7 @@
 // not sure if this is a good idea...
 // but having the ability to add a click event to any element would be nice, and this is a naive way to do it
 
+use alloc::boxed::Box;
 use crate::{
   element::{MeasureContext, ProcessContext, UiElement},
   signal::{trigger::SignalTrigger, Signal},
diff --git a/hui/src/element/builtin/progress_bar.rs b/hui/src/element/builtin/progress_bar.rs
index 8003c82..04657cf 100644
--- a/hui/src/element/builtin/progress_bar.rs
+++ b/hui/src/element/builtin/progress_bar.rs
@@ -1,3 +1,4 @@
+use alloc::boxed::Box;
 use derive_setters::Setters;
 use glam::vec2;
 use crate::{
@@ -10,7 +11,7 @@ use crate::{
 //TODO: Use Frames here instead of FillColor
 
 #[derive(Setters)]
-#[setters(prefix = "with_")]
+#[setters(no_std, prefix = "with_")]
 pub struct ProgressBar {
   /// Current progress, should be in the range 0.0..=1.0
   pub value: f32,
diff --git a/hui/src/element/builtin/slider.rs b/hui/src/element/builtin/slider.rs
index c4396ca..7fbfe23 100644
--- a/hui/src/element/builtin/slider.rs
+++ b/hui/src/element/builtin/slider.rs
@@ -1,5 +1,6 @@
 //! a slider element that allows selecting a value in a range
 
+use alloc::boxed::Box;
 use derive_setters::Setters;
 use glam::{Vec2, vec2};
 
@@ -33,7 +34,7 @@ pub enum SliderFollowMode {
 
 /// A slider element that allows selecting a value in a range
 #[derive(Setters)]
-#[setters(prefix = "with_")]
+#[setters(no_std, prefix = "with_")]
 pub struct Slider {
   /// Value of the slider, should be in range 0..1
   ///
diff --git a/hui/src/element/builtin/text.rs b/hui/src/element/builtin/text.rs
index 4154dd8..a0e705e 100644
--- a/hui/src/element/builtin/text.rs
+++ b/hui/src/element/builtin/text.rs
@@ -1,6 +1,6 @@
 //! simple text element, renders a string of text
 
-use std::borrow::Cow;
+use alloc::borrow::Cow;
 use derive_setters::Setters;
 use glam::{Affine2, Vec4};
 use hui_painter::{
@@ -22,7 +22,7 @@ use crate::{
 
 /// Simple text element, renders a string of text
 #[derive(Setters)]
-#[setters(prefix = "with_")]
+#[setters(no_std, prefix = "with_")]
 pub struct Text {
   /// Text to render
   #[setters(into)]
diff --git a/hui/src/element/builtin/transformer.rs b/hui/src/element/builtin/transformer.rs
index f6f35f6..8059542 100644
--- a/hui/src/element/builtin/transformer.rs
+++ b/hui/src/element/builtin/transformer.rs
@@ -1,5 +1,6 @@
 //! wrapper that allows applying various transformations to an element, such as translation, rotation, or scaling
 
+use alloc::boxed::Box;
 use glam::{Affine2, Vec2};
 use hui_painter::paint::command::{PaintList, PaintTransform};
 use crate::{
diff --git a/hui/src/event.rs b/hui/src/event.rs
index 37eb87b..bc6af9f 100644
--- a/hui/src/event.rs
+++ b/hui/src/event.rs
@@ -1,5 +1,6 @@
 //! input, window events and event handling
 
+use alloc::vec::{Vec, Drain};
 use glam::Vec2;
 use crate::input::{MouseButton, ButtonState, KeyboardKey};
 
@@ -31,7 +32,7 @@ impl EventQueue {
     self.events.push(event);
   }
 
-  pub(crate) fn drain(&mut self) -> std::vec::Drain<UiEvent> {
+  pub(crate) fn drain(&mut self) -> Drain<UiEvent> {
     self.events.drain(..)
   }
 }
diff --git a/hui/src/font.rs b/hui/src/font.rs
index 0bfccde..efa59e8 100644
--- a/hui/src/font.rs
+++ b/hui/src/font.rs
@@ -1,4 +1,5 @@
 
+use alloc::vec::Vec;
 use hui_painter::text::{FontHandle, DEFAULT_FONT};
 
 pub struct FontStack {
diff --git a/hui/src/frame/stack.rs b/hui/src/frame/stack.rs
index fd47f79..16ae43f 100644
--- a/hui/src/frame/stack.rs
+++ b/hui/src/frame/stack.rs
@@ -1,5 +1,6 @@
 //! allows stacking two frames on top of each other
 
+use alloc::boxed::Box;
 use hui_painter::paint::command::PaintList;
 use crate::rect::Rect;
 use super::Frame;
diff --git a/hui/src/input.rs b/hui/src/input.rs
index 12f3ede..1f5a004 100644
--- a/hui/src/input.rs
+++ b/hui/src/input.rs
@@ -1,6 +1,7 @@
 //! keyboard, mouse, and touch input handling
 
-use std::hash::{Hash, Hasher};
+use core::hash::{Hash, Hasher};
+use alloc::vec::Vec;
 use glam::Vec2;
 use hashbrown::HashMap;
 use nohash_hasher::BuildNoHashHasher;
diff --git a/hui/src/lib.rs b/hui/src/lib.rs
index 642e1e0..9aeb1f5 100644
--- a/hui/src/lib.rs
+++ b/hui/src/lib.rs
@@ -1,4 +1,4 @@
-// #![no_std]
+#![no_std]
 #![doc(html_logo_url = "https://raw.githubusercontent.com/griffi-gh/hui/master/.assets/hui.svg")]
 //!
 //! Simple UI library for games and other interactive applications
@@ -11,8 +11,8 @@
 #![forbid(unsafe_op_in_unsafe_fn)]
 #![allow(unused_parens)]
 
-// #[macro_use]
-// extern crate alloc;
+#[macro_use]
+extern crate alloc;
 
 pub use hui_shared::*;
 pub use hui_painter as painter;
diff --git a/hui/src/measure.rs b/hui/src/measure.rs
index a131c75..33d723d 100644
--- a/hui/src/measure.rs
+++ b/hui/src/measure.rs
@@ -1,5 +1,7 @@
 //! element measurement, hints and responses
 
+use core::any::Any;
+use alloc::{boxed::Box, vec::Vec};
 use glam::Vec2;
 use crate::rect::Rect;
 
@@ -20,7 +22,7 @@ pub struct Response {
   pub hints: Hints,
 
   /// Arbitrary user data, can be used to pass data (for example, cache) between measure and process stages
-  pub user_data: Option<Box<dyn std::any::Any>>,
+  pub user_data: Option<Box<dyn Any>>,
 
   /// If true, the element should always cause the content to wrap to the next line\
   /// (the element itself gets wrapped to the next line too)
diff --git a/hui/src/signal.rs b/hui/src/signal.rs
index f5f91cc..e6db5b4 100644
--- a/hui/src/signal.rs
+++ b/hui/src/signal.rs
@@ -1,6 +1,7 @@
 //! signal handling for UI events
 
-use std::any::{Any, TypeId};
+use core::any::{Any, TypeId};
+use alloc::{boxed::Box, vec::Vec};
 use hashbrown::HashMap;
 use nohash_hasher::BuildNoHashHasher;
 
diff --git a/hui/src/signal/trigger.rs b/hui/src/signal/trigger.rs
index f5081b8..5e0aff3 100644
--- a/hui/src/signal/trigger.rs
+++ b/hui/src/signal/trigger.rs
@@ -1,7 +1,7 @@
 //! Contains the implementation of signal triggers, which simplify creation of custom elements
 
+use alloc::boxed::Box;
 use super::{Signal, SignalStore};
-//use crate::element::UiElement;
 
 /// Signal trigger that does not take any arguments
 #[allow(clippy::complexity)]
diff --git a/hui/src/state.rs b/hui/src/state.rs
index d9598b1..f16ca23 100644
--- a/hui/src/state.rs
+++ b/hui/src/state.rs
@@ -1,8 +1,9 @@
 //! state managment for stateful elements
 
+use alloc::{boxed::Box, vec::Vec};
 use hashbrown::{HashMap, HashSet};
 use nohash_hasher::BuildNoHashHasher;
-use std::{any::Any, hash::{Hash, Hasher}};
+use core::{any::Any, hash::{Hash, Hasher}};
 use rustc_hash::FxHasher;
 
 //TODO impl StateRepo functions and automatic cleanup of inactive ids
@@ -78,7 +79,7 @@ impl StateRepo {
   pub fn acquire_or_insert<T: State>(&mut self, id: impl Hash, state: T) -> &T {
     let id = hash_local(id, &self.id_stack);
     self.state.entry(id)
-      .or_insert_with(|| Box::new(state))
+      .or_insert_with(|| Box::new(state) as Box<dyn Any>)
       .downcast_ref::<T>().unwrap()
   }
 
@@ -86,7 +87,7 @@ impl StateRepo {
   pub fn acquire_or_default<T: State + Default>(&mut self, id: impl Hash) -> &T {
     let id = hash_local(id, &self.id_stack);
     self.state.entry(id)
-      .or_insert_with(|| Box::<T>::default())
+      .or_insert_with(|| Box::<T>::default() as Box<dyn Any>)
       .downcast_ref::<T>().unwrap()
   }
 
@@ -101,7 +102,7 @@ impl StateRepo {
   pub fn acquire_mut_or_insert<T: State>(&mut self, id: impl Hash, state: T) -> &mut T {
     let id = hash_local(id, &self.id_stack);
     self.state.entry(id)
-      .or_insert_with(|| Box::new(state))
+      .or_insert_with(|| Box::new(state) as Box<dyn Any>)
       .downcast_mut::<T>().unwrap()
   }
 
@@ -109,7 +110,7 @@ impl StateRepo {
   pub fn acquire_mut_or_default<T: State + Default>(&mut self, id: impl Hash) -> &mut T {
     let id = hash_local(id, &self.id_stack);
     self.state.entry(id)
-      .or_insert_with(|| Box::<T>::default())
+      .or_insert_with(|| Box::<T>::default() as Box<dyn Any>)
       .downcast_mut::<T>().unwrap()
   }
 
@@ -118,9 +119,9 @@ impl StateRepo {
   /// Can be useful for state management of non-hierarchical objects, e.g. popups
   pub fn global<R>(&mut self, f: impl FnOnce(&mut Self) -> R) -> R {
     self.standby.clear();
-    std::mem::swap(&mut self.id_stack, &mut self.standby);
+    core::mem::swap(&mut self.id_stack, &mut self.standby);
     let ret = f(self);
-    std::mem::swap(&mut self.id_stack, &mut self.standby);
+    core::mem::swap(&mut self.id_stack, &mut self.standby);
     ret
   }
 
@@ -132,7 +133,7 @@ impl StateRepo {
     self.standby.clear();
     self.standby.extend(self.id_stack.iter().copied());
     let ret = f(self);
-    std::mem::swap(&mut self.id_stack, &mut self.standby);
+    core::mem::swap(&mut self.id_stack, &mut self.standby);
     ret
     //XXX: this is super efficient, but works only for pushes, if anything is popped, it will be lost
     // let len = self.id_stack.len();