From d17214640548eba26f10ba0d55e70545d58cb1b9 Mon Sep 17 00:00:00 2001 From: Munsterlander Date: Sun, 20 Aug 2023 10:30:32 -0700 Subject: [PATCH] fix: Prevent `onRemove`/`onDetach` being called for initial Gesture Detector addition (#2653) This fixes: https://github.com/flame-engine/flame/issues/2647 Reverts: https://github.com/flame-engine/flame/pull/2602 OBE: https://github.com/flame-engine/flame/pull/2650 To be clear, the memory leak issue is not addressed because it is not a Flame issue. Flame properly calls the `onRemove` event and if it is needed to pass that on to children, users should do as documented: https://docs.flame-engine.org/latest/flame/game.html#lifecycle. So if you are using `GameWidget.controlled`, add a tappable component and then remove the game, the lifecycle looks like: ``` flutter: onLoad flutter: mount flutter: onMount flutter: attach flutter: add has_tappable_components flutter: refreshWidget // Remove Game flutter: detach flutter: removeGameStateListener flutter: onRemove flutter: remove ``` If you do not use `GameWidget.controlled`, add a tappable component and then remove the game, the lifecycle looks like: ``` flutter: onLoad flutter: mount flutter: onMount flutter: attach flutter: add has_tappable_components flutter: refreshWidget // Remove Game flutter: detach flutter: removeGameStateListener flutter: onRemove flutter: remove ``` If you do not use `GameWidget.controlled`, do not add a tappable component and then remove the game, the lifecycle looks like: ``` flutter: onLoadFuture flutter: mount flutter: onMount flutter: attach // Remove Game flutter: detach flutter: removeGameStateListener flutter: onRemove flutter: remove ``` Previously, the lifecycle would look like: ``` flutter: onLoad flutter: mount flutter: onMount flutter: attach flutter: add has_tappable_components flutter: refreshWidget flutter: detach // These unnecessary calls have been eliminated flutter: removeGameStateListener // These unnecessary calls have been eliminated flutter: onRemove // These unnecessary calls have been eliminated flutter: attach // Remove Game flutter: detach flutter: removeGameStateListener flutter: onRemove flutter: remove ``` I have updated the below diagram for those who may need it in the future: ![Flame and Flutter Events Fixed](https://github.com/flame-engine/flame/assets/17042581/1d98bdaf-db13-4e2c-b0a6-74da2ada89f3) --- doc/flame/game.md | 3 ++- packages/flame/lib/src/game/game.dart | 19 ++++++++++++++----- .../lib/src/game/game_widget/game_widget.dart | 1 + .../flame/lib/src/game/overlay_manager.dart | 10 +++++----- 4 files changed, 22 insertions(+), 11 deletions(-) diff --git a/doc/flame/game.md b/doc/flame/game.md index baeab55a090..cdfdafd6b0d 100644 --- a/doc/flame/game.md +++ b/doc/flame/game.md @@ -94,12 +94,13 @@ Once the `GameWidget` is removed from the tree, `onRemove` is called, just like component is removed from the component tree. ```{note} -The `onRemove` can be used to clean up potential memory leaks by doing the following: +The `onRemove` can be used to clean up potential memory leaks such as the following: ``` ```dart @override void onRemove() { + // Optional based on your game needs. removeAll(children); processLifecycleEvents(); Flame.images.clearCache(); diff --git a/packages/flame/lib/src/game/game.dart b/packages/flame/lib/src/game/game.dart index 53f2e2e88bb..a028943b9ce 100644 --- a/packages/flame/lib/src/game/game.dart +++ b/packages/flame/lib/src/game/game.dart @@ -190,7 +190,10 @@ abstract mixin class Game { ); } _gameRenderBox = gameRenderBox; - onAttach(); + if (!_isInternalRefresh) { + onAttach(); + } + _isInternalRefresh = false; } /// Called when the game has been attached. This can be overridden @@ -202,10 +205,10 @@ abstract mixin class Game { /// /// Should not be called manually. void detach() { - onRemove(); + if (!_isInternalRefresh) { + onDetach(); + } _gameRenderBox = null; - - onDetach(); } /// Called when the game is about to be removed from the Flutter widget tree, @@ -358,11 +361,17 @@ abstract mixin class Game { gameStateListeners.remove(callback); } + bool _isInternalRefresh = false; + /// When a Game is attached to a `GameWidget`, this method will force that /// widget to be rebuilt. This can be used when updating any property which is /// implemented within the Flutter tree. + /// + /// When [isInternalRefresh] is passed as false it will trigger the `onAttach` + /// and `onDetach` events; otherwise, those events will not be called. @internal - void refreshWidget() { + void refreshWidget({bool isInternalRefresh = true}) { + _isInternalRefresh = isInternalRefresh; gameStateListeners.forEach((callback) => callback()); } } diff --git a/packages/flame/lib/src/game/game_widget/game_widget.dart b/packages/flame/lib/src/game/game_widget/game_widget.dart index fdf734c3bac..f21dade9a15 100644 --- a/packages/flame/lib/src/game/game_widget/game_widget.dart +++ b/packages/flame/lib/src/game/game_widget/game_widget.dart @@ -259,6 +259,7 @@ class GameWidgetState extends State> { /// `currentGame`'s `onDispose` method will be called; otherwise, it will not. void disposeCurrentGame({bool callGameOnDispose = false}) { currentGame.removeGameStateListener(_onGameStateChange); + currentGame.onRemove(); if (callGameOnDispose) { currentGame.onDispose(); } diff --git a/packages/flame/lib/src/game/overlay_manager.dart b/packages/flame/lib/src/game/overlay_manager.dart index b53fe6c6e6c..e7f723a17b4 100644 --- a/packages/flame/lib/src/game/overlay_manager.dart +++ b/packages/flame/lib/src/game/overlay_manager.dart @@ -25,14 +25,14 @@ class OverlayManager { /// Clears all active overlays. void clear() { _activeOverlays.clear(); - _game.refreshWidget(); + _game.refreshWidget(isInternalRefresh: false); } /// Marks the [overlayName] to be rendered. bool add(String overlayName) { final setChanged = _addImpl(overlayName); if (setChanged) { - _game.refreshWidget(); + _game.refreshWidget(isInternalRefresh: false); } return setChanged; } @@ -42,7 +42,7 @@ class OverlayManager { final initialCount = _activeOverlays.length; overlayNames.forEach(_addImpl); if (initialCount != _activeOverlays.length) { - _game.refreshWidget(); + _game.refreshWidget(isInternalRefresh: false); } } @@ -67,7 +67,7 @@ class OverlayManager { bool remove(String overlayName) { final hasRemoved = _activeOverlays.remove(overlayName); if (hasRemoved) { - _game.refreshWidget(); + _game.refreshWidget(isInternalRefresh: false); } return hasRemoved; } @@ -77,7 +77,7 @@ class OverlayManager { final initialCount = _activeOverlays.length; overlayNames.forEach(_activeOverlays.remove); if (_activeOverlays.length != initialCount) { - _game.refreshWidget(); + _game.refreshWidget(isInternalRefresh: false); } }