Add transactions for temporary modifications to the document (#77)
* Add transactions for temporary modifications to the document * Add DiscardTempFolder Operation * Add clearFolder operation * Rename Temp to Work
This commit is contained in:
parent
90df412aab
commit
2849b99b59
File diff suppressed because it is too large
Load Diff
|
|
@ -137,11 +137,21 @@ impl Default for Folder {
|
||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
pub struct Document {
|
pub struct Document {
|
||||||
pub root: Folder,
|
pub root: Folder,
|
||||||
|
pub work: Folder,
|
||||||
|
pub work_mount_path: Vec<LayerId>,
|
||||||
|
pub work_operations: Vec<Operation>,
|
||||||
|
pub work_mounted: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Document {
|
impl Default for Document {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Self { root: Folder::default() }
|
Self {
|
||||||
|
root: Folder::default(),
|
||||||
|
work: Folder::default(),
|
||||||
|
work_mount_path: Vec::new(),
|
||||||
|
work_operations: Vec::new(),
|
||||||
|
work_mounted: false,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -152,11 +162,60 @@ fn split_path(path: &[LayerId]) -> Result<(&[LayerId], LayerId), DocumentError>
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Document {
|
impl Document {
|
||||||
pub fn render(&self) -> String {
|
pub fn render(&self, path: &mut Vec<LayerId>) -> String {
|
||||||
self.root.render()
|
if !self.work_mount_path.as_slice().starts_with(path) {
|
||||||
|
match &self.layer(path).unwrap().data {
|
||||||
|
LayerType::Folder(_) => (),
|
||||||
|
element => {
|
||||||
|
path.pop();
|
||||||
|
return element.render();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if path.as_slice() == self.work_mount_path {
|
||||||
|
let mut out = self.document_folder(path).unwrap().render();
|
||||||
|
out += self.work.render().as_str();
|
||||||
|
path.pop();
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
let mut out = String::with_capacity(30);
|
||||||
|
for element in self.folder(path).unwrap().layer_ids.iter() {
|
||||||
|
path.push(*element);
|
||||||
|
out += self.render(path).as_str();
|
||||||
|
}
|
||||||
|
out
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn folder(&self, path: &[LayerId]) -> Result<&Folder, DocumentError> {
|
fn is_mounted(&self, mount_path: &[LayerId], path: &[LayerId]) -> bool {
|
||||||
|
path.starts_with(mount_path) && self.work_mounted
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn folder(&self, mut path: &[LayerId]) -> Result<&Folder, DocumentError> {
|
||||||
|
let mut root = &self.root;
|
||||||
|
if self.is_mounted(self.work_mount_path.as_slice(), path) {
|
||||||
|
path = &path[self.work_mount_path.len()..];
|
||||||
|
root = &self.work;
|
||||||
|
}
|
||||||
|
for id in path {
|
||||||
|
root = root.folder(*id).ok_or(DocumentError::LayerNotFound)?;
|
||||||
|
}
|
||||||
|
Ok(root)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn folder_mut(&mut self, mut path: &[LayerId]) -> Result<&mut Folder, DocumentError> {
|
||||||
|
let mut root = if self.is_mounted(self.work_mount_path.as_slice(), path) {
|
||||||
|
path = &path[self.work_mount_path.len()..];
|
||||||
|
&mut self.work
|
||||||
|
} else {
|
||||||
|
&mut self.root
|
||||||
|
};
|
||||||
|
for id in path {
|
||||||
|
root = root.folder_mut(*id).ok_or(DocumentError::LayerNotFound)?;
|
||||||
|
}
|
||||||
|
Ok(root)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn document_folder(&self, path: &[LayerId]) -> Result<&Folder, DocumentError> {
|
||||||
let mut root = &self.root;
|
let mut root = &self.root;
|
||||||
for id in path {
|
for id in path {
|
||||||
root = root.folder(*id).ok_or(DocumentError::LayerNotFound)?;
|
root = root.folder(*id).ok_or(DocumentError::LayerNotFound)?;
|
||||||
|
|
@ -164,7 +223,7 @@ impl Document {
|
||||||
Ok(root)
|
Ok(root)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn folder_mut(&mut self, path: &[LayerId]) -> Result<&mut Folder, DocumentError> {
|
pub fn document_folder_mut(&mut self, path: &[LayerId]) -> Result<&mut Folder, DocumentError> {
|
||||||
let mut root = &mut self.root;
|
let mut root = &mut self.root;
|
||||||
for id in path {
|
for id in path {
|
||||||
root = root.folder_mut(*id).ok_or(DocumentError::LayerNotFound)?;
|
root = root.folder_mut(*id).ok_or(DocumentError::LayerNotFound)?;
|
||||||
|
|
@ -208,22 +267,23 @@ impl Document {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn handle_operation<F: Fn(String)>(&mut self, operation: Operation, update_frontend: F) -> Result<(), DocumentError> {
|
pub fn handle_operation<F: Fn(String)>(&mut self, operation: Operation, update_frontend: &F) -> Result<(), DocumentError> {
|
||||||
|
self.work_operations.push(operation.clone());
|
||||||
match operation {
|
match operation {
|
||||||
Operation::AddCircle { path, insert_index, cx, cy, r } => {
|
Operation::AddCircle { path, insert_index, cx, cy, r } => {
|
||||||
self.add_layer(&path, Layer::new(LayerType::Circle(Circle::new(Point::new(cx, cy), r))), insert_index)?;
|
self.add_layer(&path, Layer::new(LayerType::Circle(Circle::new(Point::new(cx, cy), r))), insert_index)?;
|
||||||
|
|
||||||
update_frontend(self.render());
|
update_frontend(self.render(&mut vec![]));
|
||||||
}
|
}
|
||||||
Operation::AddRect { path, insert_index, x0, y0, x1, y1 } => {
|
Operation::AddRect { path, insert_index, x0, y0, x1, y1 } => {
|
||||||
self.add_layer(&path, Layer::new(LayerType::Rect(Rect::from_points(Point::new(x0, y0), Point::new(x1, y1)))), insert_index)?;
|
self.add_layer(&path, Layer::new(LayerType::Rect(Rect::from_points(Point::new(x0, y0), Point::new(x1, y1)))), insert_index)?;
|
||||||
|
|
||||||
update_frontend(self.render());
|
update_frontend(self.render(&mut vec![]));
|
||||||
}
|
}
|
||||||
Operation::AddLine { path, insert_index, x0, y0, x1, y1 } => {
|
Operation::AddLine { path, insert_index, x0, y0, x1, y1 } => {
|
||||||
self.add_layer(&path, Layer::new(LayerType::Line(Line::new(Point::new(x0, y0), Point::new(x1, y1)))), insert_index)?;
|
self.add_layer(&path, Layer::new(LayerType::Line(Line::new(Point::new(x0, y0), Point::new(x1, y1)))), insert_index)?;
|
||||||
|
|
||||||
update_frontend(self.render());
|
update_frontend(self.render(&mut vec![]));
|
||||||
}
|
}
|
||||||
Operation::AddShape {
|
Operation::AddShape {
|
||||||
path,
|
path,
|
||||||
|
|
@ -237,14 +297,43 @@ impl Document {
|
||||||
let s = shape_points::ShapePoints::new(Point::new(x0, y0), Vec2 { x: x0 - x1, y: y0 - y1 }, sides);
|
let s = shape_points::ShapePoints::new(Point::new(x0, y0), Vec2 { x: x0 - x1, y: y0 - y1 }, sides);
|
||||||
self.add_layer(&path, Layer::new(LayerType::Shape(s)), insert_index)?;
|
self.add_layer(&path, Layer::new(LayerType::Shape(s)), insert_index)?;
|
||||||
|
|
||||||
update_frontend(self.render());
|
update_frontend(self.render(&mut vec![]));
|
||||||
}
|
}
|
||||||
Operation::DeleteLayer { path } => {
|
Operation::DeleteLayer { path } => {
|
||||||
self.delete(&path)?;
|
self.delete(&path)?;
|
||||||
|
|
||||||
update_frontend(self.render());
|
update_frontend(self.render(&mut vec![]));
|
||||||
}
|
}
|
||||||
Operation::AddFolder { path } => self.set_layer(&path, Layer::new(LayerType::Folder(Folder::default())))?,
|
Operation::AddFolder { path } => self.set_layer(&path, Layer::new(LayerType::Folder(Folder::default())))?,
|
||||||
|
Operation::MountWorkingFolder { path } => {
|
||||||
|
self.work_operations.clear();
|
||||||
|
self.work_mount_path = path;
|
||||||
|
self.work = Folder::default();
|
||||||
|
self.work_mounted = true;
|
||||||
|
}
|
||||||
|
Operation::DiscardWorkingFolder => {
|
||||||
|
self.work_operations.clear();
|
||||||
|
self.work_mount_path = vec![];
|
||||||
|
self.work = Folder::default();
|
||||||
|
self.work_mounted = false;
|
||||||
|
}
|
||||||
|
Operation::ClearWorkingFolder => {
|
||||||
|
self.work_operations.clear();
|
||||||
|
self.work = Folder::default();
|
||||||
|
}
|
||||||
|
Operation::CommitTransaction => {
|
||||||
|
let mut ops = Vec::new();
|
||||||
|
std::mem::swap(&mut ops, &mut self.work_operations);
|
||||||
|
let len = ops.len() - 1;
|
||||||
|
self.work_mounted = false;
|
||||||
|
self.work_mount_path = vec![];
|
||||||
|
self.work = Folder::default();
|
||||||
|
for operation in ops.into_iter().take(len) {
|
||||||
|
self.handle_operation(operation, update_frontend)?
|
||||||
|
}
|
||||||
|
|
||||||
|
update_frontend(self.render(&mut vec![]));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,6 @@
|
||||||
use crate::LayerId;
|
use crate::LayerId;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
pub enum Operation {
|
pub enum Operation {
|
||||||
AddCircle {
|
AddCircle {
|
||||||
path: Vec<LayerId>,
|
path: Vec<LayerId>,
|
||||||
|
|
@ -39,4 +40,10 @@ pub enum Operation {
|
||||||
AddFolder {
|
AddFolder {
|
||||||
path: Vec<LayerId>,
|
path: Vec<LayerId>,
|
||||||
},
|
},
|
||||||
|
MountWorkingFolder {
|
||||||
|
path: Vec<LayerId>,
|
||||||
|
},
|
||||||
|
DiscardWorkingFolder,
|
||||||
|
ClearWorkingFolder,
|
||||||
|
CommitTransaction,
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -112,7 +112,7 @@ impl Dispatcher {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn dispatch_operation(&self, document: &mut Document, operation: Operation) -> Result<(), EditorError> {
|
fn dispatch_operation(&self, document: &mut Document, operation: Operation) -> Result<(), EditorError> {
|
||||||
document.handle_operation(operation, |svg: String| self.dispatch_response(Response::UpdateCanvas { document: svg }))?;
|
document.handle_operation(operation, &|svg: String| self.dispatch_response(Response::UpdateCanvas { document: svg }))?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -43,8 +43,10 @@ impl Fsm for EllipseToolFsmState {
|
||||||
match (self, event) {
|
match (self, event) {
|
||||||
(EllipseToolFsmState::Ready, Event::MouseDown(mouse_state)) if mouse_state.mouse_keys.contains(MouseKeys::LEFT) => {
|
(EllipseToolFsmState::Ready, Event::MouseDown(mouse_state)) if mouse_state.mouse_keys.contains(MouseKeys::LEFT) => {
|
||||||
data.drag_start = mouse_state.position;
|
data.drag_start = mouse_state.position;
|
||||||
|
operations.push(Operation::MountWorkingFolder { path: vec![] });
|
||||||
EllipseToolFsmState::LmbDown
|
EllipseToolFsmState::LmbDown
|
||||||
}
|
}
|
||||||
|
|
||||||
(EllipseToolFsmState::Ready, Event::KeyDown(Key::KeyZ)) => {
|
(EllipseToolFsmState::Ready, Event::KeyDown(Key::KeyZ)) => {
|
||||||
if let Some(id) = document.root.list_layers().last() {
|
if let Some(id) = document.root.list_layers().last() {
|
||||||
operations.push(Operation::DeleteLayer { path: vec![*id] })
|
operations.push(Operation::DeleteLayer { path: vec![*id] })
|
||||||
|
|
@ -52,10 +54,24 @@ impl Fsm for EllipseToolFsmState {
|
||||||
EllipseToolFsmState::Ready
|
EllipseToolFsmState::Ready
|
||||||
}
|
}
|
||||||
|
|
||||||
|
(EllipseToolFsmState::LmbDown, Event::MouseMove(mouse_state)) => {
|
||||||
|
operations.push(Operation::ClearWorkingFolder);
|
||||||
|
operations.push(Operation::AddCircle {
|
||||||
|
path: vec![],
|
||||||
|
insert_index: -1,
|
||||||
|
cx: data.drag_start.x as f64,
|
||||||
|
cy: data.drag_start.y as f64,
|
||||||
|
r: data.drag_start.distance(&mouse_state),
|
||||||
|
});
|
||||||
|
|
||||||
|
EllipseToolFsmState::LmbDown
|
||||||
|
}
|
||||||
|
|
||||||
// TODO - Check for left mouse button
|
// TODO - Check for left mouse button
|
||||||
(EllipseToolFsmState::LmbDown, Event::MouseUp(mouse_state)) => {
|
(EllipseToolFsmState::LmbDown, Event::MouseUp(mouse_state)) => {
|
||||||
let r = data.drag_start.distance(&mouse_state.position);
|
let r = data.drag_start.distance(&mouse_state.position);
|
||||||
log::info!("draw ellipse with radius: {:.2}", r);
|
log::info!("draw ellipse with radius: {:.2}", r);
|
||||||
|
operations.push(Operation::ClearWorkingFolder);
|
||||||
operations.push(Operation::AddCircle {
|
operations.push(Operation::AddCircle {
|
||||||
path: vec![],
|
path: vec![],
|
||||||
insert_index: -1,
|
insert_index: -1,
|
||||||
|
|
@ -63,6 +79,7 @@ impl Fsm for EllipseToolFsmState {
|
||||||
cy: data.drag_start.y as f64,
|
cy: data.drag_start.y as f64,
|
||||||
r: data.drag_start.distance(&mouse_state.position),
|
r: data.drag_start.distance(&mouse_state.position),
|
||||||
});
|
});
|
||||||
|
operations.push(Operation::CommitTransaction);
|
||||||
|
|
||||||
EllipseToolFsmState::Ready
|
EllipseToolFsmState::Ready
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue