use super::shared::resize::Resize; use crate::consts::DRAG_THRESHOLD; use crate::document::DocumentMessageHandler; use crate::frontend::utility_types::MouseCursorIcon; use crate::input::keyboard::{Key, MouseMotion}; use crate::input::InputPreprocessorMessageHandler; use crate::layout::widgets::PropertyHolder; use crate::message_prelude::*; use crate::misc::{HintData, HintGroup, HintInfo, KeysGroup}; use crate::viewport_tools::tool::{DocumentToolData, Fsm, ToolActionHandlerData}; use graphene::layers::style; use graphene::Operation; use glam::DAffine2; use serde::{Deserialize, Serialize}; #[derive(Default)] pub struct EllipseTool { fsm_state: EllipseToolFsmState, data: EllipseToolData, } #[remain::sorted] #[impl_message(Message, ToolMessage, Ellipse)] #[derive(PartialEq, Clone, Debug, Hash, Serialize, Deserialize)] pub enum EllipseToolMessage { // Standard messages #[remain::unsorted] Abort, // Tool-specific messages DragStart, DragStop, Resize { center: Key, lock_ratio: Key, }, } impl PropertyHolder for EllipseTool {} impl<'a> MessageHandler> for EllipseTool { fn process_action(&mut self, action: ToolMessage, data: ToolActionHandlerData<'a>, responses: &mut VecDeque) { if action == ToolMessage::UpdateHints { self.fsm_state.update_hints(responses); return; } if action == ToolMessage::UpdateCursor { self.fsm_state.update_cursor(responses); return; } let new_state = self.fsm_state.transition(action, data.0, data.1, &mut self.data, &(), data.2, responses); if self.fsm_state != new_state { self.fsm_state = new_state; self.fsm_state.update_hints(responses); self.fsm_state.update_cursor(responses); } } fn actions(&self) -> ActionList { use EllipseToolFsmState::*; match self.fsm_state { Ready => actions!(EllipseToolMessageDiscriminant; DragStart), Drawing => actions!(EllipseToolMessageDiscriminant; DragStop, Abort, Resize), } } } #[derive(Clone, Copy, Debug, PartialEq, Eq)] enum EllipseToolFsmState { Ready, Drawing, } impl Default for EllipseToolFsmState { fn default() -> Self { EllipseToolFsmState::Ready } } #[derive(Clone, Debug, Default)] struct EllipseToolData { data: Resize, } impl Fsm for EllipseToolFsmState { type ToolData = EllipseToolData; type ToolOptions = (); fn transition( self, event: ToolMessage, document: &DocumentMessageHandler, tool_data: &DocumentToolData, data: &mut Self::ToolData, _tool_options: &Self::ToolOptions, input: &InputPreprocessorMessageHandler, responses: &mut VecDeque, ) -> Self { use EllipseToolFsmState::*; use EllipseToolMessage::*; let mut shape_data = &mut data.data; if let ToolMessage::Ellipse(event) = event { match (self, event) { (Ready, DragStart) => { shape_data.start(responses, document, input.mouse.position); responses.push_back(DocumentMessage::StartTransaction.into()); shape_data.path = Some(document.get_path_for_new_layer()); responses.push_back(DocumentMessage::DeselectAllLayers.into()); responses.push_back( Operation::AddEllipse { path: shape_data.path.clone().unwrap(), insert_index: -1, transform: DAffine2::ZERO.to_cols_array(), style: style::PathStyle::new(None, style::Fill::solid(tool_data.primary_color)), } .into(), ); Drawing } (state, Resize { center, lock_ratio }) => { if let Some(message) = shape_data.calculate_transform(responses, document, center, lock_ratio, input) { responses.push_back(message); } state } (Drawing, DragStop) => { match shape_data.drag_start.distance(input.mouse.position) <= DRAG_THRESHOLD { true => responses.push_back(DocumentMessage::AbortTransaction.into()), false => responses.push_back(DocumentMessage::CommitTransaction.into()), } shape_data.cleanup(responses); Ready } (Drawing, Abort) => { responses.push_back(DocumentMessage::AbortTransaction.into()); shape_data.cleanup(responses); Ready } _ => self, } } else { self } } fn update_hints(&self, responses: &mut VecDeque) { let hint_data = match self { EllipseToolFsmState::Ready => HintData(vec![HintGroup(vec![ HintInfo { key_groups: vec![], mouse: Some(MouseMotion::LmbDrag), label: String::from("Draw Ellipse"), plus: false, }, HintInfo { key_groups: vec![KeysGroup(vec![Key::KeyShift])], mouse: None, label: String::from("Constrain Circular"), plus: true, }, HintInfo { key_groups: vec![KeysGroup(vec![Key::KeyAlt])], mouse: None, label: String::from("From Center"), plus: true, }, ])]), EllipseToolFsmState::Drawing => HintData(vec![HintGroup(vec![ HintInfo { key_groups: vec![KeysGroup(vec![Key::KeyShift])], mouse: None, label: String::from("Constrain Circular"), plus: false, }, HintInfo { key_groups: vec![KeysGroup(vec![Key::KeyAlt])], mouse: None, label: String::from("From Center"), plus: false, }, ])]), }; responses.push_back(FrontendMessage::UpdateInputHints { hint_data }.into()); } fn update_cursor(&self, responses: &mut VecDeque) { responses.push_back(FrontendMessage::UpdateMouseCursor { cursor: MouseCursorIcon::Crosshair }.into()); } }