瀏覽代碼

Made pasting with selection active possible

flabbet 1 年之前
父節點
當前提交
7151f45aec

+ 108 - 9
src/PixiEditor/Models/DocumentModels/Public/DocumentOperationsModule.cs

@@ -44,6 +44,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(
         Internals.ActionAccumulator.AddFinishedActions(
             new SelectRectangle_Action(rect, mode),
             new SelectRectangle_Action(rect, mode),
             new EndSelectRectangle_Action());
             new EndSelectRectangle_Action());
@@ -56,6 +59,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(new ClearSelection_Action());
         Internals.ActionAccumulator.AddFinishedActions(new ClearSelection_Action());
     }
     }
 
 
@@ -68,6 +74,9 @@ internal class DocumentOperationsModule : IDocumentOperations
         var member = Document.SelectedStructureMember;
         var member = Document.SelectedStructureMember;
         if (Internals.ChangeController.IsBlockingChangeActive || member is null)
         if (Internals.ChangeController.IsBlockingChangeActive || member is null)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         bool drawOnMask = member is not ILayerHandler layer || layer.ShouldDrawOnMask;
         bool drawOnMask = member is not ILayerHandler layer || layer.ShouldDrawOnMask;
         if (drawOnMask && !member.HasMaskBindable)
         if (drawOnMask && !member.HasMaskBindable)
             return;
             return;
@@ -86,6 +95,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive || value is > 1 or < 0)
         if (Internals.ChangeController.IsBlockingChangeActive || value is > 1 or < 0)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(
         Internals.ActionAccumulator.AddFinishedActions(
             new StructureMemberOpacity_Action(memberGuid, value),
             new StructureMemberOpacity_Action(memberGuid, value),
             new EndStructureMemberOpacity_Action());
             new EndStructureMemberOpacity_Action());
@@ -111,6 +123,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddActions(new DeleteRecordedChanges_Action());
         Internals.ActionAccumulator.AddActions(new DeleteRecordedChanges_Action());
     }
     }
 
 
@@ -123,6 +138,8 @@ internal class DocumentOperationsModule : IDocumentOperations
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+
         RectI maxSize = new RectI(VecI.Zero, Document.SizeBindable);
         RectI maxSize = new RectI(VecI.Zero, Document.SizeBindable);
         foreach (var imageWithName in images)
         foreach (var imageWithName in images)
         {
         {
@@ -154,6 +171,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return null;
             return null;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         return Internals.StructureHelper.CreateNewStructureMember(type, name, finish);
         return Internals.StructureHelper.CreateNewStructureMember(type, name, finish);
     }
     }
 
 
@@ -161,7 +181,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return null;
             return null;
-        
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         return Internals.StructureHelper.CreateNewStructureMember(structureMemberType, name, finish);
         return Internals.StructureHelper.CreateNewStructureMember(structureMemberType, name, finish);
     }
     }
 
 
@@ -173,6 +195,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(new DuplicateLayer_Action(guidValue));
         Internals.ActionAccumulator.AddFinishedActions(new DuplicateLayer_Action(guidValue));
     }
     }
 
 
@@ -184,6 +209,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(new DeleteStructureMember_Action(guidValue));
         Internals.ActionAccumulator.AddFinishedActions(new DeleteStructureMember_Action(guidValue));
     }
     }
 
 
@@ -196,8 +224,10 @@ internal class DocumentOperationsModule : IDocumentOperations
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Guid closestMember = Document.StructureHelper.FindClosestMember(guids);
         Guid closestMember = Document.StructureHelper.FindClosestMember(guids);
