refactor context_menu to only require parser_state on tick
This commit is contained in:
@@ -8,9 +8,9 @@ use glam::Vec2;
|
|||||||
use wgui::{
|
use wgui::{
|
||||||
assets::AssetPath,
|
assets::AssetPath,
|
||||||
components::{
|
components::{
|
||||||
Component,
|
|
||||||
button::{ButtonClickCallback, ComponentButton},
|
button::{ButtonClickCallback, ComponentButton},
|
||||||
checkbox::ComponentCheckbox,
|
checkbox::ComponentCheckbox,
|
||||||
|
Component,
|
||||||
},
|
},
|
||||||
drawing::Color,
|
drawing::Color,
|
||||||
event::StyleSetRequest,
|
event::StyleSetRequest,
|
||||||
@@ -35,15 +35,13 @@ pub enum TestbedTask {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct Data {
|
struct Data {
|
||||||
#[allow(dead_code)]
|
|
||||||
state: ParserState,
|
|
||||||
|
|
||||||
popup_window: WguiWindow,
|
popup_window: WguiWindow,
|
||||||
context_menu: context_menu::ContextMenu,
|
context_menu: context_menu::ContextMenu,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct TestbedGeneric {
|
pub struct TestbedGeneric {
|
||||||
pub layout: Layout,
|
pub layout: Layout,
|
||||||
|
pub parser_state: ParserState,
|
||||||
tasks: Tasks<TestbedTask>,
|
tasks: Tasks<TestbedTask>,
|
||||||
|
|
||||||
globals: WguiGlobals,
|
globals: WguiGlobals,
|
||||||
@@ -122,15 +120,15 @@ impl TestbedGeneric {
|
|||||||
dev_mode: false,
|
dev_mode: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
let (layout, state) = wgui::parser::new_layout_from_assets(
|
let (layout, parser_state) = wgui::parser::new_layout_from_assets(
|
||||||
&TestbedGeneric::doc_params(&globals, extra),
|
&TestbedGeneric::doc_params(&globals, extra),
|
||||||
&LayoutParams {
|
&LayoutParams {
|
||||||
resize_to_parent: true,
|
resize_to_parent: true,
|
||||||
},
|
},
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
let cb_visible = state.fetch_component_as::<ComponentCheckbox>("cb_visible")?;
|
let cb_visible = parser_state.fetch_component_as::<ComponentCheckbox>("cb_visible")?;
|
||||||
let div_visibility = state.fetch_widget(&layout.state, "div_visibility")?;
|
let div_visibility = parser_state.fetch_widget(&layout.state, "div_visibility")?;
|
||||||
|
|
||||||
cb_visible.on_toggle(Box::new(move |common, evt| {
|
cb_visible.on_toggle(Box::new(move |common, evt| {
|
||||||
common.alterables.set_style(
|
common.alterables.set_style(
|
||||||
@@ -144,20 +142,21 @@ impl TestbedGeneric {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}));
|
}));
|
||||||
|
|
||||||
let label_cur_option = state.fetch_widget(&layout.state, "label_current_option")?;
|
let label_cur_option = parser_state.fetch_widget(&layout.state, "label_current_option")?;
|
||||||
|
|
||||||
let button_context_menu = state.fetch_component_as::<ComponentButton>("button_context_menu")?;
|
let button_context_menu =
|
||||||
let button_click_me = state.fetch_component_as::<ComponentButton>("button_click_me")?;
|
parser_state.fetch_component_as::<ComponentButton>("button_context_menu")?;
|
||||||
|
let button_click_me = parser_state.fetch_component_as::<ComponentButton>("button_click_me")?;
|
||||||
let button = button_click_me.clone();
|
let button = button_click_me.clone();
|
||||||
button_click_me.on_click(Box::new(move |common, _e| {
|
button_click_me.on_click(Box::new(move |common, _e| {
|
||||||
button.set_text(common, Translation::from_raw_text("congrats!"));
|
button.set_text(common, Translation::from_raw_text("congrats!"));
|
||||||
Ok(())
|
Ok(())
|
||||||
}));
|
}));
|
||||||
|
|
||||||
let button_popup = state.fetch_component_as::<ComponentButton>("button_popup")?;
|
let button_popup = parser_state.fetch_component_as::<ComponentButton>("button_popup")?;
|
||||||
let button_red = state.fetch_component_as::<ComponentButton>("button_red")?;
|
let button_red = parser_state.fetch_component_as::<ComponentButton>("button_red")?;
|
||||||
let button_aqua = state.fetch_component_as::<ComponentButton>("button_aqua")?;
|
let button_aqua = parser_state.fetch_component_as::<ComponentButton>("button_aqua")?;
|
||||||
let button_yellow = state.fetch_component_as::<ComponentButton>("button_yellow")?;
|
let button_yellow = parser_state.fetch_component_as::<ComponentButton>("button_yellow")?;
|
||||||
|
|
||||||
handle_button_click(button_red, label_cur_option.widget.clone(), "Clicked red");
|
handle_button_click(button_red, label_cur_option.widget.clone(), "Clicked red");
|
||||||
handle_button_click(button_aqua, label_cur_option.widget.clone(), "Clicked aqua");
|
handle_button_click(button_aqua, label_cur_option.widget.clone(), "Clicked aqua");
|
||||||
@@ -167,7 +166,7 @@ impl TestbedGeneric {
|
|||||||
"Clicked yellow",
|
"Clicked yellow",
|
||||||
);
|
);
|
||||||
|
|
||||||
let cb_first = state.fetch_component_as::<ComponentCheckbox>("cb_first")?;
|
let cb_first = parser_state.fetch_component_as::<ComponentCheckbox>("cb_first")?;
|
||||||
let label = label_cur_option.widget.clone();
|
let label = label_cur_option.widget.clone();
|
||||||
cb_first.on_toggle(Box::new(move |common, e| {
|
cb_first.on_toggle(Box::new(move |common, e| {
|
||||||
let mut widget = label.get_as::<WidgetLabel>().unwrap();
|
let mut widget = label.get_as::<WidgetLabel>().unwrap();
|
||||||
@@ -178,10 +177,10 @@ impl TestbedGeneric {
|
|||||||
|
|
||||||
let testbed = Self {
|
let testbed = Self {
|
||||||
layout,
|
layout,
|
||||||
|
parser_state,
|
||||||
tasks: Default::default(),
|
tasks: Default::default(),
|
||||||
globals: globals.clone(),
|
globals: globals.clone(),
|
||||||
data: Rc::new(RefCell::new(Data {
|
data: Rc::new(RefCell::new(Data {
|
||||||
state,
|
|
||||||
popup_window: WguiWindow::default(),
|
popup_window: WguiWindow::default(),
|
||||||
context_menu: context_menu::ContextMenu::default(),
|
context_menu: context_menu::ContextMenu::default(),
|
||||||
})),
|
})),
|
||||||
@@ -214,7 +213,7 @@ impl TestbedGeneric {
|
|||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
match task {
|
match task {
|
||||||
TestbedTask::ShowPopup => self.show_popup(params, data)?,
|
TestbedTask::ShowPopup => self.show_popup(params, data)?,
|
||||||
TestbedTask::ShowContextMenu(position) => self.show_context_menu(params, data, *position)?,
|
TestbedTask::ShowContextMenu(position) => self.show_context_menu(params, data, *position),
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -255,18 +254,15 @@ impl TestbedGeneric {
|
|||||||
_params: &mut TestbedUpdateParams,
|
_params: &mut TestbedUpdateParams,
|
||||||
data: &mut Data,
|
data: &mut Data,
|
||||||
position: Vec2,
|
position: Vec2,
|
||||||
) -> anyhow::Result<()> {
|
) {
|
||||||
data.state.instantiate_context_menu(
|
data.context_menu.open(context_menu::OpenParams {
|
||||||
Some(Rc::new(move |custom_attribs| {
|
on_custom_attribs: Some(Rc::new(move |custom_attribs| {
|
||||||
log::info!("custom attribs {:?}", custom_attribs.pairs);
|
log::info!("custom attribs {:?}", custom_attribs.pairs);
|
||||||
})),
|
})),
|
||||||
"my_context_menu",
|
template_name: "my_context_menu".into(),
|
||||||
Default::default(),
|
template_params: Default::default(),
|
||||||
&mut data.context_menu,
|
|
||||||
position,
|
position,
|
||||||
)?;
|
});
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -286,7 +282,9 @@ impl Testbed for TestbedGeneric {
|
|||||||
self.process_task(&task, &mut params, &mut data)?;
|
self.process_task(&task, &mut params, &mut data)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
let res = data.context_menu.tick(&mut self.layout)?;
|
let res = data
|
||||||
|
.context_menu
|
||||||
|
.tick(&mut self.layout, &mut self.parser_state)?;
|
||||||
if let Some(action_name) = res.action_name {
|
if let Some(action_name) = res.action_name {
|
||||||
log::info!("got action: {}", action_name);
|
log::info!("got action: {}", action_name);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -260,14 +260,12 @@ impl ParserState {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn instantiate_context_menu(
|
pub(crate) fn context_menu_create_blueprint(
|
||||||
&mut self,
|
&mut self,
|
||||||
on_custom_attribs: Option<OnCustomAttribsFunc>,
|
|
||||||
template_name: &str,
|
template_name: &str,
|
||||||
template_params: HashMap<Rc<str>, Rc<str>>,
|
template_params: &HashMap<Rc<str>, Rc<str>>,
|
||||||
context_menu: &mut context_menu::ContextMenu,
|
|
||||||
position: Vec2,
|
position: Vec2,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<context_menu::Blueprint> {
|
||||||
let Some(template) = self.data.templates.get(template_name) else {
|
let Some(template) = self.data.templates.get(template_name) else {
|
||||||
anyhow::bail!("no template named \"{template_name}\" found");
|
anyhow::bail!("no template named \"{template_name}\" found");
|
||||||
};
|
};
|
||||||
@@ -319,13 +317,12 @@ impl ParserState {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
context_menu.open(context_menu::OpenParams {
|
Ok(
|
||||||
cells,
|
context_menu::Blueprint {
|
||||||
on_custom_attribs,
|
cells,
|
||||||
position,
|
position,
|
||||||
});
|
}
|
||||||
|
)
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,4 @@
|
|||||||
use std::{
|
use std::{collections::HashMap, rc::Rc};
|
||||||
collections::{HashMap, HashSet},
|
|
||||||
rc::Rc,
|
|
||||||
};
|
|
||||||
|
|
||||||
use glam::Vec2;
|
use glam::Vec2;
|
||||||
|
|
||||||
@@ -11,7 +8,7 @@ use crate::{
|
|||||||
globals::WguiGlobals,
|
globals::WguiGlobals,
|
||||||
i18n::Translation,
|
i18n::Translation,
|
||||||
layout::Layout,
|
layout::Layout,
|
||||||
parser::{self, Fetchable},
|
parser::{self, Fetchable, ParserState},
|
||||||
task::Tasks,
|
task::Tasks,
|
||||||
windowing::window::{WguiWindow, WguiWindowParams, WguiWindowParamsExtra},
|
windowing::window::{WguiWindow, WguiWindowParams, WguiWindowParamsExtra},
|
||||||
};
|
};
|
||||||
@@ -22,10 +19,16 @@ pub struct Cell {
|
|||||||
pub attribs: Vec<parser::AttribPair>,
|
pub attribs: Vec<parser::AttribPair>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct OpenParams {
|
pub(crate) struct Blueprint {
|
||||||
pub position: Vec2,
|
pub position: Vec2,
|
||||||
pub cells: Vec<Cell>,
|
pub cells: Vec<Cell>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct OpenParams {
|
||||||
pub on_custom_attribs: Option<parser::OnCustomAttribsFunc>,
|
pub on_custom_attribs: Option<parser::OnCustomAttribsFunc>,
|
||||||
|
pub template_name: Rc<str>,
|
||||||
|
pub template_params: HashMap<Rc<str>, Rc<str>>,
|
||||||
|
pub position: Vec2,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
@@ -40,11 +43,17 @@ pub struct ContextMenu {
|
|||||||
tasks: Tasks<Task>,
|
tasks: Tasks<Task>,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn doc_params<'a>(globals: &WguiGlobals) -> parser::ParseDocumentParams<'a> {
|
fn doc_params<'a>(
|
||||||
|
globals: &WguiGlobals,
|
||||||
|
on_custom_attribs: Option<parser::OnCustomAttribsFunc>,
|
||||||
|
) -> parser::ParseDocumentParams<'a> {
|
||||||
parser::ParseDocumentParams {
|
parser::ParseDocumentParams {
|
||||||
globals: globals.clone(),
|
globals: globals.clone(),
|
||||||
path: AssetPath::WguiInternal("wgui/context_menu.xml"),
|
path: AssetPath::WguiInternal("wgui/context_menu.xml"),
|
||||||
extra: Default::default(),
|
extra: parser::ParseDocumentExtra {
|
||||||
|
on_custom_attribs,
|
||||||
|
..Default::default()
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -62,7 +71,15 @@ impl ContextMenu {
|
|||||||
self.window.close();
|
self.window.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
fn open_process(&mut self, params: &mut OpenParams, layout: &mut Layout) -> anyhow::Result<()> {
|
fn open_process(
|
||||||
|
&mut self,
|
||||||
|
params: &mut OpenParams,
|
||||||
|
layout: &mut Layout,
|
||||||
|
parser_state: &mut ParserState,
|
||||||
|
) -> anyhow::Result<()> {
|
||||||
|
let blueprint =
|
||||||
|
parser_state.context_menu_create_blueprint(¶ms.template_name, ¶ms.template_params, params.position)?;
|
||||||
|
|
||||||
let globals = layout.state.globals.clone();
|
let globals = layout.state.globals.clone();
|
||||||
|
|
||||||
self.window.open(&mut WguiWindowParams {
|
self.window.open(&mut WguiWindowParams {
|
||||||
@@ -77,15 +94,16 @@ impl ContextMenu {
|
|||||||
})?;
|
})?;
|
||||||
|
|
||||||
let content = self.window.get_content();
|
let content = self.window.get_content();
|
||||||
|
let doc_params = doc_params(&globals, params.on_custom_attribs.clone());
|
||||||
|
|
||||||
let mut state = parser::parse_from_assets(&doc_params(&globals), layout, content.id)?;
|
let mut state = parser::parse_from_assets(&doc_params, layout, content.id)?;
|
||||||
|
|
||||||
let id_buttons = state.get_widget_id("buttons")?;
|
let id_buttons = state.get_widget_id("buttons")?;
|
||||||
|
|
||||||
for (idx, cell) in params.cells.iter().enumerate() {
|
for (idx, cell) in blueprint.cells.iter().enumerate() {
|
||||||
let mut par = HashMap::new();
|
let mut par = HashMap::new();
|
||||||
par.insert(Rc::from("text"), cell.title.generate(&mut globals.i18n()));
|
par.insert(Rc::from("text"), cell.title.generate(&mut globals.i18n()));
|
||||||
let data_cell = state.parse_template(&doc_params(&globals), "Cell", layout, id_buttons, par)?;
|
let data_cell = state.parse_template(&doc_params, "Cell", layout, id_buttons, par)?;
|
||||||
|
|
||||||
let button = data_cell.fetch_component_as::<ComponentButton>("button")?;
|
let button = data_cell.fetch_component_as::<ComponentButton>("button")?;
|
||||||
let button_id = button.base().get_id();
|
let button_id = button.base().get_id();
|
||||||
@@ -102,23 +120,16 @@ impl ContextMenu {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if idx < params.cells.len() - 1 {
|
if idx < blueprint.cells.len() - 1 {
|
||||||
state.parse_template(
|
state.parse_template(&doc_params, "Separator", layout, id_buttons, Default::default())?;
|
||||||
&doc_params(&globals),
|
|
||||||
"Separator",
|
|
||||||
layout,
|
|
||||||
id_buttons,
|
|
||||||
Default::default(),
|
|
||||||
)?;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn tick(&mut self, layout: &mut Layout) -> anyhow::Result<TickResult> {
|
pub fn tick(&mut self, layout: &mut Layout, parser_state: &mut ParserState) -> anyhow::Result<TickResult> {
|
||||||
if let Some(mut p) = self.pending_open.take() {
|
if let Some(mut p) = self.pending_open.take() {
|
||||||
self.open_process(&mut p, layout)?;
|
self.open_process(&mut p, layout, parser_state)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut result = TickResult::default();
|
let mut result = TickResult::default();
|
||||||
|
|||||||
Reference in New Issue
Block a user