Implement Pen Tool (#79)

This commit is contained in:
Edwin Cheng 2021-04-23 00:12:24 +08:00 committed by Keavon Chambers
parent b556dd6bfd
commit e40727e914
9 changed files with 143 additions and 8 deletions

View File

@ -72,10 +72,10 @@
<ItemDivider horizontal />
<ShelfItem title="Path Tool" :active="activeTool === 'Path'" @click="'tool not implemented' || selectTool('Path')"><PathTool /></ShelfItem>
<ShelfItem title="Pen Tool" :active="activeTool === 'Pen'" @click="'tool not implemented' || selectTool('Pen')"><PenTool /></ShelfItem>
<ShelfItem title="Pen Tool (P)" :active="activeTool === 'Pen'" @click="selectTool('Pen')"><PenTool /></ShelfItem>
<ShelfItem title="Freehand Tool" :active="activeTool === 'Freehand'" @click="'tool not implemented' || selectTool('Freehand')"><FreehandTool /></ShelfItem>
<ShelfItem title="Spline Tool" :active="activeTool === 'Spline'" @click="'tool not implemented' || selectTool('Spline')"><SplineTool /></ShelfItem>
<ShelfItem title="Line Tool" :active="activeTool === 'Line'" @click="selectTool('Line')"><LineTool /></ShelfItem>
<ShelfItem title="Line Tool (L)" :active="activeTool === 'Line'" @click="selectTool('Line')"><LineTool /></ShelfItem>
<ShelfItem title="Rectangle Tool (M)" :active="activeTool === 'Rectangle'" @click="selectTool('Rectangle')"><RectangleTool /></ShelfItem>
<ShelfItem title="Ellipse Tool (E)" :active="activeTool === 'Ellipse'" @click="selectTool('Ellipse')"><EllipseTool /></ShelfItem>
<ShelfItem title="Shape Tool (Y)" :active="activeTool === 'Shape'" @click="selectTool('Shape')"><ShapeTool /></ShelfItem>

View File

@ -80,6 +80,7 @@ pub fn translate_key(name: &str) -> events::Key {
"e" => K::KeyE,
"v" => K::KeyV,
"l" => K::KeyL,
"p" => K::KeyP,
"r" => K::KeyR,
"m" => K::KeyM,
"x" => K::KeyX,
@ -95,6 +96,7 @@ pub fn translate_key(name: &str) -> events::Key {
"7" => K::Key7,
"8" => K::Key8,
"9" => K::Key9,
"Enter" => K::KeyEnter,
_ => K::UnknownKey,
}
}

View File

@ -1,3 +1,5 @@
use layers::PolyLine;
use crate::{
layers::{self, Folder, Layer, LayerData, LayerDataTypes, Line, Rect, Shape},
DocumentError, LayerId, Operation,
@ -178,6 +180,12 @@ impl Document {
update_frontend(self.render(&mut vec![]));
}
Operation::AddPen { path, insert_index, points, style } => {
let points: Vec<kurbo::Point> = points.into_iter().map(|it| it.into()).collect();
let polyline = PolyLine::new(points, style);
self.add_layer(&path, Layer::new(LayerDataTypes::PolyLine(polyline)), insert_index)?;
update_frontend(self.render(&mut vec![]));
}
Operation::AddShape {
path,
insert_index,

View File

@ -9,6 +9,9 @@ pub use line::Line;
pub mod rect;
pub use rect::Rect;
pub mod polyline;
pub use polyline::PolyLine;
pub mod shape;
pub use shape::Shape;
@ -25,6 +28,7 @@ pub enum LayerDataTypes {
Circle(Circle),
Rect(Rect),
Line(Line),
PolyLine(PolyLine),
Shape(Shape),
}
@ -35,6 +39,7 @@ impl LayerDataTypes {
Self::Circle(c) => c.render(),
Self::Rect(r) => r.render(),
Self::Line(l) => l.render(),
Self::PolyLine(pl) => pl.render(),
Self::Shape(s) => s.render(),
}
}

View File

@ -0,0 +1,24 @@
use super::style;
use super::LayerData;
#[derive(Debug, Clone, PartialEq)]
pub struct PolyLine {
points: Vec<kurbo::Point>,
style: style::PathStyle,
}
impl PolyLine {
pub fn new(points: Vec<impl Into<kurbo::Point>>, style: style::PathStyle) -> PolyLine {
PolyLine {
points: points.into_iter().map(|it| it.into()).collect(),
style,
}
}
}
impl LayerData for PolyLine {
fn render(&self) -> String {
let points = self.points.iter().map(|p| format!("{},{}", p.x, p.y)).collect::<Vec<_>>().join(" ");
format!(r#"<polyline points="{}" {}" />"#, points, self.style.render())
}
}

View File

@ -3,14 +3,20 @@ use crate::color::Color;
#[repr(C)]
#[derive(Debug, Clone, Copy, PartialEq, Default)]
pub struct Fill {
color: Color,
color: Option<Color>,
}
impl Fill {
pub fn new(color: Color) -> Self {
Self { color }
Self { color: Some(color) }
}
pub fn none() -> Self {
Self { color: None }
}
pub fn render(&self) -> String {
format!("fill: #{};", self.color.as_hex())
match self.color {
Some(c) => format!("fill: #{};", c.as_hex()),
None => format!("fill: none;"),
}
}
}

View File

@ -28,6 +28,12 @@ pub enum Operation {
y1: f64,
style: style::PathStyle,
},
AddPen {
path: Vec<LayerId>,
insert_index: isize,
points: Vec<(f64, f64)>,
style: style::PathStyle,
},
AddShape {
path: Vec<LayerId>,
insert_index: isize,

View File

@ -112,10 +112,12 @@ pub enum Key {
KeyM,
KeyE,
KeyL,
KeyP,
KeyV,
KeyX,
KeyZ,
KeyY,
KeyEnter,
Key0,
Key1,
Key2,

View File

@ -1,15 +1,97 @@
use crate::events::{Event, Response};
use crate::tools::Tool;
use crate::events::{Key, MouseKeys, ViewportPosition};
use crate::tools::{Fsm, Tool};
use crate::Document;
use document_core::layers::style;
use document_core::Operation;
use super::DocumentToolData;
#[derive(Default)]
pub struct Pen;
pub struct Pen {
fsm_state: PenToolFsmState,
data: PenToolData,
}
impl Tool for Pen {
fn handle_input(&mut self, event: &Event, document: &Document, tool_data: &DocumentToolData) -> (Vec<Response>, Vec<Operation>) {
todo!("{}::handle_input {:?} {:?} {:?}", module_path!(), event, document, tool_data)
let mut responses = Vec::new();
let mut operations = Vec::new();
self.fsm_state = self.fsm_state.transition(event, document, tool_data, &mut self.data, &mut responses, &mut operations);
(responses, operations)
}
}
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
enum PenToolFsmState {
Ready,
LmbDown,
}
impl Default for PenToolFsmState {
fn default() -> Self {
PenToolFsmState::Ready
}
}
#[derive(Clone, Debug, Default)]
struct PenToolData {
points: Vec<ViewportPosition>,
}
impl Fsm for PenToolFsmState {
type ToolData = PenToolData;
fn transition(self, event: &Event, document: &Document, tool_data: &DocumentToolData, data: &mut Self::ToolData, _responses: &mut Vec<Response>, operations: &mut Vec<Operation>) -> Self {
let stroke = style::Stroke::new(tool_data.primary_color, 5.);
let fill = style::Fill::none();
let style = style::PathStyle::new(Some(stroke), Some(fill));
match (self, event) {
(PenToolFsmState::Ready, Event::MouseDown(mouse_state)) if mouse_state.mouse_keys.contains(MouseKeys::LEFT) => {
operations.push(Operation::MountWorkingFolder { path: vec![] });
data.points.push(mouse_state.position);
PenToolFsmState::LmbDown
}
(PenToolFsmState::Ready, Event::KeyDown(Key::KeyZ)) => {
if let Some(id) = document.root.list_layers().last() {
operations.push(Operation::DeleteLayer { path: vec![*id] })
}
PenToolFsmState::Ready
}
// TODO - Check for left mouse button
(PenToolFsmState::LmbDown, Event::MouseDown(mouse_state)) => {
data.points.push(mouse_state.position);
PenToolFsmState::LmbDown
}
(PenToolFsmState::LmbDown, Event::MouseMove(mouse_state)) => {
let mut points: Vec<_> = data.points.iter().map(|p| (p.x as f64, p.y as f64)).collect();
points.push((mouse_state.x as f64, mouse_state.y as f64));
operations.push(Operation::ClearWorkingFolder);
operations.push(Operation::AddPen {
path: vec![],
insert_index: -1,
points,
style,
});
PenToolFsmState::LmbDown
}
(PenToolFsmState::LmbDown, Event::KeyDown(Key::KeyEnter)) => {
let points = data.points.drain(..).map(|p| (p.x as f64, p.y as f64)).collect();
operations.push(Operation::ClearWorkingFolder);
operations.push(Operation::AddPen {
path: vec![],
insert_index: -1,
points,
style,
});
operations.push(Operation::CommitTransaction);
PenToolFsmState::Ready
}
_ => self,
}
}
}