-        
+
         IAction[] actions = new IAction[guids.Count + (selectNext ? 1 : 0)];
         IAction[] actions = new IAction[guids.Count + (selectNext ? 1 : 0)];
         for (int i = 0; i < guids.Count; i++)
         for (int i = 0; i < guids.Count; i++)
         {
         {
@@ -222,10 +252,13 @@ internal class DocumentOperationsModule : IDocumentOperations
     /// <param name="anchor">Where the existing content should be put</param>
     /// <param name="anchor">Where the existing content should be put</param>
     public void ResizeCanvas(VecI newSize, ResizeAnchor anchor)
     public void ResizeCanvas(VecI newSize, ResizeAnchor anchor)
     {
     {
-        if (Internals.ChangeController.IsBlockingChangeActive || newSize.X > 9999 || newSize.Y > 9999 || newSize.X < 1 ||
+        if (Internals.ChangeController.IsBlockingChangeActive || newSize.X > 9999 || newSize.Y > 9999 ||
+            newSize.X < 1 ||
             newSize.Y < 1)
             newSize.Y < 1)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+
         if (Document.ReferenceLayerHandler.ReferenceBitmap is not null)
         if (Document.ReferenceLayerHandler.ReferenceBitmap is not null)
         {
         {
             VecI offset = anchor.FindOffsetFor(Document.SizeBindable, newSize);
             VecI offset = anchor.FindOffsetFor(Document.SizeBindable, newSize);
@@ -251,10 +284,13 @@ internal class DocumentOperationsModule : IDocumentOperations
     /// <param name="resampling">The resampling method to use</param>
     /// <param name="resampling">The resampling method to use</param>
     public void ResizeImage(VecI newSize, ResamplingMethod resampling)
     public void ResizeImage(VecI newSize, ResamplingMethod resampling)
     {
     {
-        if (Internals.ChangeController.IsBlockingChangeActive || newSize.X > 9999 || newSize.Y > 9999 || newSize.X < 1 ||
+        if (Internals.ChangeController.IsBlockingChangeActive || newSize.X > 9999 || newSize.Y > 9999 ||
+            newSize.X < 1 ||
             newSize.Y < 1)
             newSize.Y < 1)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+
         if (Document.ReferenceLayerHandler.ReferenceBitmap is not null)
         if (Document.ReferenceLayerHandler.ReferenceBitmap is not null)
         {
         {
             VecD scale = ((VecD)newSize).Divide(Document.SizeBindable);
             VecD scale = ((VecD)newSize).Divide(Document.SizeBindable);
@@ -283,6 +319,8 @@ internal class DocumentOperationsModule : IDocumentOperations
         if (Internals.ChangeController.IsBlockingChangeActive || oldColor == newColor)
         if (Internals.ChangeController.IsBlockingChangeActive || oldColor == newColor)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(new ReplaceColor_Action(oldColor.ToColor(), newColor.ToColor(),
         Internals.ActionAccumulator.AddFinishedActions(new ReplaceColor_Action(oldColor.ToColor(), newColor.ToColor(),
             frame));
             frame));
         ReplaceInPalette(oldColor, newColor);
         ReplaceInPalette(oldColor, newColor);
@@ -305,6 +343,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         if (!member.MaskIsVisibleBindable)
         if (!member.MaskIsVisibleBindable)
             Internals.ActionAccumulator.AddActions(new StructureMemberMaskIsVisible_Action(true, member.Id));
             Internals.ActionAccumulator.AddActions(new StructureMemberMaskIsVisible_Action(true, member.Id));
         Internals.ActionAccumulator.AddFinishedActions(new CreateStructureMemberMask_Action(member.Id));
         Internals.ActionAccumulator.AddFinishedActions(new CreateStructureMemberMask_Action(member.Id));
@@ -317,6 +358,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(new DeleteStructureMemberMask_Action(member.Id));
         Internals.ActionAccumulator.AddFinishedActions(new DeleteStructureMemberMask_Action(member.Id));
     }
     }
 
 
@@ -328,6 +372,8 @@ internal class DocumentOperationsModule : IDocumentOperations
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(new ApplyMask_Action(member.Id, frame),
         Internals.ActionAccumulator.AddFinishedActions(new ApplyMask_Action(member.Id, frame),
             new DeleteStructureMemberMask_Action(member.Id));
             new DeleteStructureMemberMask_Action(member.Id));
     }
     }
@@ -418,6 +464,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive || memberToMove == memberToMoveIntoOrNextTo)
         if (Internals.ChangeController.IsBlockingChangeActive || memberToMove == memberToMoveIntoOrNextTo)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.StructureHelper.TryMoveStructureMember(memberToMove, memberToMoveIntoOrNextTo, placement);
         Internals.StructureHelper.TryMoveStructureMember(memberToMove, memberToMoveIntoOrNextTo, placement);
     }
     }
 
 
@@ -429,6 +478,8 @@ internal class DocumentOperationsModule : IDocumentOperations
         if (Internals.ChangeController.IsBlockingChangeActive || members.Count < 2)
         if (Internals.ChangeController.IsBlockingChangeActive || members.Count < 2)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+
         IStructureMemberHandler? node = Document.StructureHelper.FindNode<IStructureMemberHandler>(members[0]);
         IStructureMemberHandler? node = Document.StructureHelper.FindNode<IStructureMemberHandler>(members[0]);
 
 
         if (node is null)
         if (node is null)
@@ -470,8 +521,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     /// <param name="startPos">Where the transform should start</param>
     /// <param name="startPos">Where the transform should start</param>
     public void PasteImageWithTransform(Surface image, VecI startPos)
     public void PasteImageWithTransform(Surface image, VecI startPos)
     {
     {
-        if (Document.SelectedStructureMember is null)
+        if (Document.SelectedStructureMember is null || Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+
         Internals.ChangeController.TryStartExecutor(new PasteImageExecutor(image, startPos));
         Internals.ChangeController.TryStartExecutor(new PasteImageExecutor(image, startPos));
     }
     }
 
 
@@ -482,6 +534,11 @@ internal class DocumentOperationsModule : IDocumentOperations
     /// <param name="startPos">Where the transform should start</param>
     /// <param name="startPos">Where the transform should start</param>
     public void PasteImageWithTransform(Surface image, VecI startPos, Guid memberGuid, bool drawOnMask)
     public void PasteImageWithTransform(Surface image, VecI startPos, Guid memberGuid, bool drawOnMask)
     {
     {
+        if (Internals.ChangeController.IsBlockingChangeActive)
+            return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ChangeController.TryStartExecutor(new PasteImageExecutor(image, startPos, memberGuid, drawOnMask));
         Internals.ChangeController.TryStartExecutor(new PasteImageExecutor(image, startPos, memberGuid, drawOnMask));
     }
     }
 
 
@@ -494,6 +551,9 @@ internal class DocumentOperationsModule : IDocumentOperations
         if (Document.SelectedStructureMember is null ||
         if (Document.SelectedStructureMember is null ||
             Internals.ChangeController.IsBlockingChangeActive && !toolLinked)
             Internals.ChangeController.IsBlockingChangeActive && !toolLinked)
             return;
             return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ChangeController.TryStartExecutor(new TransformSelectedExecutor(toolLinked));
         Internals.ChangeController.TryStartExecutor(new TransformSelectedExecutor(toolLinked));
     }
     }
 
 
@@ -524,6 +584,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
+        
         Internals.ActionAccumulator.AddActions(
         Internals.ActionAccumulator.AddActions(
             new PasteImage_Action(image, corners, memberGuid, ignoreClipSymmetriesEtc, drawOnMask, atFrame, default),
             new PasteImage_Action(image, corners, memberGuid, ignoreClipSymmetriesEtc, drawOnMask, atFrame, default),
             new EndPasteImage_Action());
             new EndPasteImage_Action());
@@ -538,6 +601,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
+        
         Internals.ActionAccumulator.AddFinishedActions(
         Internals.ActionAccumulator.AddFinishedActions(
             new ClipCanvas_Action(Document.AnimationHandler.ActiveFrameBindable));
             new ClipCanvas_Action(Document.AnimationHandler.ActiveFrameBindable));
     }
     }
