Skip to content

Commit 8e3d237

Browse files
otdaviesTrueDoctor
andauthored
Ungroup layers (#465)
* WIP handling corner cases, like ungrouping subfolders * Resolved hanging * Fix recursive ungrouping * Functional, corner case free Ungroup. Small Undo issue & warnings * Update layertree upon undo * Also update layerdata upon redo * Add some polish * Resolved TODOs * Oops didn't save all after rename, ha. Co-authored-by: Dennis <[email protected]>
1 parent c1c7192 commit 8e3d237

File tree

7 files changed

+111
-35
lines changed

7 files changed

+111
-35
lines changed

editor/src/communication/dispatcher.rs

Lines changed: 4 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -339,7 +339,7 @@ mod test {
339339
init_logger();
340340
let mut editor = create_editor_with_three_layers();
341341

342-
fn map_to_vec<'a>(paths: Vec<&'a [LayerId]>) -> Vec<Vec<LayerId>> {
342+
fn map_to_vec(paths: Vec<&[LayerId]>) -> Vec<Vec<LayerId>> {
343343
paths.iter().map(|layer| layer.to_vec()).collect::<Vec<_>>()
344344
}
345345
let sorted_layers = map_to_vec(editor.dispatcher.documents_message_handler.active_document().all_layers_sorted());
@@ -356,15 +356,15 @@ mod test {
356356
editor.handle_message(DocumentMessage::SetSelectedLayers(sorted_layers[..2].to_vec()));
357357

358358
editor.handle_message(DocumentMessage::ReorderSelectedLayers(1));
359-
let (all, non_selected, selected) = verify_order(&mut editor.dispatcher.documents_message_handler.active_document_mut());
359+
let (all, non_selected, selected) = verify_order(editor.dispatcher.documents_message_handler.active_document_mut());
360360
assert_eq!(all, non_selected.into_iter().chain(selected.into_iter()).collect::<Vec<_>>());
361361

362362
editor.handle_message(DocumentMessage::ReorderSelectedLayers(-1));
363-
let (all, non_selected, selected) = verify_order(&mut editor.dispatcher.documents_message_handler.active_document_mut());
363+
let (all, non_selected, selected) = verify_order(editor.dispatcher.documents_message_handler.active_document_mut());
364364
assert_eq!(all, selected.into_iter().chain(non_selected.into_iter()).collect::<Vec<_>>());
365365

366366
editor.handle_message(DocumentMessage::ReorderSelectedLayers(i32::MAX));
367-
let (all, non_selected, selected) = verify_order(&mut editor.dispatcher.documents_message_handler.active_document_mut());
367+
let (all, non_selected, selected) = verify_order(editor.dispatcher.documents_message_handler.active_document_mut());
368368
assert_eq!(all, non_selected.into_iter().chain(selected.into_iter()).collect::<Vec<_>>());
369369
}
370370
}

editor/src/document/document_file.rs

Lines changed: 56 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -139,6 +139,8 @@ pub enum DocumentMessage {
139139
StartTransaction,
140140
RollbackTransaction,
141141
GroupSelectedLayers,
142+
UngroupSelectedLayers,
143+
UngroupLayers(Vec<LayerId>),
142144
AbortTransaction,
143145
CommitTransaction,
144146
ExportDocument,
@@ -217,9 +219,14 @@ impl DocumentMessageHandler {
217219
fn select_layer(&mut self, path: &[LayerId]) -> Option<Message> {
218220
println!("Select_layer fail: {:?}", self.all_layers_sorted());
219221

220-
self.layer_metadata_mut(path).selected = true;
221-
let data = self.layer_panel_entry(path.to_vec()).ok()?;
222-
(!path.is_empty()).then(|| FrontendMessage::UpdateLayer { data }.into())
222+
if let Some(layer) = self.layer_metadata.get_mut(path) {
223+
layer.selected = true;
224+
let data = self.layer_panel_entry(path.to_vec()).ok()?;
225+
(!path.is_empty()).then(|| FrontendMessage::UpdateLayer { data }.into())
226+
} else {
227+
log::warn!("Tried to select non existing layer {:?}", path);
228+
None
229+
}
223230
}
224231

225232
pub fn selected_visible_layers_bounding_box(&self) -> Option<[DVec2; 2]> {
@@ -274,13 +281,10 @@ impl DocumentMessageHandler {
274281
}
275282

276283
pub fn selected_layers_without_children(&self) -> Vec<&[LayerId]> {
277-
let mut sorted_layers = self.selected_layers().collect::<Vec<_>>();
278-
// Sorting here creates groups of similar UUID paths
279-
sorted_layers.sort();
280-
sorted_layers.dedup_by(|a, b| a.starts_with(b));
284+
let unique_layers = self.graphene_document.shallowest_unique_layers(self.selected_layers());
281285

282286
// We need to maintain layer ordering
283-
self.sort_layers(sorted_layers.iter().copied())
287+
self.sort_layers(unique_layers.iter().copied())
284288
}
285289

286290
pub fn selected_layers_contains(&self, path: &[LayerId]) -> bool {
@@ -427,6 +431,9 @@ impl DocumentMessageHandler {
427431
let document = std::mem::replace(&mut self.graphene_document, document);
428432
let layer_metadata = std::mem::replace(&mut self.layer_metadata, layer_metadata);
429433
self.document_redo_history.push((document, layer_metadata));
434+
for layer in self.layer_metadata.keys() {
435+
responses.push_back(DocumentMessage::LayerChanged(layer.clone()).into())
436+
}
430437
Ok(())
431438
}
432439
None => Err(EditorError::NoTransactionInProgress),
@@ -442,6 +449,9 @@ impl DocumentMessageHandler {
442449
let document = std::mem::replace(&mut self.graphene_document, document);
443450
let layer_metadata = std::mem::replace(&mut self.layer_metadata, layer_metadata);
444451
self.document_undo_history.push((document, layer_metadata));
452+
for layer in self.layer_metadata.keys() {
453+
responses.push_back(DocumentMessage::LayerChanged(layer.clone()).into())
454+
}
445455
Ok(())
446456
}
447457
None => Err(EditorError::NoTransactionInProgress),
@@ -470,7 +480,10 @@ impl DocumentMessageHandler {
470480
}
471481

472482
pub fn layer_panel_entry(&mut self, path: Vec<LayerId>) -> Result<LayerPanelEntry, EditorError> {
473-
let data: LayerMetadata = *self.layer_metadata_mut(&path);
483+
let data: LayerMetadata = *self
484+
.layer_metadata
485+
.get_mut(&path)
486+
.ok_or_else(|| EditorError::Document(format!("Could not get layer metadata for {:?}", path)))?;
474487
let layer = self.graphene_document.layer(&path)?;
475488
let entry = layer_panel_entry(&data, self.graphene_document.multiply_transforms(&path)?, layer, path);
476489
Ok(entry)
@@ -505,7 +518,7 @@ impl MessageHandler<DocumentMessage, &InputPreprocessor> for DocumentMessageHand
505518
TransformLayers(message) => self
506519
.transform_layer_handler
507520
.process_action(message, (&mut self.layer_metadata, &mut self.graphene_document, ipp), responses),
508-
DeleteLayer(path) => responses.push_back(DocumentOperation::DeleteLayer { path }.into()),
521+
DeleteLayer(path) => responses.push_front(DocumentOperation::DeleteLayer { path }.into()),
509522
StartTransaction => self.backup(responses),
510523
RollbackTransaction => {
511524
self.rollback(responses).unwrap_or_else(|e| log::warn!("{}", e));
@@ -595,6 +608,37 @@ impl MessageHandler<DocumentMessage, &InputPreprocessor> for DocumentMessageHand
595608
);
596609
responses.push_back(DocumentMessage::SetSelectedLayers(vec![new_folder_path]).into());
597610
}
611+
UngroupLayers(folder_path) => {
612+
// Select all the children of the folder
613+
let to_select = self.graphene_document.folder_children_paths(&folder_path);
614+
let message_buffer = [
615+
// Copy them
616+
DocumentMessage::SetSelectedLayers(to_select).into(),
617+
DocumentsMessage::Copy(Clipboard::System).into(),
618+
// Paste them into the folder above
619+
DocumentsMessage::PasteIntoFolder {
620+
clipboard: Clipboard::System,
621+
path: folder_path[..folder_path.len() - 1].to_vec(),
622+
insert_index: -1,
623+
}
624+
.into(),
625+
// Delete parent folder
626+
DocumentMessage::DeleteLayer(folder_path).into(),
627+
];
628+
629+
// Push these messages in reverse due to push_front
630+
for message in message_buffer.into_iter().rev() {
631+
responses.push_front(message);
632+
}
633+
}
634+
UngroupSelectedLayers => {
635+
responses.push_back(DocumentMessage::StartTransaction.into());
636+
let folder_paths = self.graphene_document.sorted_folders_by_depth(self.selected_layers());
637+
for folder_path in folder_paths {
638+
responses.push_back(DocumentMessage::UngroupLayers(folder_path.to_vec()).into());
639+
}
640+
responses.push_back(DocumentMessage::CommitTransaction.into());
641+
}
598642
SetBlendModeForSelectedLayers(blend_mode) => {
599643
self.backup(responses);
600644
for path in self.layer_metadata.iter().filter_map(|(path, data)| data.selected.then(|| path.clone())) {
@@ -864,7 +908,7 @@ impl MessageHandler<DocumentMessage, &InputPreprocessor> for DocumentMessageHand
864908
let insert = all_layer_paths.get(insert_pos);
865909
if let Some(insert_path) = insert {
866910
let (id, path) = insert_path.split_last().expect("Can't move the root folder");
867-
if let Some(folder) = self.graphene_document.layer(path).ok().map(|layer| layer.as_folder().ok()).flatten() {
911+
if let Some(folder) = self.graphene_document.layer(path).ok().and_then(|layer| layer.as_folder().ok()) {
868912
let selected: Vec<_> = selected_layers
869913
.iter()
870914
.filter(|layer| layer.starts_with(path) && layer.len() == path.len() + 1)
@@ -1003,6 +1047,7 @@ impl MessageHandler<DocumentMessage, &InputPreprocessor> for DocumentMessageHand
10031047
NudgeSelectedLayers,
10041048
ReorderSelectedLayers,
10051049
GroupSelectedLayers,
1050+
UngroupSelectedLayers,
10061051
);
10071052
common.extend(select);
10081053
}

editor/src/document/document_message_handler.rs

Lines changed: 11 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -394,29 +394,29 @@ impl MessageHandler<DocumentsMessage, &InputPreprocessor> for DocumentsMessageHa
394394

395395
let destination_path = [path.to_vec(), vec![generate_uuid()]].concat();
396396

397-
responses.push_back(
398-
DocumentOperation::InsertLayer {
399-
layer: entry.layer.clone(),
400-
destination_path: destination_path.clone(),
401-
insert_index,
397+
responses.push_front(
398+
DocumentMessage::UpdateLayerMetadata {
399+
layer_path: destination_path.clone(),
400+
layer_metadata: entry.layer_metadata,
402401
}
403402
.into(),
404403
);
405-
responses.push_back(
406-
DocumentMessage::UpdateLayerMetadata {
407-
layer_path: destination_path,
408-
layer_metadata: entry.layer_metadata,
404+
responses.push_front(
405+
DocumentOperation::InsertLayer {
406+
layer: entry.layer.clone(),
407+
destination_path,
408+
insert_index,
409409
}
410410
.into(),
411411
);
412412
};
413413

414414
if insert_index == -1 {
415-
for entry in self.copy_buffer[clipboard as usize].iter() {
415+
for entry in self.copy_buffer[clipboard as usize].iter().rev() {
416416
paste(entry, responses)
417417
}
418418
} else {
419-
for entry in self.copy_buffer[clipboard as usize].iter().rev() {
419+
for entry in self.copy_buffer[clipboard as usize].iter() {
420420
paste(entry, responses)
421421
}
422422
}

editor/src/input/input_mapper.rs

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -261,6 +261,7 @@ impl Default for Mapping {
261261
entry! {action=DocumentsMessage::Copy(User), key_down=KeyC, modifiers=[KeyControl]},
262262
entry! {action=DocumentsMessage::Cut(User), key_down=KeyX, modifiers=[KeyControl]},
263263
entry! {action=DocumentMessage::GroupSelectedLayers, key_down=KeyG, modifiers=[KeyControl]},
264+
entry! {action=DocumentMessage::UngroupSelectedLayers, key_down=KeyG, modifiers=[KeyControl, KeyShift]},
264265
// Nudging
265266
entry! {action=DocumentMessage::NudgeSelectedLayers(-SHIFT_NUDGE_AMOUNT, -SHIFT_NUDGE_AMOUNT), key_down=KeyArrowUp, modifiers=[KeyShift, KeyArrowLeft]},
266267
entry! {action=DocumentMessage::NudgeSelectedLayers(SHIFT_NUDGE_AMOUNT, -SHIFT_NUDGE_AMOUNT), key_down=KeyArrowUp, modifiers=[KeyShift, KeyArrowRight]},

frontend/src/components/panels/LayerTree.vue

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,7 @@
5757
:data-index="index"
5858
draggable="true"
5959
@dragstart="dragStart($event, layer)"
60+
:title="layer.path"
6061
>
6162
<div class="layer-thumbnail" v-html="layer.thumbnail"></div>
6263
<div class="layer-type-icon">

frontend/wasm/src/lib.rs

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@ fn panic_hook(info: &panic::PanicInfo) {
3333
let panic_info = info.to_string();
3434
let title = "The editor crashed — sorry about that".to_string();
3535
let description = "An internal error occurred. Reload the editor to continue. Please report this by filing an issue on GitHub.".to_string();
36+
log::error!("{}", info);
3637
EDITOR_INSTANCES.with(|instances| {
3738
instances.borrow_mut().values_mut().for_each(|instance| {
3839
instance.1.handle_response_rust_proxy(FrontendMessage::DisplayPanic {

graphene/src/document.rs

Lines changed: 37 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -95,6 +95,15 @@ impl Document {
9595
self.folder_mut(path)?.layer_mut(id).ok_or_else(|| DocumentError::LayerNotFound(path.into()))
9696
}
9797

98+
pub fn common_layer_path_prefix<'a>(&self, layers: impl Iterator<Item = &'a [LayerId]>) -> &'a [LayerId] {
99+
layers.reduce(|a, b| &a[..a.iter().zip(b.iter()).take_while(|&(a, b)| a == b).count()]).unwrap_or_default()
100+
}
101+
102+
pub fn folders<'a>(&'a self, layers: impl Iterator<Item = &'a [LayerId]>) -> impl Iterator<Item = &'a [LayerId]> {
103+
layers.filter(|layer| self.is_folder(layer))
104+
}
105+
106+
// Returns the shallowest folder given the selection, even if the selection doesn't contain any folders
98107
pub fn shallowest_common_folder<'a>(&self, layers: impl Iterator<Item = &'a [LayerId]>) -> Result<&'a [LayerId], DocumentError> {
99108
let common_prefix_of_path = self.common_layer_path_prefix(layers);
100109

@@ -104,13 +113,32 @@ impl Document {
104113
})
105114
}
106115

107-
pub fn common_layer_path_prefix<'a>(&self, layers: impl Iterator<Item = &'a [LayerId]>) -> &'a [LayerId] {
108-
layers
109-
.reduce(|a, b| {
110-
let number_of_uncommon_ids_in_a = (0..a.len()).position(|i| b.starts_with(&a[..a.len() - i])).unwrap_or_default();
111-
&a[..(a.len() - number_of_uncommon_ids_in_a)]
112-
})
113-
.unwrap_or_default()
116+
// Return returns all folders that are not contained in any other of the given folders
117+
pub fn shallowest_folders<'a>(&'a self, layers: impl Iterator<Item = &'a [LayerId]>) -> Vec<&[LayerId]> {
118+
self.shallowest_unique_layers(self.folders(layers))
119+
}
120+
121+
// Return returns all layers that are not contained in any other of the given folders
122+
pub fn shallowest_unique_layers<'a>(&'a self, layers: impl Iterator<Item = &'a [LayerId]>) -> Vec<&[LayerId]> {
123+
let mut sorted_layers: Vec<_> = layers.collect();
124+
sorted_layers.sort();
125+
// Sorting here creates groups of similar UUID paths
126+
sorted_layers.dedup_by(|a, b| a.starts_with(b));
127+
sorted_layers
128+
}
129+
// Deepest to shallowest (longest to shortest path length)
130+
pub fn sorted_folders_by_depth<'a>(&'a self, layers: impl Iterator<Item = &'a [LayerId]>) -> Vec<&'a [LayerId]> {
131+
let mut folders: Vec<_> = self.folders(layers).collect();
132+
folders.sort_by_key(|a| std::cmp::Reverse(a.len()));
133+
folders
134+
}
135+
136+
pub fn folder_children_paths(&self, path: &[LayerId]) -> Vec<Vec<LayerId>> {
137+
if let Ok(folder) = self.folder(path) {
138+
folder.list_layers().iter().map(|f| [path, &[*f]].concat()).collect()
139+
} else {
140+
vec![]
141+
}
114142
}
115143

116144
pub fn is_folder(&self, path: &[LayerId]) -> bool {
@@ -457,7 +485,7 @@ impl Document {
457485
};
458486
self.delete(path)?;
459487

460-
let (folder, _) = split_path(path.as_slice()).unwrap_or_else(|_| (&[], 0));
488+
let (folder, _) = split_path(path.as_slice()).unwrap_or((&[], 0));
461489
responses.extend([DocumentChanged, DeletedLayer { path: path.clone() }, FolderChanged { path: folder.to_vec() }]);
462490
responses.extend(update_thumbnails_upstream(folder));
463491
Some(responses)
@@ -494,7 +522,7 @@ impl Document {
494522
}
495523
Operation::DuplicateLayer { path } => {
496524
let layer = self.layer(path)?.clone();
497-
let (folder_path, _) = split_path(path.as_slice()).unwrap_or_else(|_| (&[], 0));
525+
let (folder_path, _) = split_path(path.as_slice()).unwrap_or((&[], 0));
498526
let folder = self.folder_mut(folder_path)?;
499527
if let Some(new_layer_id) = folder.add_layer(layer, None, -1) {
500528
let new_path = [folder_path, &[new_layer_id]].concat();

0 commit comments

Comments
 (0)