Free from context dependency

This commit is contained in:
Michael Hübner 2023-01-16 12:58:59 +01:00
parent a46b65fece
commit b1fa659e49
30 changed files with 272 additions and 42 deletions

View file

@ -7,11 +7,12 @@ edition = "2021"
[dependencies]
quick-xml = "0.27.1"
serde = { version = "1.0.152", features = ["derive"] }
# context = { path = "../context" }
vulkan-rs = { git = "https://gavania.de/hodasemi/vulkan_lib.git" }
utilities = { git = "https://gavania.de/hodasemi/utilities.git" }
paste = "1.0.11"
assetpath = { git = "https://gavania.de/hodasemi/vulkan_lib.git" }
anyhow = { version = "1.0.68", features = ["backtrace"] }
cgmath = "0.18.0"
[features]
audio = []

View file

@ -1,5 +1,6 @@
use crate::prelude::*;
use anyhow::Result;
use utilities::prelude::*;
use assetpath::AssetPath;

View file

@ -1,6 +1,7 @@
use crate::prelude::*;
use anyhow::Result;
use assetpath::AssetPath;
use utilities::prelude::*;
use super::gridinfo::GridInfo;
use super::mandatory::Mandatory;

View file

@ -1,5 +1,6 @@
use crate::prelude::*;
use anyhow::Result;
use utilities::prelude::*;
use super::gridinfo::GridInfo;
use super::mandatory::Mandatory;

View file

@ -1,5 +1,6 @@
use crate::prelude::*;
use anyhow::Result;
use utilities::prelude::*;
use super::gridinfo::GridInfo;
use super::mandatory::Mandatory;

View file

@ -1,5 +1,6 @@
use crate::prelude::*;
use anyhow::Result;
use utilities::prelude::*;
use super::gridinfo::GridInfo;
use super::mandatory::Mandatory;

View file

@ -1,5 +1,6 @@
use crate::prelude::*;
use anyhow::Result;
use utilities::prelude::*;
use super::gridinfo::GridInfo;
use super::mandatory::Mandatory;

View file

@ -1,5 +1,6 @@
use crate::prelude::*;
use anyhow::Result;
use utilities::prelude::*;
use super::gridinfo::GridInfo;
use super::mandatory::Mandatory;

View file

@ -1,6 +1,7 @@
use crate::prelude::*;
use anyhow::{Context, Result};
use assetpath::AssetPath;
use utilities::prelude::*;
use quick_xml::{events::Event, Reader};

60
src/context_interface.rs Normal file
View file

@ -0,0 +1,60 @@
use std::sync::{Arc, Mutex};
use vulkan_rs::prelude::*;
pub trait ContextInterface: Send + Sync {
fn device(&self) -> &Arc<Device>;
fn queue(&self) -> &Arc<Mutex<Queue>>;
fn format(&self) -> VkFormat;
fn image_layout(&self) -> VkImageLayout;
fn image_count(&self) -> usize;
fn images(&self) -> TargetMode<Vec<Arc<Image>>>;
fn width(&self) -> u32;
fn height(&self) -> u32;
}
pub enum TargetMode<T> {
Mono(T),
Stereo(T, T),
}
impl<T> TargetMode<T> {
pub fn mono(&self) -> &T {
match self {
TargetMode::Mono(s) => s,
TargetMode::Stereo(_, _) => panic!("Expected another target mode"),
}
}
pub fn mono_mut(&mut self) -> &mut T {
match self {
TargetMode::Mono(s) => s,
TargetMode::Stereo(_, _) => panic!("Expected another target mode"),
}
}
pub fn stereo(&self) -> (&T, &T) {
match self {
TargetMode::Mono(_) => panic!("Expected another target mode"),
TargetMode::Stereo(l, r) => (l, r),
}
}
pub fn stereo_mut(&mut self) -> (&mut T, &mut T) {
match self {
TargetMode::Mono(_) => panic!("Expected another target mode"),
TargetMode::Stereo(l, r) => (l, r),
}
}
}
impl<T: Clone> Clone for TargetMode<T> {
fn clone(&self) -> TargetMode<T> {
match self {
TargetMode::Mono(t) => TargetMode::Mono(t.clone()),
TargetMode::Stereo(lhs, rhs) => TargetMode::Stereo(lhs.clone(), rhs.clone()),
}
}
}

115
src/controller_button.rs Normal file
View file

