mirror of
https://github.com/griffi-gh/kubi.git
synced 2024-11-25 16:28:42 -06:00
Compare commits
6 commits
63f862c2be
...
0b210eac5d
Author | SHA1 | Date | |
---|---|---|---|
griffi-gh | 0b210eac5d | ||
griffi-gh | 0fa723fc00 | ||
griffi-gh | a95eb96843 | ||
griffi-gh | 0f4264292c | ||
griffi-gh | 4e47901117 | ||
griffi-gh | 447a848009 |
2
Cargo.lock
generated
2
Cargo.lock
generated
|
@ -1033,8 +1033,10 @@ dependencies = [
|
||||||
"glam",
|
"glam",
|
||||||
"glium",
|
"glium",
|
||||||
"hashbrown",
|
"hashbrown",
|
||||||
|
"kubi-logging",
|
||||||
"log",
|
"log",
|
||||||
"nohash-hasher",
|
"nohash-hasher",
|
||||||
|
"winit",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|
|
@ -12,9 +12,11 @@ glium = { git = "https://github.com/glium/glium", rev = "968fc92378caf", optiona
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
|
kubi-logging = { path = "../kubi-logging" }
|
||||||
glium = { git = "https://github.com/glium/glium", rev = "968fc92378caf" }
|
glium = { git = "https://github.com/glium/glium", rev = "968fc92378caf" }
|
||||||
|
winit = "0.29"
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["backend_glium", "builtin_elements"]
|
default = ["backend_glium", "builtin"]
|
||||||
backend_glium = ["dep:glium"]
|
backend_glium = ["dep:glium"]
|
||||||
builtin_elements = []
|
builtin = []
|
||||||
|
|
|
@ -1,3 +1,82 @@
|
||||||
pub fn main() {
|
use std::time::Instant;
|
||||||
//TODO ui demo
|
use glam::{Vec2, IVec2, UVec2};
|
||||||
|
use glium::{backend::glutin::SimpleWindowBuilder, Surface};
|
||||||
|
use winit::{
|
||||||
|
event::{Event, WindowEvent},
|
||||||
|
event_loop::{EventLoopBuilder, ControlFlow}
|
||||||
|
};
|
||||||
|
use kubi_ui::{
|
||||||
|
KubiUi,
|
||||||
|
backend::glium::GliumUiRenderer, element::{progress_bar::ProgressBar, container::{Container, Sides}, UiElement}, UiSize
|
||||||
|
};
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
kubi_logging::init();
|
||||||
|
|
||||||
|
let event_loop = EventLoopBuilder::new().build().unwrap();
|
||||||
|
let (window, display) = SimpleWindowBuilder::new().build(&event_loop);
|
||||||
|
|
||||||
|
let mut kui = KubiUi::new();
|
||||||
|
let mut backend = GliumUiRenderer::new(&display);
|
||||||
|
|
||||||
|
let instant = Instant::now();
|
||||||
|
|
||||||
|
event_loop.run(|event, window_target| {
|
||||||
|
window_target.set_control_flow(ControlFlow::Poll);
|
||||||
|
match event {
|
||||||
|
Event::WindowEvent { event: WindowEvent::CloseRequested, .. } => {
|
||||||
|
window_target.exit();
|
||||||
|
},
|
||||||
|
Event::AboutToWait => {
|
||||||
|
let mut frame = display.draw();
|
||||||
|
frame.clear_color_srgb(0.5, 0.5, 0.5, 0.);
|
||||||
|
|
||||||
|
let resolution = UVec2::from(display.get_framebuffer_dimensions()).as_vec2();
|
||||||
|
|
||||||
|
kui.begin();
|
||||||
|
|
||||||
|
kui.add(Container {
|
||||||
|
gap: 5.,
|
||||||
|
padding: Sides::all(5.),
|
||||||
|
elements: vec![
|
||||||
|
Box::new(ProgressBar {
|
||||||
|
value: 0.5,
|
||||||
|
..Default::default()
|
||||||
|
}),
|
||||||
|
Box::new(ProgressBar {
|
||||||
|
value: instant.elapsed().as_secs_f32().sin().powi(2),
|
||||||
|
..Default::default()
|
||||||
|
}),
|
||||||
|
Box::new(Container {
|
||||||
|
gap: 1.,
|
||||||
|
elements: {
|
||||||
|
let mut elements: Vec<Box<dyn UiElement>> = vec![];
|
||||||
|
for i in 0..10000 {
|
||||||
|
elements.push(Box::new(ProgressBar {
|
||||||
|
value: (instant.elapsed().as_secs_f32() + (i as f32 / 10.)).sin().powi(2),
|
||||||
|
size: (UiSize::Auto, UiSize::Pixels(5.)),
|
||||||
|
..Default::default()
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
elements
|
||||||
|
},
|
||||||
|
..Default::default()
|
||||||
|
})
|
||||||
|
],
|
||||||
|
..Default::default()
|
||||||
|
}, resolution);
|
||||||
|
|
||||||
|
kui.end();
|
||||||
|
|
||||||
|
let plan = kui.draw_plan();
|
||||||
|
if plan.0 {
|
||||||
|
backend.update(plan.1);
|
||||||
|
}
|
||||||
|
backend.draw(&mut frame, resolution);
|
||||||
|
|
||||||
|
frame.finish().unwrap();
|
||||||
|
}
|
||||||
|
_ => (),
|
||||||
|
}
|
||||||
|
}).unwrap();
|
||||||
}
|
}
|
||||||
|
|
|
@ -48,8 +48,9 @@ impl BufferPair {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn ensure_buffer_size(&mut self, need_vtx: usize, need_idx: usize) {
|
pub fn ensure_buffer_size(&mut self, need_vtx: usize, need_idx: usize) {
|
||||||
let current_vtx_size = self.vertex_buffer.get_size();
|
let current_vtx_size = self.vertex_buffer.get_size() / std::mem::size_of::<Vertex>();
|
||||||
let current_idx_size = self.index_buffer.get_size();
|
let current_idx_size = self.index_buffer.get_size() / std::mem::size_of::<u32>();
|
||||||
|
//log::debug!("current vtx size: {}, current idx size: {}", current_vtx_size, current_idx_size);
|
||||||
if current_vtx_size >= need_vtx && current_idx_size >= need_idx {
|
if current_vtx_size >= need_vtx && current_idx_size >= need_idx {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -84,9 +85,10 @@ impl BufferPair {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
self.ensure_buffer_size(vtx.len(), idx.len());
|
self.ensure_buffer_size(self.vertex_count, self.index_count);
|
||||||
self.vertex_buffer.slice_mut(0..vtx.len()).unwrap().write(vtx);
|
|
||||||
self.index_buffer.slice_mut(0..idx.len()).unwrap().write(idx);
|
self.vertex_buffer.slice_mut(0..self.vertex_count).unwrap().write(vtx);
|
||||||
|
self.index_buffer.slice_mut(0..self.index_count).unwrap().write(idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn is_empty(&self) -> bool {
|
pub fn is_empty(&self) -> bool {
|
||||||
|
|
|
@ -6,10 +6,16 @@ use crate::{
|
||||||
state::StateRepo
|
state::StateRepo
|
||||||
};
|
};
|
||||||
|
|
||||||
#[cfg(feature = "builtin_elements")] pub mod rect;
|
#[cfg(feature = "builtin")]
|
||||||
#[cfg(feature = "builtin_elements")] pub mod container;
|
mod builtin {
|
||||||
#[cfg(feature = "builtin_elements")] pub mod spacer;
|
pub mod rect;
|
||||||
#[cfg(feature = "builtin_elements")] pub mod progress_bar;
|
pub mod container;
|
||||||
|
pub mod spacer;
|
||||||
|
pub mod progress_bar;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "builtin")]
|
||||||
|
pub use builtin::*;
|
||||||
|
|
||||||
pub trait UiElement {
|
pub trait UiElement {
|
||||||
fn name(&self) -> &'static str { "UiElement" }
|
fn name(&self) -> &'static str { "UiElement" }
|
||||||
|
|
|
@ -1,6 +1,13 @@
|
||||||
use glam::{Vec2, vec2, Vec4};
|
use glam::{Vec2, vec2, Vec4};
|
||||||
use crate::{UiDirection, LayoutInfo, draw::UiDrawCommand, measure::Response, state::StateRepo, UiSize};
|
use crate::{
|
||||||
use super::UiElement;
|
UiDirection,
|
||||||
|
UiSize,
|
||||||
|
LayoutInfo,
|
||||||
|
draw::UiDrawCommand,
|
||||||
|
measure::Response,
|
||||||
|
state::StateRepo,
|
||||||
|
element::UiElement
|
||||||
|
};
|
||||||
|
|
||||||
pub enum Alignment {
|
pub enum Alignment {
|
||||||
Begin,
|
Begin,
|
||||||
|
@ -75,6 +82,15 @@ impl Default for Container {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Container {
|
||||||
|
pub fn measure_max_size(&self, layout: &LayoutInfo) -> Vec2 {
|
||||||
|
layout.max_size - vec2(
|
||||||
|
self.padding.left + self.padding.right,
|
||||||
|
self.padding.top + self.padding.bottom,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl UiElement for Container {
|
impl UiElement for Container {
|
||||||
fn measure(&self, state: &StateRepo, layout: &LayoutInfo) -> Response {
|
fn measure(&self, state: &StateRepo, layout: &LayoutInfo) -> Response {
|
||||||
let mut size = Vec2::ZERO;
|
let mut size = Vec2::ZERO;
|
||||||
|
@ -82,7 +98,7 @@ impl UiElement for Container {
|
||||||
for element in &self.elements {
|
for element in &self.elements {
|
||||||
let measure = element.measure(state, &LayoutInfo {
|
let measure = element.measure(state, &LayoutInfo {
|
||||||
position: layout.position + size,
|
position: layout.position + size,
|
||||||
max_size: layout.max_size, // - size, //TODO
|
max_size: self.measure_max_size(layout), // - size TODO
|
||||||
direction: self.direction,
|
direction: self.direction,
|
||||||
});
|
});
|
||||||
match self.direction {
|
match self.direction {
|
||||||
|
@ -141,19 +157,17 @@ impl UiElement for Container {
|
||||||
for element in &self.elements {
|
for element in &self.elements {
|
||||||
//(passing max size from layout rather than actual bounds for the sake of consistency with measure() above)
|
//(passing max size from layout rather than actual bounds for the sake of consistency with measure() above)
|
||||||
|
|
||||||
//measure
|
let layout = LayoutInfo {
|
||||||
let el_measure = element.measure(state, &LayoutInfo {
|
|
||||||
position,
|
position,
|
||||||
max_size: layout.max_size,
|
max_size: self.measure_max_size(layout),
|
||||||
direction: self.direction,
|
direction: self.direction,
|
||||||
});
|
};
|
||||||
|
|
||||||
|
//measure
|
||||||
|
let el_measure = element.measure(state, &layout);
|
||||||
|
|
||||||
//process
|
//process
|
||||||
element.process(&el_measure, state, &LayoutInfo {
|
element.process(&el_measure, state, &layout, draw);
|
||||||
position,
|
|
||||||
max_size: layout.max_size,
|
|
||||||
direction: self.direction,
|
|
||||||
}, draw);
|
|
||||||
|
|
||||||
//layout
|
//layout
|
||||||
match self.direction {
|
match self.direction {
|
|
@ -3,10 +3,11 @@ use crate::{
|
||||||
UiSize, LayoutInfo,
|
UiSize, LayoutInfo,
|
||||||
draw::UiDrawCommand,
|
draw::UiDrawCommand,
|
||||||
measure::Response,
|
measure::Response,
|
||||||
state::StateRepo
|
state::StateRepo,
|
||||||
|
element::UiElement
|
||||||
};
|
};
|
||||||
use super::UiElement;
|
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct ProgressBar {
|
pub struct ProgressBar {
|
||||||
pub size: (UiSize, UiSize),
|
pub size: (UiSize, UiSize),
|
||||||
pub value: f32,
|
pub value: f32,
|
|
@ -1,6 +1,12 @@
|
||||||
use glam::{vec2, Vec4};
|
use glam::{vec2, Vec4};
|
||||||
use crate::{state::StateRepo, LayoutInfo, measure::Response, draw::UiDrawCommand, UiSize};
|
use crate::{
|
||||||
use super::UiElement;
|
LayoutInfo,
|
||||||
|
UiSize,
|
||||||
|
element::UiElement,
|
||||||
|
state::StateRepo,
|
||||||
|
measure::Response,
|
||||||
|
draw::UiDrawCommand
|
||||||
|
};
|
||||||
|
|
||||||
pub struct Rect {
|
pub struct Rect {
|
||||||
pub size: (UiSize, UiSize),
|
pub size: (UiSize, UiSize),
|
|
@ -1,6 +1,12 @@
|
||||||
use glam::vec2;
|
use glam::vec2;
|
||||||
use crate::{state::StateRepo, LayoutInfo, measure::Response, draw::UiDrawCommand, UiDirection};
|
use crate::{
|
||||||
use super::UiElement;
|
LayoutInfo,
|
||||||
|
UiDirection,
|
||||||
|
element::UiElement,
|
||||||
|
state::StateRepo,
|
||||||
|
measure::Response,
|
||||||
|
draw::UiDrawCommand
|
||||||
|
};
|
||||||
|
|
||||||
pub struct Spacer(f32);
|
pub struct Spacer(f32);
|
||||||
|
|
|
@ -72,7 +72,7 @@ impl Default for KubiUi {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default, Debug, Clone, Copy)]
|
||||||
pub enum UiSize {
|
pub enum UiSize {
|
||||||
#[default]
|
#[default]
|
||||||
Auto,
|
Auto,
|
||||||
|
|
|
@ -59,11 +59,11 @@ pub(super) fn init_cube_primitive(
|
||||||
display: NonSendSync<UniqueView<Renderer>>
|
display: NonSendSync<UniqueView<Renderer>>
|
||||||
) {
|
) {
|
||||||
{
|
{
|
||||||
let vert = VertexBuffer::new(
|
let vert = VertexBuffer::immutable(
|
||||||
&display.display,
|
&display.display,
|
||||||
CUBE_VERTICES
|
CUBE_VERTICES
|
||||||
).unwrap();
|
).unwrap();
|
||||||
let index = IndexBuffer::new(
|
let index = IndexBuffer::immutable(
|
||||||
&display.display,
|
&display.display,
|
||||||
PrimitiveType::TrianglesList,
|
PrimitiveType::TrianglesList,
|
||||||
CUBE_INDICES
|
CUBE_INDICES
|
||||||
|
@ -71,11 +71,11 @@ pub(super) fn init_cube_primitive(
|
||||||
storages.add_unique_non_send_sync(CubePrimitive(vert, index));
|
storages.add_unique_non_send_sync(CubePrimitive(vert, index));
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
let vert = VertexBuffer::new(
|
let vert = VertexBuffer::immutable(
|
||||||
&display.display,
|
&display.display,
|
||||||
CENTERED_CUBE_VERTICES
|
CENTERED_CUBE_VERTICES
|
||||||
).unwrap();
|
).unwrap();
|
||||||
let index = IndexBuffer::new(
|
let index = IndexBuffer::immutable(
|
||||||
&display.display,
|
&display.display,
|
||||||
PrimitiveType::TrianglesList,
|
PrimitiveType::TrianglesList,
|
||||||
CUBE_INDICES
|
CUBE_INDICES
|
||||||
|
|
|
@ -18,11 +18,11 @@ pub(super) fn init_rect_primitive(
|
||||||
storages: AllStoragesView,
|
storages: AllStoragesView,
|
||||||
display: NonSendSync<UniqueView<Renderer>>
|
display: NonSendSync<UniqueView<Renderer>>
|
||||||
) {
|
) {
|
||||||
let vert = VertexBuffer::new(
|
let vert = VertexBuffer::immutable(
|
||||||
&display.display,
|
&display.display,
|
||||||
RECT_VERTEX
|
RECT_VERTEX
|
||||||
).unwrap();
|
).unwrap();
|
||||||
let index = IndexBuffer::new(
|
let index = IndexBuffer::immutable(
|
||||||
&display.display,
|
&display.display,
|
||||||
PrimitiveType::TrianglesList,
|
PrimitiveType::TrianglesList,
|
||||||
RECT_INDEX
|
RECT_INDEX
|
||||||
|
|
Loading…
Reference in a new issue