@@ -554,6 +620,8 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
 
 
         Internals.ActionAccumulator.AddFinishedActions(new FlipImage_Action(flipType, frame, membersToFlip));
         Internals.ActionAccumulator.AddFinishedActions(new FlipImage_Action(flipType, frame, membersToFlip));
     }
     }
@@ -573,6 +641,8 @@ internal class DocumentOperationsModule : IDocumentOperations
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+        
         Internals.ActionAccumulator.AddFinishedActions(new RotateImage_Action(rotation, membersToRotate, frame));
         Internals.ActionAccumulator.AddFinishedActions(new RotateImage_Action(rotation, membersToRotate, frame));
     }
     }
 
 
@@ -583,6 +653,8 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
 
 
         Internals.ActionAccumulator.AddFinishedActions(new CenterContent_Action(structureMembers.ToList(), frame));
         Internals.ActionAccumulator.AddFinishedActions(new CenterContent_Action(structureMembers.ToList(), frame));
     }
     }
@@ -595,6 +667,8 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
 
 
         RectD referenceImageRect =
         RectD referenceImageRect =
             new RectD(VecD.Zero, Document.SizeBindable).AspectFit(new RectD(VecD.Zero, imageSize));
             new RectD(VecD.Zero, Document.SizeBindable).AspectFit(new RectD(VecD.Zero, imageSize));
