Skip to content

Commit

Permalink
Editor components now close all active modals on next press (#5838)
Browse files Browse the repository at this point in the history
Made editor components close all active modals on next

or finish press to not leave them open, which can lead to all kinds of issues
  • Loading branch information
hhyyrylainen authored Jan 22, 2025
1 parent dc448a2 commit c23557d
Show file tree
Hide file tree
Showing 2 changed files with 50 additions and 10 deletions.
6 changes: 6 additions & 0 deletions src/general/base_stage/EditorComponentBase.cs
Original file line number Diff line number Diff line change
Expand Up @@ -209,6 +209,12 @@ protected virtual void NextOrFinishClicked()
{
GUICommon.Instance.PlayButtonPressSound();

if (!ModalManager.Instance.TryCancelModals())
{
GD.PrintErr("Cannot close open modals before continuing, not triggering next / finish action");
return;
}

if (OnFinish != null)
{
if (OnFinish!.Invoke(null))
Expand Down
54 changes: 44 additions & 10 deletions src/gui_common/ModalManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -121,7 +121,7 @@ public void MakeModal(TopLevelContainer popup)
}

/// <summary>
/// Closes the top-most modal popup if any.
/// Closes the top-most modal popup, if any.
/// </summary>
[RunOnKeyDown("ui_cancel", Priority = Constants.POPUP_CANCEL_PRIORITY)]
public bool HideTopMostPopup()
Expand All @@ -134,15 +134,7 @@ public bool HideTopMostPopup()
if (popup.Exclusive && !popup.ExclusiveAllowCloseOnEscape)
return false;

// This is emitted before closing to allow window using components to differentiate between "cancel" and
// "any other reason for closing" in case some logic can be simplified by handling just those two situations.
if (popup is CustomWindow dialog)
dialog.EmitSignal(CustomWindow.SignalName.Canceled);

popup.Close();

// TODO: make sure removing this is not a problem (looks like this signal no longer exists at all)
// popup.Notification(Control.NotificationModalClose);
ClosePopupDialog(popup);

return true;
}
Expand All @@ -163,6 +155,35 @@ public bool HideTopMostPopup()
return null;
}

/// <summary>
/// Tries to cancel all open modals (or close if the modals don't support cancellation)
/// </summary>
/// <returns>True if all modals are closed, false if some refused to close</returns>
public bool TryCancelModals()
{
// If no modals, we don't need to do anything
if (modalStack.Count <= 0)
return true;

while (modalStack.Count > 0)
{
var modal = modalStack.First();

// TODO: implement a way for important popups to ignore this close
ClosePopupDialog(modal);

// Fail if modal is still not closed
if (modal.IsVisibleInTree())
{
GD.Print("Modal doesn't want to cancel / close failing close operation of all modals");
return false;
}
}

// All modals are now closed
return true;
}

protected override void Dispose(bool disposing)
{
if (disposing)
Expand All @@ -173,6 +194,19 @@ protected override void Dispose(bool disposing)
base.Dispose(disposing);
}

private void ClosePopupDialog(TopLevelContainer popup)
{
// This is emitted before closing to allow window-using components to differentiate between "cancel" and
// "any other reason for closing" in case some logic can be simplified by handling just those two situations.
if (popup is CustomWindow dialog)
dialog.EmitSignal(CustomWindow.SignalName.Canceled);

popup.Close();

// TODO: make sure removing this is not a problem (looks like this signal no longer exists at all)
// popup.Notification(Control.NotificationModalClose);
}

/// <summary>
/// Parent lost signals are added when a modal is created so that the modal can be automatically deleted.
/// </summary>
Expand Down

0 comments on commit c23557d

Please sign in to comment.