@ -0,0 +1,115 @@
use anyhow::Result;
#[derive(PartialEq, Eq, Hash, Copy, Clone, Debug)]
pub enum ControllerButton {
A,
B,
Y,
X,
Start,
Select,
RightButton,
LeftButton,
RightTrigger,
LeftTrigger,
DPadUp,
DPadDown,
DPadRight,
DPadLeft,
Guide,
LeftStick,
RightStick,
Misc,
Paddle1,
Paddle2,
Paddle3,
Paddle4,
Touchpad,
}
impl TryFrom<&str> for ControllerButton {
type Error = anyhow::Error;
fn try_from(text: &str) -> Result<Self> {
match text {
"A" => Ok(ControllerButton::A),
"B" => Ok(ControllerButton::B),
"X" => Ok(ControllerButton::X),
"Y" => Ok(ControllerButton::Y),
"Select" => Ok(ControllerButton::Select),
"Start" => Ok(ControllerButton::Start),
"RightButton" => Ok(ControllerButton::RightButton),
"LeftButton" => Ok(ControllerButton::LeftButton),
"RightTrigger" => Ok(ControllerButton::RightTrigger),
"LeftTrigger" => Ok(ControllerButton::LeftTrigger),
"DPadUp" => Ok(ControllerButton::DPadUp),
"DPadDown" => Ok(ControllerButton::DPadDown),
"DPadLeft" => Ok(ControllerButton::DPadLeft),
"DPadRight" => Ok(ControllerButton::DPadRight),
"Guide" => Ok(ControllerButton::Guide),
"LeftStick" => Ok(ControllerButton::LeftStick),
"RightStick" => Ok(ControllerButton::RightStick),
_ => Err(anyhow::Error::msg(format!(
"Failed converting ControllerButton from {}",
text
))),
}
}
}
impl Into<&'static str> for ControllerButton {
fn into(self) -> &'static str {
use ControllerButton::*;
match self {
A => "A",
B => "B",
Y => "Y",
X => "X",
Start => "Start",
Select => "Select",
RightButton => "RightButton",
LeftButton => "LeftButton",
RightTrigger => "RightTrigger",
LeftTrigger => "LeftTrigger",
DPadUp => "DPadUp",
DPadDown => "DPadDown",
DPadRight => "DPadRight",
DPadLeft => "DPadLeft",
Guide => "Guide",
LeftStick => "LeftStick",
RightStick => "RightStick",
Misc => "Misc",
Paddle1 => "Paddle1",
Paddle2 => "Paddle2",
Paddle3 => "Paddle3",
Paddle4 => "Paddle4",
Touchpad => "Touchpad",
}
}
}

View file