@@ -611,6 +685,8 @@ internal class DocumentOperationsModule : IDocumentOperations
         if (Internals.ChangeController.IsBlockingChangeActive || Document.ReferenceLayerHandler.ReferenceBitmap is null)
         if (Internals.ChangeController.IsBlockingChangeActive || Document.ReferenceLayerHandler.ReferenceBitmap is null)
             return;
             return;
 
 
+        Internals.ChangeController.TryStopActiveExecutor();
+
         Internals.ActionAccumulator.AddFinishedActions(new DeleteReferenceLayer_Action());
         Internals.ActionAccumulator.AddFinishedActions(new DeleteReferenceLayer_Action());
     }
     }
 
 
@@ -621,6 +697,9 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Document.ReferenceLayerHandler.ReferenceBitmap is null || Internals.ChangeController.IsBlockingChangeActive)
         if (Document.ReferenceLayerHandler.ReferenceBitmap is null || Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
+        
         Internals.ChangeController.TryStartExecutor(new TransformReferenceLayerExecutor());
         Internals.ChangeController.TryStartExecutor(new TransformReferenceLayerExecutor());
     }
     }
 
 
@@ -631,7 +710,8 @@ internal class DocumentOperationsModule : IDocumentOperations
     {
     {
         if (Document.ReferenceLayerHandler.ReferenceBitmap is null || Internals.ChangeController.IsBlockingChangeActive)
         if (Document.ReferenceLayerHandler.ReferenceBitmap is null || Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
-
+        
+        Internals.ChangeController.TryStopActiveExecutor();
 
 
         VecD size = new(Document.ReferenceLayerHandler.ReferenceBitmap.Size.X,
         VecD size = new(Document.ReferenceLayerHandler.ReferenceBitmap.Size.X,
             Document.ReferenceLayerHandler.ReferenceBitmap.Size.Y);
             Document.ReferenceLayerHandler.ReferenceBitmap.Size.Y);
@@ -645,6 +725,11 @@ internal class DocumentOperationsModule : IDocumentOperations
 
 
     public void SelectionToMask(SelectionMode mode, int frame)
     public void SelectionToMask(SelectionMode mode, int frame)
     {
     {
+        if (Internals.ChangeController.IsBlockingChangeActive)
+            return;
+
+        Internals.ChangeController.TryStopActiveExecutor();
+        
         if (Document.SelectedStructureMember is not { } member || Document.SelectionPathBindable.IsEmpty)
         if (Document.SelectedStructureMember is not { } member || Document.SelectionPathBindable.IsEmpty)
             return;
             return;
 
 
@@ -658,6 +743,11 @@ internal class DocumentOperationsModule : IDocumentOperations
 
 
     public void CropToSelection(int frame, bool clearSelection = true)
     public void CropToSelection(int frame, bool clearSelection = true)
     {
     {
+        if (Internals.ChangeController.IsBlockingChangeActive)
+            return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
+        
         var bounds = Document.SelectionPathBindable.TightBounds;
         var bounds = Document.SelectionPathBindable.TightBounds;
         if (Document.SelectionPathBindable.IsEmpty || bounds.Width <= 0 || bounds.Height <= 0)
         if (Document.SelectionPathBindable.IsEmpty || bounds.Width <= 0 || bounds.Height <= 0)
             return;
             return;
@@ -676,6 +766,11 @@ internal class DocumentOperationsModule : IDocumentOperations
 
 
     public void InvertSelection()
     public void InvertSelection()
     {
     {
+        if (Internals.ChangeController.IsBlockingChangeActive)
+            return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
+        
         var selection = Document.SelectionPathBindable;
         var selection = Document.SelectionPathBindable;
         var inverse = new VectorPath();
         var inverse = new VectorPath();
         inverse.AddRect(new RectI(new(0, 0), Document.SizeBindable));
         inverse.AddRect(new RectI(new(0, 0), Document.SizeBindable));
@@ -683,22 +778,26 @@ internal class DocumentOperationsModule : IDocumentOperations
         Internals.ActionAccumulator.AddFinishedActions(
         Internals.ActionAccumulator.AddFinishedActions(
             new SetSelection_Action(inverse.Op(selection, VectorPathOp.Difference)));
             new SetSelection_Action(inverse.Op(selection, VectorPathOp.Difference)));
     }
     }
-    
+
     public void Rasterize(Guid memberId)
     public void Rasterize(Guid memberId)
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
 
 
-        Internals.ActionAccumulator.AddFinishedActions(new RasterizeMember_Action(memberId));    
+        Internals.ActionAccumulator.AddFinishedActions(new RasterizeMember_Action(memberId));
     }
     }
 
 
     public void InvokeCustomAction(Action action)
     public void InvokeCustomAction(Action action)
     {
     {
         if (Internals.ChangeController.IsBlockingChangeActive)
         if (Internals.ChangeController.IsBlockingChangeActive)
             return;
             return;
+        
+        Internals.ChangeController.TryStopActiveExecutor();
 
 
         IAction targetAction = new InvokeAction_PassthroughAction(action);
         IAction targetAction = new InvokeAction_PassthroughAction(action);
-        
+
         Internals.ActionAccumulator.AddActions(targetAction);
         Internals.ActionAccumulator.AddActions(targetAction);
     }
     }
 }
 }

+ 2 - 0
src/PixiEditor/Models/DocumentModels/UpdateableChangeExecutors/TransformSelectedExecutor.cs

@@ -21,6 +21,8 @@ internal class TransformSelectedExecutor : UpdateableChangeExecutor, ITransforma
     private bool isInProgress;
     private bool isInProgress;
     public override ExecutorType Type { get; }
     public override ExecutorType Type { get; }
 
 
+    public override bool BlocksOtherActions => false; 
+
     public TransformSelectedExecutor(bool toolLinked)
     public TransformSelectedExecutor(bool toolLinked)
     {
     {
         Type = toolLinked ? ExecutorType.ToolLinked : ExecutorType.Regular;
         Type = toolLinked ? ExecutorType.ToolLinked : ExecutorType.Regular;

+ 0 - 1
src/PixiEditor/Models/DocumentModels/UpdateableChangeExecutors/UpdateableChangeExecutor.cs

@@ -23,7 +23,6 @@ internal abstract class UpdateableChangeExecutor
     public virtual ExecutorType Type => ExecutorType.Regular;
     public virtual ExecutorType Type => ExecutorType.Regular;
     public virtual ExecutorStartMode StartMode => ExecutorStartMode.RightAway;
     public virtual ExecutorStartMode StartMode => ExecutorStartMode.RightAway;
     public virtual bool BlocksOtherActions => true;
     public virtual bool BlocksOtherActions => true;
-    public virtual bool IsUndoable => true;
 
 
     public void Initialize(IDocument document, DocumentInternalParts internals, IServiceProvider services,
     public void Initialize(IDocument document, DocumentInternalParts internals, IServiceProvider services,
         ChangeExecutionController controller, Action<UpdateableChangeExecutor> onEnded)
         ChangeExecutionController controller, Action<UpdateableChangeExecutor> onEnded)

+ 1 - 1
src/PixiEditor/ViewModels/Tools/ToolViewModel.cs

@@ -101,7 +101,7 @@ internal abstract class ToolViewModel : ObservableObject, IToolHandler
     {
     {
         if (layers.Length is > 1 or 0)
         if (layers.Length is > 1 or 0)
         {
         {
-            CanBeUsedOnActiveLayer = false;
+            CanBeUsedOnActiveLayer = SupportedLayerTypes == null;
             return;
             return;
         }
         }
         
         

+ 6 - 0
src/PixiEditor/ViewModels/Tools/Tools/MoveToolViewModel.cs

@@ -87,6 +87,12 @@ internal class MoveToolViewModel : ToolViewModel, IMoveToolHandler
         ViewModelMain.Current.DocumentManagerSubViewModel.ActiveDocument?.Operations.TryStopToolLinkedExecutor();
         ViewModelMain.Current.DocumentManagerSubViewModel.ActiveDocument?.Operations.TryStopToolLinkedExecutor();
     }
     }
 
 
+    protected override void OnSelectedLayersChanged(IStructureMemberHandler[] layers)
+    {
+        OnDeselecting();
+        OnSelected();
+    }
+
     private static RectI? GetSelectedLayersBounds()
     private static RectI? GetSelectedLayersBounds()
     {
     {
         var layers = ViewModelMain.Current.DocumentManagerSubViewModel.ActiveDocument?.ExtractSelectedLayers();
         var layers = ViewModelMain.Current.DocumentManagerSubViewModel.ActiveDocument?.ExtractSelectedLayers();