Skip to content

Commit

Permalink
fix: Fix possible collection changed while enumerating in EVP propaga…
Browse files Browse the repository at this point in the history
…tion
  • Loading branch information
dr1rrb committed Feb 17, 2023
1 parent 8b8abd4 commit 5bc413a
Show file tree
Hide file tree
Showing 2 changed files with 98 additions and 4 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -1056,6 +1056,80 @@ await RetryAssert(() =>
}
#endif

[TestMethod]
[RunsOnUIThread]
public async Task EVP_When_RemoveHandlerWhileRaisingEvent()
{
const bool canHorizontallyScroll = true, canVerticallyScroll = true;
Border sut;
ScrollViewer sv;
var root = new Border
{
HorizontalAlignment = HorizontalAlignment.Left,
VerticalAlignment = VerticalAlignment.Top,
Child = sv = new ScrollViewer
{
Width = 512,
Height = 512,
HorizontalAlignment = HorizontalAlignment.Left,
VerticalAlignment = VerticalAlignment.Top,
HorizontalScrollBarVisibility = canHorizontallyScroll ? ScrollBarVisibility.Auto : ScrollBarVisibility.Disabled,
VerticalScrollBarVisibility = canVerticallyScroll ? ScrollBarVisibility.Auto : ScrollBarVisibility.Disabled,
HorizontalScrollMode = canHorizontallyScroll ? ScrollMode.Enabled : ScrollMode.Disabled,
VerticalScrollMode = canVerticallyScroll ? ScrollMode.Enabled : ScrollMode.Disabled,
Content = new Grid
{
Height = 1024,
Children =
{
(sut = new Border
{
HorizontalAlignment = HorizontalAlignment.Left,
VerticalAlignment = VerticalAlignment.Top,
Background = new SolidColorBrush(Color.FromArgb(0xFF, 0x00, 0x00, 0xF9)),
Width = 100,
Height = 100,
RenderTransform = new CompositeTransform { TranslateX = 50, TranslateY = 25, ScaleY = 2 }
})
}
}
}
};

var result = new TaskCompletionSource<object?>();
var allowDetach = false;
sut.EffectiveViewportChanged += OnSutEVPChanged;

WindowContent = root;

// First make sure to be loaded (and actually ignore them)
await WaitForIdle();
allowDetach = true;

// Then cause a layout update
sv.ChangeView(null, 512, null, disableAnimation: true);

await result.Task;

void OnSutEVPChanged(FrameworkElement sender, EffectiveViewportChangedEventArgs args)
{
try
{
if (!allowDetach)
{
return;
}

sut.EffectiveViewportChanged -= OnSutEVPChanged;
result.TrySetResult(default);
}
catch (Exception e)
{
result.TrySetException(e);
}
}
}

private async Task RetryAssert(Action assertion)
{
var attempt = 0;
Expand Down
28 changes: 24 additions & 4 deletions src/Uno.UI/UI/Xaml/FrameworkElement.EffectiveViewport.cs
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ partial class FrameworkElement : IFrameworkElement_EffectiveViewport
private event TypedEventHandler<_This, EffectiveViewportChangedEventArgs>? _effectiveViewportChanged;
private bool _hasNewHandler;
private List<IFrameworkElement_EffectiveViewport>? _childrenInterestedInViewportUpdates;
private bool _isEnumeratingChildrenInterestedInViewportUpdates;
private IDisposable? _parentViewportUpdatesSubscription;
private ViewportInfo _parentViewport = ViewportInfo.Empty; // WARNING: Stored in parent's coordinates space, use GetParentViewport()
private ViewportInfo _lastEffectiveViewport;
Expand Down Expand Up @@ -153,12 +154,21 @@ IDisposable IFrameworkElement_EffectiveViewport.RequestViewportUpdates(bool isIn
Uno.UI.Extensions.DependencyObjectExtensions.GetChildren(this).OfType<IFrameworkElement_EffectiveViewport>().Contains(child)
|| (child as _View)?.FindFirstAncestor<IFrameworkElement_EffectiveViewport>() == this);

(_childrenInterestedInViewportUpdates ??= new()).Add(child);
var childrenInterestedInViewportUpdates = _childrenInterestedInViewportUpdates switch
{
null => (_childrenInterestedInViewportUpdates = new()),
_ when _isEnumeratingChildrenInterestedInViewportUpdates => (_childrenInterestedInViewportUpdates = new(_childrenInterestedInViewportUpdates)),
_ => _childrenInterestedInViewportUpdates,
};
childrenInterestedInViewportUpdates.Add(child);
ReconfigureViewportPropagation(isInternalUpdate, child);

return Disposable.Create(() =>
{
_childrenInterestedInViewportUpdates!.Remove(child);
var childrenInterestedInViewportUpdates = _isEnumeratingChildrenInterestedInViewportUpdates
? (_childrenInterestedInViewportUpdates = new(_childrenInterestedInViewportUpdates))
: _childrenInterestedInViewportUpdates!;
childrenInterestedInViewportUpdates.Remove(child);
ReconfigureViewportPropagation();
});
}
Expand Down Expand Up @@ -351,9 +361,19 @@ private void PropagateEffectiveViewportChange(

if (_childrenInterestedInViewportUpdates is { Count: > 0 } && (isInitial || viewportUpdated))
{
foreach (var child in _childrenInterestedInViewportUpdates)
_isEnumeratingChildrenInterestedInViewportUpdates = true;
var enumerator = _childrenInterestedInViewportUpdates.GetEnumerator();
try
{
while (enumerator.MoveNext())
{
enumerator.Current!.OnParentViewportChanged(isInitial, isInternal, this, viewport);
}
}
finally
{
child.OnParentViewportChanged(isInitial, isInternal, this, viewport);
_isEnumeratingChildrenInterestedInViewportUpdates = false;
enumerator.Dispose();
}
}
}
Expand Down

0 comments on commit 5bc413a

Please sign in to comment.