refactor context_menu to only require parser_state on tick

This commit is contained in:
galister
2026-01-07 13:59:14 +09:00
parent 5123363454
commit 165070da51
3 changed files with 68 additions and 62 deletions

View File

@@ -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);
} }

View File

@@ -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(())
} }
} }

View File

@@ -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(&params.template_name, &params.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();