@ -2,6 +2,8 @@ use crate::{builder::validator::buttoninfo::ButtonInfo, prelude::*};
use anyhow::Result;
use assetpath::AssetPath;
use utilities::prelude::*;
use vulkan_rs::prelude::*;
use super::{
fill_type::{FillType, InnerFillType},

View file

@ -1,6 +1,7 @@
use crate::prelude::*;
use anyhow::Result;
use assetpath::AssetPath;
use utilities::prelude::*;
use std::sync::{
atomic::{AtomicBool, Ordering::SeqCst},

View file

@ -1,5 +1,7 @@
use crate::{builder::validator::iconinfo::IconInfo, prelude::*};
use anyhow::Result;
use utilities::prelude::*;
use vulkan_rs::prelude::*;
use std::sync::{
atomic::{AtomicBool, Ordering::SeqCst},

View file

@ -8,6 +8,7 @@ use std::sync::{
};
use anyhow::Result;
use utilities::prelude::*;
pub struct LabelBuilder {
text_alignment: TextAlignment,

View file

@ -4,6 +4,8 @@ use std::sync::{
atomic::{AtomicU32, Ordering::SeqCst},
Arc, Mutex,
};
use utilities::prelude::*;
use vulkan_rs::prelude::*;
pub mod button;
pub mod fill_type;
@ -115,7 +117,12 @@ impl TextableWrapper {
}
pub(crate) fn text(&self) -> Result<Option<String>> {
Ok(self.textable.lock().unwrap().as_ref().map(|textable| textable.text()))
Ok(self
.textable
.lock()
.unwrap()
.as_ref()
.map(|textable| textable.text()))
}
pub(crate) fn set_height_ratio(&self, height_ratio: f32) -> Result<()> {

View file

@ -6,6 +6,7 @@ use std::sync::{
};
use anyhow::Result;
use utilities::prelude::*;
pub struct MultiLineLabelBuilder {
text_alignment: TextAlignment,

View file

@ -9,6 +9,7 @@ use std::sync::{
};
use anyhow::Result;
use utilities::prelude::*;
use super::fill_type::InnerFillType;

View file

@ -5,6 +5,7 @@ use std::sync::{
atomic::{AtomicBool, Ordering::SeqCst},
Arc, Mutex,
};
use utilities::prelude::*;
use super::{
fill_type::{FillType, InnerFillType},

View file

@ -1,5 +1,6 @@
use crate::{builder::validator::textfieldinfo::TextFieldInfo, prelude::*};
use anyhow::Result;
use utilities::prelude::*;
use std::sync::{
atomic::{AtomicBool, Ordering::SeqCst},

10
src/guidirection.rs Normal file
View file

@ -0,0 +1,10 @@
//! Enum for gui direction input abstraction
#[derive(Copy, Clone, PartialEq, Debug)]
pub enum GuiDirection {
Left,
Right,
Up,
Down,
None,
}

View file

@ -2,6 +2,8 @@
use crate::prelude::*;
use anyhow::Result;
use utilities::prelude::*;
use vulkan_rs::prelude::*;
use cgmath::{vec4, Vector4};

View file

@ -3,6 +3,8 @@
use crate::prelude::*;
use anyhow::Result;
use assetpath::AssetPath;
use utilities::prelude::*;
use vulkan_rs::prelude::*;
use cgmath::{vec2, vec4};

View file

@ -3,6 +3,8 @@
use crate::prelude::*;
use anyhow::Result;
use assetpath::AssetPath;
use utilities::prelude::*;
use vulkan_rs::prelude::*;
use super::texturedvertex::TexturedVertex;

View file

@ -3,6 +3,8 @@
use super::writeable::ModifyText;
use crate::prelude::*;
use anyhow::Result;
use utilities::prelude::*;
use vulkan_rs::prelude::*;
use super::texturedvertex::TexturedVertex;

View file

@ -1,4 +1,6 @@
use cgmath::prelude::Zero;
use utilities::prelude::*;
use vulkan_rs::prelude::*;
use std::mem;

View file

@ -1,6 +1,8 @@
use crate::prelude::*;
use anyhow::Result;
use assetpath::AssetPath;
use utilities::prelude::*;
use vulkan_rs::prelude::*;
use super::{
elements::Elements,
@ -64,7 +66,7 @@ struct CommandBufferState {
}
pub struct GuiHandler {
context: Arc<Context>,
context: Arc<dyn ContextInterface>,
device: Arc<Device>,
queue: Arc<Mutex<Queue>>,
@ -126,31 +128,28 @@ pub struct GuiHandler {
impl GuiHandler {
pub fn new(
gui_handler_create_info: GuiHandlerCreateInfo,
context: &Arc<Context>,
context: &Arc<dyn ContextInterface>,
) -> Result<Arc<GuiHandler>> {
let device = context.device();
let queue = context.queue();
let render_core = context.render_core();
let command_buffers = match render_core.images() {
TargetMode::Single(_) => {
let command_buffers = Self::create_command_buffers(device, queue, render_core)?;
let command_buffers = match context.images() {
TargetMode::Mono(_) => {
let command_buffers = Self::create_command_buffers(device, queue, context)?;
TargetMode::Single(command_buffers)
TargetMode::Mono(command_buffers)
}
TargetMode::Stereo(_, _) => {
let left_command_buffers =
Self::create_command_buffers(device, queue, render_core)?;
let right_command_buffers =
Self::create_command_buffers(device, queue, render_core)?;
let left_command_buffers = Self::create_command_buffers(device, queue, context)?;
let right_command_buffers = Self::create_command_buffers(device, queue, context)?;
TargetMode::Stereo(left_command_buffers, right_command_buffers)
}
};
let render_pass =
Self::create_render_pass(device, render_core.format(), render_core.image_layout())?;
let framebuffers = Self::create_framebuffers(device, &render_core.images(), &render_pass)?;
Self::create_render_pass(device, context.format(), context.image_layout())?;
let framebuffers = Self::create_framebuffers(device, &context.images(), &render_pass)?;
let (text_objs, color_layout) = Self::init_text_objects(device, &render_pass)?;
let rect_objs = Self::init_rectangle_objects(device, &render_pass)?;
@ -172,14 +171,14 @@ impl GuiHandler {
)
.build(device.clone())?;
let gui_handler = Arc::new(GuiHandler {
Ok(Arc::new(GuiHandler {
context: context.clone(),
device: device.clone(),
queue: queue.clone(),
width: AtomicU32::new(render_core.width()),
height: AtomicU32::new(render_core.height()),
width: AtomicU32::new(context.width()),
height: AtomicU32::new(context.height()),
menu_button: {
let mut menu_button = gui_handler_create_info.menu_button;
@ -244,9 +243,9 @@ impl GuiHandler {
ortho: RwLock::new(ortho(
0.0,
render_core.width() as f32,
context.width() as f32,
0.0,
render_core.height() as f32,
context.height() as f32,
-1.0,
1.0,
)),
@ -260,11 +259,7 @@ impl GuiHandler {
current_hoverable: RwLock::new(None),
current_selectable: RwLock::new(None),
current_writeable: RwLock::new(None),
});
render_core.add_post_processing_routine(gui_handler.clone());
Ok(gui_handler)
}))
}
pub fn device(&self) -> &Arc<Device> {
@ -1199,10 +1194,10 @@ impl GuiHandler {
};
match target_images {
TargetMode::Single(images) => {
TargetMode::Mono(images) => {
let framebuffers = create_framebuffer(device, images, render_pass)?;
Ok(TargetMode::Single(framebuffers))
Ok(TargetMode::Mono(framebuffers))
}
TargetMode::Stereo(left_images, right_images) => {
let left_framebuffers = create_framebuffer(device, left_images, render_pass)?;
@ -1216,11 +1211,11 @@ impl GuiHandler {
fn create_command_buffers(
device: &Arc<Device>,
queue: &Arc<Mutex<Queue>>,
render_core: &Box<dyn RenderCore>,
context: &Arc<dyn ContextInterface>,
) -> Result<Vec<CommandBufferState>> {
let mut command_buffers = Vec::with_capacity(render_core.image_count());
let mut command_buffers = Vec::with_capacity(context.image_count());
for _ in 0..render_core.image_count() {
for _ in 0..context.image_count() {
command_buffers.push(CommandBufferState {
command_buffer: CommandBuffer::new_secondary()
.build(device.clone(), queue.clone())?,
@ -1460,19 +1455,15 @@ impl GuiHandler {
}
}
impl PostProcess for GuiHandler {
fn priority(&self) -> u32 {
50
}
fn process(
impl GuiHandler {
pub fn process(
&self,
buffer_recorder: &mut CommandBufferRecorder<'_>,
indices: &TargetMode<usize>,
) -> Result<()> {
if self.needs_update.load(SeqCst) {
match &self.command_buffers {
TargetMode::Single(command_buffers) => {
TargetMode::Mono(command_buffers) => {
for state in command_buffers {
state.valid.store(false, SeqCst);
}
@ -1507,9 +1498,9 @@ impl PostProcess for GuiHandler {
indices,
) {
(
TargetMode::Single(command_buffers),
TargetMode::Single(framebuffers),
TargetMode::Single(index),
TargetMode::Mono(command_buffers),
TargetMode::Mono(framebuffers),
TargetMode::Mono(index),
) => {
self.select_rendering(buffer_recorder, command_buffers, framebuffers, *index)?;
}
@ -1527,10 +1518,10 @@ impl PostProcess for GuiHandler {
Ok(())
}
fn resize(&self, width: u32, height: u32) -> Result<()> {
pub fn resize(&self, width: u32, height: u32) -> Result<()> {
*self.framebuffers.lock().unwrap() = Self::create_framebuffers(
self.context.device(),
&self.context.render_core().images(),
&self.context.images(),
&self.render_pass,
)?;

View file

@ -7,6 +7,10 @@ pub mod keyboard;
pub mod states;
pub mod tab_control;
mod context_interface;
mod controller_button;
mod guidirection;
mod mousebutton;
pub mod prelude;
pub type CustomCallback<I, O> = dyn Fn(I) -> anyhow::Result<O> + Send + Sync;

10
src/mousebutton.rs Normal file
View file

@ -0,0 +1,10 @@
//! Mouse button abstraction
#[derive(PartialEq, Copy, Clone, Debug)]
pub enum MouseButton {
Left,
Middle,
Right,
Forward,
Backward,
}

View file

@ -1,7 +1,11 @@
pub use super::builder::{builder::GuiBuilder, snippet::GuiSnippet};
pub use super::context_interface::{ContextInterface, TargetMode};
pub use super::controller_button::ControllerButton;
pub use super::elements::prelude::*;
pub use super::guidirection::GuiDirection;
pub use super::guihandler::prelude::*;
pub use super::keyboard::Keyboard;
pub use super::mousebutton::MouseButton;
pub use super::states::*;
pub use super::tab_control::TabControl;