mirror of
https://github.com/GraphiteEditor/Graphite.git
synced 2025-12-23 10:11:54 +00:00
Merge f732d3b123 into 64ed21f694
This commit is contained in:
commit
be30905862
1 changed files with 189 additions and 1 deletions
|
|
@ -147,6 +147,7 @@ pub enum PathToolMessage {
|
|||
Duplicate,
|
||||
TogglePointEditing,
|
||||
ToggleSegmentEditing,
|
||||
MergeSelectedSegments,
|
||||
}
|
||||
|
||||
#[derive(PartialEq, Eq, Hash, Copy, Clone, Debug, Default, serde::Serialize, serde::Deserialize, specta::Type)]
|
||||
|
|
@ -329,6 +330,12 @@ impl LayoutHolder for PathTool {
|
|||
.disabled(!self.tool_data.make_path_editable_is_allowed)
|
||||
.widget_instance();
|
||||
|
||||
let merge_segments_button = IconButton::new("Folder", 24)
|
||||
.tooltip("Merge selected selgments")
|
||||
.on_update(|_| PathToolMessage::MergeSelectedSegments.into())
|
||||
.disabled(!self.tool_data.merging_segments_enabled)
|
||||
.widget_holder();
|
||||
|
||||
let [_checkbox, _dropdown] = {
|
||||
let pivot_gizmo_type_widget = pivot_gizmo_type_widget(self.tool_data.pivot_gizmo.state, PivotToolSource::Path);
|
||||
[pivot_gizmo_type_widget[0].clone(), pivot_gizmo_type_widget[2].clone()]
|
||||
|
|
@ -360,6 +367,9 @@ impl LayoutHolder for PathTool {
|
|||
path_overlay_mode_widget,
|
||||
unrelated_seperator.clone(),
|
||||
path_node_button,
|
||||
unrelated_seperator.clone(),
|
||||
merge_segments_button,
|
||||
unrelated_seperator.clone(),
|
||||
// checkbox.clone(),
|
||||
// related_seperator.clone(),
|
||||
// dropdown.clone(),
|
||||
|
|
@ -455,7 +465,8 @@ impl<'a> MessageHandler<ToolMessage, &mut ToolActionMessageContext<'a>> for Path
|
|||
Paste,
|
||||
Duplicate,
|
||||
TogglePointEditing,
|
||||
ToggleSegmentEditing
|
||||
ToggleSegmentEditing,
|
||||
MergeSelectedSegments
|
||||
),
|
||||
PathToolFsmState::Dragging(_) => actions!(PathToolMessageDiscriminant;
|
||||
Escape,
|
||||
|
|
@ -604,6 +615,7 @@ struct PathToolData {
|
|||
hovered_layers: Vec<LayerNodeIdentifier>,
|
||||
ghost_outline: Vec<(Vec<ClickTargetType>, LayerNodeIdentifier)>,
|
||||
make_path_editable_is_allowed: bool,
|
||||
merging_segments_enabled: bool,
|
||||
}
|
||||
|
||||
impl PathToolData {
|
||||
|
|
@ -665,6 +677,50 @@ impl PathToolData {
|
|||
self.selection_status = selection_status;
|
||||
}
|
||||
|
||||
fn update_merge_segments_toggle(&mut self, shape_editor: &mut ShapeState, document: &DocumentMessageHandler, vector_meshes: bool) {
|
||||
if !vector_meshes {
|
||||
self.merging_segments_enabled = false;
|
||||
return;
|
||||
}
|
||||
|
||||
let mut non_empty_layers = shape_editor.selected_shape_state.iter().filter(|(_, state)| !state.is_empty());
|
||||
let Some((layer, _)) = non_empty_layers.next() else {
|
||||
self.merging_segments_enabled = false;
|
||||
return;
|
||||
};
|
||||
|
||||
if non_empty_layers.next().is_some() {
|
||||
self.merging_segments_enabled = false;
|
||||
return;
|
||||
}
|
||||
|
||||
let segments = shape_editor.selected_segments().collect::<Vec<_>>();
|
||||
if segments.len() != 2 {
|
||||
self.merging_segments_enabled = false;
|
||||
return;
|
||||
}
|
||||
let segment1 = segments[0];
|
||||
let segment2 = segments[1];
|
||||
// Both of the segments should be close to each other
|
||||
let Some(vector) = document.network_interface.compute_modified_vector(*layer) else { return };
|
||||
let Some((_, bezier1, _, _)) = vector.segment_bezier_iter().find(|(id, _, _, _)| id == segment1) else {
|
||||
return;
|
||||
};
|
||||
let Some((_, bezier2, _, _)) = vector.segment_bezier_iter().find(|(id, _, _, _)| id == segment2) else {
|
||||
return;
|
||||
};
|
||||
let segments_in_same_direction = bezier1.start.distance(bezier2.start) < SEGMENT_INSERTION_DISTANCE && bezier1.end.distance(bezier2.end) < SEGMENT_INSERTION_DISTANCE;
|
||||
let segments_in_opposite_direction = bezier1.start.distance(bezier2.end) < SEGMENT_INSERTION_DISTANCE && bezier1.end.distance(bezier2.start) < SEGMENT_INSERTION_DISTANCE;
|
||||
let near_segments = segments_in_same_direction || segments_in_opposite_direction;
|
||||
|
||||
if !near_segments {
|
||||
self.merging_segments_enabled = false;
|
||||
return;
|
||||
}
|
||||
|
||||
self.merging_segments_enabled = true;
|
||||
}
|
||||
|
||||
fn remove_saved_points(&mut self) {
|
||||
self.saved_points_before_anchor_select_toggle.clear();
|
||||
}
|
||||
|
|
@ -3122,6 +3178,137 @@ impl Fsm for PathToolFsmState {
|
|||
}
|
||||
PathToolFsmState::Ready
|
||||
}
|
||||
(_, PathToolMessage::MergeSelectedSegments) => {
|
||||
// Assuming that two segments selected are on the same layer
|
||||
let mut non_empty_layers = shape_editor.selected_shape_state.iter().filter(|(_, state)| !state.is_empty());
|
||||
let Some((layer, _)) = non_empty_layers.next() else {
|
||||
return PathToolFsmState::Ready;
|
||||
};
|
||||
if non_empty_layers.next().is_some() {
|
||||
return PathToolFsmState::Ready;
|
||||
}
|
||||
|
||||
let segments = shape_editor.selected_segments().collect::<Vec<_>>();
|
||||
if segments.len() != 2 {
|
||||
return PathToolFsmState::Ready;
|
||||
}
|
||||
|
||||
let segment1 = segments[0];
|
||||
let segment2 = segments[1];
|
||||
|
||||
let Some(vector) = document.network_interface.compute_modified_vector(*layer) else {
|
||||
return PathToolFsmState::Ready;
|
||||
};
|
||||
|
||||
responses.add(DocumentMessage::StartTransaction);
|
||||
|
||||
let (_, bezier1, start1, end1) = vector.segment_bezier_iter().find(|(id, _, _, _)| id == segment1).expect(" Couldn't get segment data");
|
||||
let (_, bezier2, start2, end2) = vector.segment_bezier_iter().find(|(id, _, _, _)| id == segment2).expect(" Couldn't get segment data");
|
||||
|
||||
// Two cases
|
||||
let mut bezier2 = bezier2;
|
||||
let mut start2 = start2;
|
||||
let mut end2 = end2;
|
||||
let dist1 = bezier1.start.distance(bezier2.start);
|
||||
let dist2 = bezier1.start.distance(bezier2.end);
|
||||
|
||||
let new_id1 = PointId::generate();
|
||||
let new_id2 = PointId::generate();
|
||||
|
||||
if dist1 > dist2 {
|
||||
// Invert bezier2 and swap start2 and end2
|
||||
bezier2 = bezier2.reverse();
|
||||
let bin = start2;
|
||||
start2 = end2;
|
||||
end2 = bin;
|
||||
}
|
||||
|
||||
// newpoint1: start1 -> start2, newpoint2: end1 -> end2
|
||||
let pos1 = ManipulatorPointId::Anchor(start1).get_position(&vector).expect("No position for point");
|
||||
let pos2 = ManipulatorPointId::Anchor(start2).get_position(&vector).expect("No position for point");
|
||||
|
||||
let new_pos1 = (pos1 + pos2) / 2.;
|
||||
|
||||
let pos1 = ManipulatorPointId::Anchor(end1).get_position(&vector).expect("No position for point");
|
||||
let pos2 = ManipulatorPointId::Anchor(end2).get_position(&vector).expect("No position for point");
|
||||
|
||||
let new_pos2 = (pos1 + pos2) / 2.;
|
||||
|
||||
// Add the two new points
|
||||
let modification_type = VectorModificationType::InsertPoint { id: new_id1, position: new_pos1 };
|
||||
responses.add(GraphOperationMessage::Vector { layer: *layer, modification_type });
|
||||
let modification_type = VectorModificationType::InsertPoint { id: new_id2, position: new_pos2 };
|
||||
responses.add(GraphOperationMessage::Vector { layer: *layer, modification_type });
|
||||
|
||||
// Remove old points
|
||||
let points_to_remove = [start1, start2, end1, end2];
|
||||
for point in points_to_remove {
|
||||
let modification_type = VectorModificationType::RemovePoint { id: point };
|
||||
responses.add(GraphOperationMessage::Vector { layer: *layer, modification_type });
|
||||
}
|
||||
|
||||
// Connect the segments to their respective anchors
|
||||
let handles = |bezier: Bezier| -> [Option<DVec2>; 2] {
|
||||
match bezier.handles {
|
||||
BezierHandles::Linear => [None, None],
|
||||
BezierHandles::Quadratic { handle } => [Some(handle - bezier.start), None],
|
||||
BezierHandles::Cubic { handle_start, handle_end } => [Some(handle_start - bezier.start), Some(handle_end - bezier.end)],
|
||||
}
|
||||
};
|
||||
|
||||
for (_, bezier, start, end) in vector.segment_bezier_iter() {
|
||||
let id = SegmentId::generate();
|
||||
|
||||
// Connecting the segments to the start point
|
||||
if start == start1 || start == start2 {
|
||||
let points = [new_id1, end];
|
||||
let handles = handles(bezier);
|
||||
let modification_type = VectorModificationType::InsertSegment { id, points, handles };
|
||||
responses.add(GraphOperationMessage::Vector { layer: *layer, modification_type });
|
||||
} else if end == start1 || end == start2 {
|
||||
let points = [start, new_id1];
|
||||
let handles = handles(bezier);
|
||||
let modification_type = VectorModificationType::InsertSegment { id, points, handles };
|
||||
responses.add(GraphOperationMessage::Vector { layer: *layer, modification_type });
|
||||
}
|
||||
|
||||
// Connecting handles to the end point of new segement
|
||||
if start == end1 || start == end2 {
|
||||
let points = [new_id2, end];
|
||||
let handles = handles(bezier);
|
||||
let modification_type = VectorModificationType::InsertSegment { id, points, handles };
|
||||
responses.add(GraphOperationMessage::Vector { layer: *layer, modification_type });
|
||||
} else if end == end1 || end == end2 {
|
||||
let points = [start, new_id2];
|
||||
let handles = handles(bezier);
|
||||
let modification_type = VectorModificationType::InsertSegment { id, points, handles };
|
||||
responses.add(GraphOperationMessage::Vector { layer: *layer, modification_type });
|
||||
}
|
||||
}
|
||||
|
||||
// Adding the new segment
|
||||
let handles1 = handles(bezier1);
|
||||
let handles2 = handles(bezier2);
|
||||
|
||||
// Now we construct a conmbined handles for the new segment
|
||||
let mut new_handles = [Some(DVec2::ZERO), Some(DVec2::ZERO)];
|
||||
let mut index = 0;
|
||||
for (handle1, handle2) in handles1.iter().zip(handles2) {
|
||||
let pos1 = handle1.unwrap_or_default();
|
||||
let pos2 = handle2.unwrap_or_default();
|
||||
let new_pos = if pos1 == DVec2::default() && pos2 == DVec2::default() { None } else { Some((pos1 + pos2) / 2.) };
|
||||
new_handles[index] = new_pos;
|
||||
index += 1;
|
||||
}
|
||||
|
||||
let id = SegmentId::generate();
|
||||
let points = [new_id1, new_id2];
|
||||
let modification_type = VectorModificationType::InsertSegment { id, points, handles: new_handles };
|
||||
responses.add(GraphOperationMessage::Vector { layer: *layer, modification_type });
|
||||
|
||||
responses.add(DocumentMessage::EndTransaction);
|
||||
PathToolFsmState::Ready
|
||||
}
|
||||
(_, PathToolMessage::SelectedPointUpdated) => {
|
||||
let colinear = shape_editor.selected_manipulator_angles(&document.network_interface);
|
||||
tool_data.dragging_state = DraggingState {
|
||||
|
|
@ -3132,6 +3319,7 @@ impl Fsm for PathToolFsmState {
|
|||
let old = tool_data.make_path_editable_is_allowed;
|
||||
tool_data.make_path_editable_is_allowed = make_path_editable_is_allowed(&mut document.network_interface).is_some();
|
||||
tool_data.update_selection_status(shape_editor, document);
|
||||
tool_data.update_merge_segments_toggle(shape_editor, document, tool_action_data.preferences.vector_meshes);
|
||||
|
||||
if old != tool_data.make_path_editable_is_allowed {
|
||||
responses.add(MenuBarMessage::SendLayout);
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue