Skip to content

Fix SwipeItemView command leak#35510

Merged
kubaflo merged 7 commits into
dotnet:inflight/currentfrom
AdamEssenmacher:issue-35498
May 19, 2026
Merged

Fix SwipeItemView command leak#35510
kubaflo merged 7 commits into
dotnet:inflight/currentfrom
AdamEssenmacher:issue-35498

Conversation

@AdamEssenmacher

Copy link
Copy Markdown

Fixes a memory leak in SwipeItemView.Command where assigning a long-lived ICommand caused each SwipeItemView to be retained through a direct CanExecuteChanged subscription.

The retained graph could keep row content, command parameters, binding contexts, and row view models alive after the containing page was closed.

This changes SwipeItemView to use the existing ICommandElement / WeakCommandSubscription infrastructure used by other command-backed controls. It also moves command enablement into IsEnabledCore, so explicit IsEnabled=false, parent disabled state, and Command.CanExecute are composed consistently.

Fixes #35498

Testing

dependabot Bot and others added 5 commits May 7, 2026 08:48
Updated [Magick.NET-Q8-AnyCPU](https://github.com/dlemstra/Magick.NET)
from 14.10.4 to 14.12.0.

<details>
<summary>Release notes</summary>

_Sourced from [Magick.NET-Q8-AnyCPU's
releases](https://github.com/dlemstra/Magick.NET/releases)._

## 14.12.0

### What's Changed
- Added `FixByteOrder` to the `DcmReadDefines` (#​1976)
- Added `IconWriteDefines`.

### Related changes in ImageMagick since the last release of Magick.NET:
- Correct bug in `Composite` when using `CopyAlpha` (#​1985)
- Fixed incorrect orientation of JPEG compressed TIFF images (#​1991)
- Heap-Buffer-Overflow write of single zero byte when parsing xml
(GHSA-cr67-pvmx-2pp2)
- Stack Overflow in DestroyXMLTree
(GHSA-fwvm-ggf6-2p4x)
- Out-of-Bounds read in sample operation
(GHSA-pcvx-ph33-r5vv)
- Stack Overflow via Recursive FX Expression Parsing
(GHSA-f4qm-vj5j-9xpw)
- Heap Buffer Overflow in ImageMagick MVG decoder
(GHSA-x9h5-r9v2-vcww)
- Heap overflow caused by integer overflow/wraparound in viff encoder on
32-bit builds
(GHSA-v67w-737x-v2c9)
- Stack-buffer-overflow in MNG encoder with oversized pallete
(GHSA-98cp-rj9f-6v5g)
- Integer overflow in despeckle operation causes heap buffer overflow on
32-bit builds
(GHSA-26qp-ffjh-2x4v)
- Off-by-One in MSL decoder could result in crash
(GHSA-5xg3-585r-9jh5)
- Heap buffer overflow when encoding JXL image with a 16-bit float
(GHSA-jvgr-9ph5-m8v4)
- Heap-use-after-free via XMP profile could result in a crash when
printing the values
(GHSA-r83h-crwp-3vm7)
- Heap buffer overflow (WRITE) in the YAML and JSON encoders
(GHSA-5592-p365-24xh)
- Heap out-of-bounds write in JP2 encoder
(GHSA-pwg5-6jfc-crvh)

### Library updates:
- ImageMagick 7.1.2-19 (2026-04-12)
- aom 3.13.3 (2026-04-02)
- openexr 3.4.9 (2026-04-03)
- freetype 2.14.3 (2026-03-22)
- gdk-pixbuf 2.44.6 (2026-03-31)
- harfbuzz 14.0.0 (2026-04-01)
- liblzma 5.8.3 (2026-04-31)
- libpng 1.6.56 (2026-03-25)

**Full Changelog**:
dlemstra/Magick.NET@14.11.1...14.12.0

## 14.11.1

### Related changes in ImageMagick since the last release of Magick.NET:
- Stack-buffer-overflow WRITE in InterpretImageFilename due to overflow
(GHSA-8793-7xv6-82cf)

### Library updates:
- ImageMagick 7.1.2-18 (2026-03-23)
- aom 3.13.2 (2026-03-19)
- openexr 3.4.7 (2026-03-15)
- harfbuzz 13.2.1 (2026-03-19)

**Full Changelog**:
dlemstra/Magick.NET@14.11.0...14.11.1

## 14.11.0

### What's Changed
- Added `DcmReadDefines`.

### Related changes in ImageMagick since the last release of Magick.NET:
- Access mode change for files created from 0666 to 0600
(ImageMagick/ImageMagick#8609)
- Heap-buffer-overflow in NewXMLTree could result in crash
(GHSA-gc62-2v5p-qpmp)

### Library updates:
- ImageMagick 7.1.2-17 (2026-03-16)
- openexr 3.4.6 (2026-03-01)
- freetype 2.14.2 (2026-03-01)
- harfbuzz 13.0.1 (2026-03-07)
- libxml2 2.15.2 (2026-03-03)

**Full Changelog**:
dlemstra/Magick.NET@14.10.4...14.11.0

Commits viewable in [compare
view](dlemstra/Magick.NET@14.10.4...14.12.0).
</details>

[![Dependabot compatibility
score](https://dependabot-badges.githubapp.com/badges/compatibility_score?dependency-name=Magick.NET-Q8-AnyCPU&package-manager=nuget&previous-version=14.10.4&new-version=14.12.0)](https://docs.github.com/en/github/managing-security-vulnerabilities/about-dependabot-security-updates#about-compatibility-scores)

Dependabot will resolve any conflicts with this PR as long as you don't
alter it yourself. You can also trigger a rebase manually by commenting
`@dependabot rebase`.

[//]: # (dependabot-automerge-start)
[//]: # (dependabot-automerge-end)

---

<details>
<summary>Dependabot commands and options</summary>
<br />

You can trigger Dependabot actions by commenting on this PR:
- `@dependabot rebase` will rebase this PR
- `@dependabot recreate` will recreate this PR, overwriting any edits
that have been made to it
- `@dependabot show <dependency name> ignore conditions` will show all
of the ignore conditions of the specified dependency
- `@dependabot ignore this major version` will close this PR and stop
Dependabot creating any more for this major version (unless you reopen
the PR or upgrade to it yourself)
- `@dependabot ignore this minor version` will close this PR and stop
Dependabot creating any more for this minor version (unless you reopen
the PR or upgrade to it yourself)
- `@dependabot ignore this dependency` will close this PR and stop
Dependabot creating any more for this dependency (unless you reopen the
PR or upgrade to it yourself)
You can disable automated security fix PRs for this repo from the
[Security Alerts page](https://github.com/dotnet/maui/network/alerts).

</details>

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
…t#35333)

Bump OpenTelemetry packages to latest stable versions in the
maui-aspire-servicedefaults template:

- OpenTelemetry.Exporter.OpenTelemetryProtocol: 1.9.0 to 1.15.3
- OpenTelemetry.Extensions.Hosting: 1.9.0 to 1.15.3
- OpenTelemetry.Instrumentation.Http: 1.9.0 to 1.15.1
- OpenTelemetry.Instrumentation.Runtime: 1.9.0 to 1.15.1

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
This pull request updates the following dependencies

[marker]: <> (Begin:a71c12d9-5aa4-4b46-e2d6-08da0cf8cd95)
## From https://github.com/dotnet/xharness
- **Subscription**:
[a71c12d9-5aa4-4b46-e2d6-08da0cf8cd95](https://maestro.dot.net/subscriptions?search=a71c12d9-5aa4-4b46-e2d6-08da0cf8cd95)
- **Build**:
[20260430.4](https://dev.azure.com/dnceng/internal/_build/results?buildId=2964906)
([312724](https://maestro.dot.net/channel/2/github:dotnet:xharness/build/312724))
- **Date Produced**: May 1, 2026 7:05:11 AM UTC
- **Commit**:
[92962e5c46ac08a66ded4c5696209cc60f1a232f](dotnet/xharness@92962e5)
- **Branch**: [main](https://github.com/dotnet/xharness/tree/main)

[DependencyUpdate]: <> (Begin)

- **Dependency Updates**:
  - From [11.0.0-prerelease.26229.1 to 11.0.0-prerelease.26230.4][1]
     - Microsoft.DotNet.XHarness.CLI
     - Microsoft.DotNet.XHarness.TestRunners.Common
     - Microsoft.DotNet.XHarness.TestRunners.Xunit

[1]: dotnet/xharness@9d5a7e9...92962e5

[DependencyUpdate]: <> (End)


[marker]: <> (End:a71c12d9-5aa4-4b46-e2d6-08da0cf8cd95)

Co-authored-by: dotnet-maestro[bot] <dotnet-maestro[bot]@users.noreply.github.com>
> [!NOTE]
> Are you waiting for the changes in this PR to be merged?
> It would be very helpful if you could [test the resulting
artifacts](https://github.com/dotnet/maui/wiki/Testing-PR-Builds) from
this PR and let us know in a comment if this change resolves your issue.
Thank you!

## Description

Replaces `review-rules.md` (flat 345-line checklist) with a dimensional
expert review agent. Single source of truth for all review rules,
organized into 30 dimensions for per-dimension sub-agent evaluation.
Adds inline file:line PR comments alongside the existing wall-of-text
summary.

Extracted from 28k review comments across 5 maintainers via
[extraction-pipeline](https://github.com/dotnet/fsharp/blob/main/.github/agents/extraction-pipeline.md).
No functional code changes.

Recreated from dotnet#35062 on a dotnet/maui branch (originally opened from a
fork).

## What changed

**Before:** `review-rules.md` had 345 lines of flat rules. `code-review`
skill loaded them all into one context. Output was a single wall-of-text
PR comment.

**After:** Rules absorbed into `maui-expert-reviewer.md` as 30
dimensions with 200+ CHECK items. Each dimension runs as an independent
sub-agent with focused context. Output is inline file:line PR comments
via `inline-findings.json`.

## CI Flow

```
Review-PR.ps1 prompt:
  1. code-review → maui-expert-reviewer agent → inline-findings.json
  2. pr-review → Pre-Flight → Try-Fix → Report (sees findings, no duplication)

Posting:
  post-inline-review.ps1    → .json → GitHub file:line comments (NEW)
  post-ai-summary-comment.ps1 → {phase}/content.md → wall-of-text (existing)

CI: COMMENTS_VIA_FILE=true → agent writes .json, script posts
Local: agent writes .json, code-review posts directly via gh api
```

## Files

| Action | File | What |
|--------|------|------|
| **Add** | `agents/maui-expert-reviewer.md` | 30 dimensions, 200+
CHECKs, routing table |
| **Add** | `instructions/collectionview-{android,ios,windows}` |
Platform-isolated CV rules |
| **Add** |
`instructions/{handler-patterns,layout-system,performance-hotpaths,public-api,threading-async}`
| Domain-specific ambient guidance |
| **Add** | `scripts/post-inline-review.ps1` | Posts .json as GitHub PR
review |
| **Del** | `skills/code-review/references/review-rules.md` | Absorbed
into agent |
| **Mod** | `skills/code-review/SKILL.md` | Delegates to agent |
| **Mod** | `scripts/Review-PR.ps1` | Prompt + inline posting wiring |
| **Mod** | `eng/pipelines/ci-copilot.yml` | `COMMENTS_VIA_FILE` env var
|

---------

Co-authored-by: kubaflo <kubaflo@users.noreply.github.com>
Co-authored-by: T-Gro <46543583+T-Gro@users.noreply.github.com>
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
Co-authored-by: Tomas Grosup <tomasgrosup@microsoft.com>
@github-actions

Copy link
Copy Markdown
Contributor

🚀 Dogfood this PR with:

⚠️ WARNING: Do not do this without first carefully reviewing the code of this PR to satisfy yourself it is safe.

curl -fsSL https://raw.githubusercontent.com/dotnet/maui/main/eng/scripts/get-maui-pr.sh | bash -s -- 35510

Or

  • Run remotely in PowerShell:
iex "& { $(irm https://raw.githubusercontent.com/dotnet/maui/main/eng/scripts/get-maui-pr.ps1) } 35510"

@dotnet-policy-service dotnet-policy-service Bot added the community ✨ Community Contribution label May 19, 2026
@dotnet-policy-service

Copy link
Copy Markdown
Contributor

Hey there @@AdamEssenmacher! Thank you so much for your PR! Someone from the team will get assigned to your PR shortly and we'll get it reviewed.

@kubaflo

kubaflo commented May 19, 2026

Copy link
Copy Markdown
Contributor

/review -b feature/regression-check

@kubaflo

kubaflo commented May 19, 2026

Copy link
Copy Markdown
Contributor

/review -b feature/regression-check -p android

@MauiBot

MauiBot commented May 19, 2026

Copy link
Copy Markdown
Collaborator

🤖 AI Summary

👋 @AdamEssenmacher — new AI review results are available. Please review the latest session below.

📊 Review Session5ef8007 · Fix SwipeItemView command leak · 2026-05-19 12:06 UTC
🚦 Gate — Test Before & After Fix

Gate Result: ✅ PASSED

Platform: IOS · Base: main · Merge base: eb20ff6e

Test Without Fix (expect FAIL) With Fix (expect PASS)
🧪 CommandTests CommandTests ✅ FAIL — 37s ✅ PASS — 23s
🧪 SwipeViewTests SwipeViewTests ✅ FAIL — 7s ✅ PASS — 7s
🔴 Without fix — 🧪 CommandTests: FAIL ✅ · 37s
  Determining projects to restore...
  Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/BindingSourceGen/Controls.BindingSourceGen.csproj (in 861 ms).
  Restored /Users/cloudtest/vss/_work/1/s/src/Graphics/src/Graphics/Graphics.csproj (in 983 ms).
  Restored /Users/cloudtest/vss/_work/1/s/src/TestUtils/src/TestUtils/TestUtils.csproj (in 4.88 sec).
  Restored /Users/cloudtest/vss/_work/1/s/src/Controls/tests/Core.UnitTests/Controls.Core.UnitTests.csproj (in 5.67 sec).
  Restored /Users/cloudtest/vss/_work/1/s/src/Essentials/src/Essentials.csproj (in 5.69 sec).
  Restored /Users/cloudtest/vss/_work/1/s/src/Controls/Maps/src/Controls.Maps.csproj (in 5.21 sec).
  Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/Xaml/Controls.Xaml.csproj (in 6.13 sec).
  Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/Core/Controls.Core.csproj (in 6.13 sec).
  Restored /Users/cloudtest/vss/_work/1/s/src/Core/src/Core.csproj (in 6.13 sec).
  Restored /Users/cloudtest/vss/_work/1/s/src/Core/maps/src/Maps.csproj (in 6.15 sec).
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Graphics -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Graphics/Debug/net10.0/Microsoft.Maui.Graphics.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Essentials -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Essentials/Debug/net10.0/Microsoft.Maui.Essentials.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Core/Debug/net10.0/Microsoft.Maui.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Maps/Debug/net10.0/Microsoft.Maui.Maps.dll
  Controls.BindingSourceGen -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.BindingSourceGen/Debug/netstandard2.0/Microsoft.Maui.Controls.BindingSourceGen.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Controls.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core/Debug/net10.0/Microsoft.Maui.Controls.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Controls.Xaml -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Xaml/Debug/net10.0/Microsoft.Maui.Controls.Xaml.dll
  Controls.Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Maps/Debug/net10.0/Microsoft.Maui.Controls.Maps.dll
  TestUtils -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/TestUtils/Debug/netstandard2.0/Microsoft.Maui.TestUtils.dll
  Controls.Core.UnitTests -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core.UnitTests/Debug/net10.0/Microsoft.Maui.Controls.Core.UnitTests.dll
Test run for /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core.UnitTests/Debug/net10.0/Microsoft.Maui.Controls.Core.UnitTests.dll (.NETCoreApp,Version=v10.0)
VSTest version 18.0.1 (arm64)

Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
[xUnit.net 00:00:00.00] xUnit.net VSTest Adapter v2.8.2+699d445a1a (64-bit .NET 10.0.0)
[xUnit.net 00:00:00.12]   Discovering: Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:00.82]   Discovered:  Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:00.83]   Starting:    Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:01.50]     CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SwipeItemView), useWeakEventHandler: False) [FAIL]
[xUnit.net 00:00:01.53]       Assert.False() Failure
[xUnit.net 00:00:01.53]       Expected: False
[xUnit.net 00:00:01.53]       Actual:   True
[xUnit.net 00:00:01.53]       Stack Trace:
[xUnit.net 00:00:01.56]         /_/src/Controls/tests/Core.UnitTests/CommandTests.cs(354,0): at Microsoft.Maui.Controls.Core.UnitTests.CommandTests.CommandsSubscribedToCanExecuteCollect(Type controlType, Boolean useWeakEventHandler)
[xUnit.net 00:00:01.57]         --- End of stack trace from previous location ---
  Passed CanExecuteReturnsFalseIfParameterIsWrongValueType [5 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SearchBar), useWeakEventHandler: False) [45 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.MenuItem), useWeakEventHandler: True) [33 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.ImageButton), useWeakEventHandler: False) [35 ms]
  Failed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SwipeItemView), useWeakEventHandler: False) [495 ms]
  Error Message:
   Assert.False() Failure
Expected: False
Actual:   True
  Stack Trace:
     at Microsoft.Maui.Controls.Core.UnitTests.CommandTests.CommandsSubscribedToCanExecuteCollect(Type controlType, Boolean useWeakEventHandler) in /_/src/Controls/tests/Core.UnitTests/CommandTests.cs:line 354
--- End of stack trace from previous location ---
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.Button), useWeakEventHandler: False) [38 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.ImageButton), useWeakEventHandler: True) [35 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.RefreshView), useWeakEventHandler: False) [38 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.TextCell), useWeakEventHandler: False) [57 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SearchHandler), useWeakEventHandler: True) [39 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SwipeItemView), useWeakEventHandler: True) [38 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.MenuItem), useWeakEventHandler: False) [40 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.RefreshView), useWeakEventHandler: True) [37 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.TextCell), useWeakEventHandler: True) [39 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.Button), useWeakEventHandler: True) [41 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SearchBar), useWeakEventHandler: True) [47 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SearchHandler), useWeakEventHandler: False) [35 ms]
  Passed ExecuteDoesNotRunIfParameterIsWrongReferenceType [< 1 ms]
  Passed ExecuteRunsIfReferenceTypeAndSetToNull [< 1 ms]
  Passed CanExecuteIgnoresParameterIfValueTypeAndSetToNull [< 1 ms]
  Passed CanExecute(expected: True) [2 ms]
  Passed CanExecute(expected: False) [< 1 ms]
  Passed GenericExecuteWithCanExecute [< 1 ms]
  Passed GenericThrowsWithNullExecute [< 1 ms]
  Passed Execute [< 1 ms]
  Passed GenericCanExecute(expected: True) [< 1 ms]
  Passed GenericCanExecute(expected: False) [< 1 ms]
  Passed ThrowsWithNullConstructor [< 1 ms]
  Passed ExecuteDoesNotRunIfParameterIsWrongValueType [< 1 ms]
  Passed GenericThrowsWithNullExecuteAndCanExecuteValid [< 1 ms]
  Passed ThrowsWithNullExecuteValidCanExecute [< 1 ms]
  Passed CanExecuteUsesParameterIfReferenceTypeAndSetToNull [< 1 ms]
  Passed ChangeCanExecute [< 1 ms]
  Passed GenericExecute [< 1 ms]
  Passed ThrowsWithNullCanExecute [< 1 ms]
  Passed ThrowsWithNullParameterizedConstructor [< 1 ms]
  Passed CanExecuteUsesParameterIfNullableAndSetToNull [< 1 ms]
  Passed ExecuteDoesNotRunIfValueTypeAndSetToNull [< 1 ms]
  Passed Constructor [< 1 ms]
  Passed ThrowsWithNullParameterizedCanExecute [< 1 ms]
[xUnit.net 00:00:02.00]   Finished:    Microsoft.Maui.Controls.Core.UnitTests
  Passed ExecuteRunsIfNullableAndSetToNull [< 1 ms]
  Passed ExecuteParameterized [1 ms]
  Passed GenericThrowsWithValidExecuteAndCanExecuteNull [< 1 ms]
  Passed ExecuteWithCanExecute [< 1 ms]
  Passed CanExecuteReturnsFalseIfParameterIsWrongReferenceType [< 1 ms]

Total tests: 45
Test Run Failed.
     Passed: 44
     Failed: 1
 Total time: 2.3741 Seconds

🟢 With fix — 🧪 CommandTests: PASS ✅ · 23s
  Determining projects to restore...
  All projects are up-to-date for restore.
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Graphics -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Graphics/Debug/net10.0/Microsoft.Maui.Graphics.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Essentials -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Essentials/Debug/net10.0/Microsoft.Maui.Essentials.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Core/Debug/net10.0/Microsoft.Maui.dll
  Controls.BindingSourceGen -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.BindingSourceGen/Debug/netstandard2.0/Microsoft.Maui.Controls.BindingSourceGen.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Maps/Debug/net10.0/Microsoft.Maui.Maps.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Controls.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core/Debug/net10.0/Microsoft.Maui.Controls.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Controls.Xaml -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Xaml/Debug/net10.0/Microsoft.Maui.Controls.Xaml.dll
  Controls.Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Maps/Debug/net10.0/Microsoft.Maui.Controls.Maps.dll
  TestUtils -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/TestUtils/Debug/netstandard2.0/Microsoft.Maui.TestUtils.dll
  Controls.Core.UnitTests -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core.UnitTests/Debug/net10.0/Microsoft.Maui.Controls.Core.UnitTests.dll
Test run for /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core.UnitTests/Debug/net10.0/Microsoft.Maui.Controls.Core.UnitTests.dll (.NETCoreApp,Version=v10.0)
VSTest version 18.0.1 (arm64)

Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
[xUnit.net 00:00:00.00] xUnit.net VSTest Adapter v2.8.2+699d445a1a (64-bit .NET 10.0.0)
[xUnit.net 00:00:00.09]   Discovering: Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:00.73]   Discovered:  Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:00.74]   Starting:    Microsoft.Maui.Controls.Core.UnitTests
  Passed CanExecuteReturnsFalseIfParameterIsWrongValueType [3 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SearchBar), useWeakEventHandler: False) [44 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.MenuItem), useWeakEventHandler: True) [33 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.ImageButton), useWeakEventHandler: False) [32 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SwipeItemView), useWeakEventHandler: False) [34 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.Button), useWeakEventHandler: False) [32 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.ImageButton), useWeakEventHandler: True) [30 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.RefreshView), useWeakEventHandler: False) [33 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.TextCell), useWeakEventHandler: False) [31 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SearchHandler), useWeakEventHandler: True) [32 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SwipeItemView), useWeakEventHandler: True) [31 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.MenuItem), useWeakEventHandler: False) [32 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.RefreshView), useWeakEventHandler: True) [33 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.TextCell), useWeakEventHandler: True) [30 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.Button), useWeakEventHandler: True) [32 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SearchBar), useWeakEventHandler: True) [31 ms]
  Passed CommandsSubscribedToCanExecuteCollect(controlType: typeof(Microsoft.Maui.Controls.SearchHandler), useWeakEventHandler: False) [32 ms]
  Passed ExecuteDoesNotRunIfParameterIsWrongReferenceType [< 1 ms]
  Passed ExecuteRunsIfReferenceTypeAndSetToNull [< 1 ms]
  Passed CanExecuteIgnoresParameterIfValueTypeAndSetToNull [< 1 ms]
  Passed CanExecute(expected: True) [1 ms]
  Passed CanExecute(expected: False) [< 1 ms]
  Passed GenericExecuteWithCanExecute [< 1 ms]
  Passed GenericThrowsWithNullExecute [< 1 ms]
  Passed Execute [< 1 ms]
  Passed GenericCanExecute(expected: True) [< 1 ms]
  Passed GenericCanExecute(expected: False) [< 1 ms]
  Passed ThrowsWithNullConstructor [< 1 ms]
  Passed ExecuteDoesNotRunIfParameterIsWrongValueType [< 1 ms]
  Passed GenericThrowsWithNullExecuteAndCanExecuteValid [< 1 ms]
  Passed ThrowsWithNullExecuteValidCanExecute [< 1 ms]
  Passed CanExecuteUsesParameterIfReferenceTypeAndSetToNull [< 1 ms]
  Passed ChangeCanExecute [< 1 ms]
  Passed GenericExecute [< 1 ms]
  Passed ThrowsWithNullCanExecute [< 1 ms]
  Passed ThrowsWithNullParameterizedConstructor [< 1 ms]
[xUnit.net 00:00:01.31]   Finished:    Microsoft.Maui.Controls.Core.UnitTests
  Passed CanExecuteUsesParameterIfNullableAndSetToNull [< 1 ms]
  Passed ExecuteDoesNotRunIfValueTypeAndSetToNull [< 1 ms]
  Passed Constructor [< 1 ms]
  Passed ThrowsWithNullParameterizedCanExecute [< 1 ms]
  Passed ExecuteRunsIfNullableAndSetToNull [< 1 ms]
  Passed ExecuteParameterized [1 ms]
  Passed GenericThrowsWithValidExecuteAndCanExecuteNull [< 1 ms]
  Passed ExecuteWithCanExecute [< 1 ms]
  Passed CanExecuteReturnsFalseIfParameterIsWrongReferenceType [< 1 ms]

Test Run Successful.
Total tests: 45
     Passed: 45
 Total time: 1.6163 Seconds

🔴 Without fix — 🧪 SwipeViewTests: FAIL ✅ · 7s
  Determining projects to restore...
  All projects are up-to-date for restore.
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Graphics -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Graphics/Debug/net10.0/Microsoft.Maui.Graphics.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Essentials -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Essentials/Debug/net10.0/Microsoft.Maui.Essentials.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Core/Debug/net10.0/Microsoft.Maui.dll
  Controls.BindingSourceGen -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.BindingSourceGen/Debug/netstandard2.0/Microsoft.Maui.Controls.BindingSourceGen.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Maps/Debug/net10.0/Microsoft.Maui.Maps.dll
  Controls.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core/Debug/net10.0/Microsoft.Maui.Controls.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Controls.Xaml -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Xaml/Debug/net10.0/Microsoft.Maui.Controls.Xaml.dll
  Controls.Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Maps/Debug/net10.0/Microsoft.Maui.Controls.Maps.dll
  TestUtils -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/TestUtils/Debug/netstandard2.0/Microsoft.Maui.TestUtils.dll
  Controls.Core.UnitTests -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core.UnitTests/Debug/net10.0/Microsoft.Maui.Controls.Core.UnitTests.dll
Test run for /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core.UnitTests/Debug/net10.0/Microsoft.Maui.Controls.Core.UnitTests.dll (.NETCoreApp,Version=v10.0)
VSTest version 18.0.1 (arm64)

Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
[xUnit.net 00:00:00.00] xUnit.net VSTest Adapter v2.8.2+699d445a1a (64-bit .NET 10.0.0)
[xUnit.net 00:00:00.09]   Discovering: Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:00.80]   Discovered:  Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:00.81]   Starting:    Microsoft.Maui.Controls.Core.UnitTests
  Passed TestSwipeItemView [21 ms]
  Passed TestContentBindingContextPropagatesToPassedInSwipeItem [5 ms]
  Passed BindingContextTransfersToNewSetOfSwipeItems [2 ms]
  Passed TestRightItems [< 1 ms]
  Passed TestContentBindingContextPropagatesToAddedSwipeItems [< 1 ms]
  Passed SwipeViewFindsScrollParentDirectlyWhenTreeIsConnected [1 ms]
  Passed TestBottomItems [< 1 ms]
  Passed TestDefaultSwipeItems [2 ms]
  Passed SwipeViewResubscribesToScrollParentAfterRemovalAndReaddition [< 1 ms]
[xUnit.net 00:00:00.90]       Assert.False() Failure
[xUnit.net 00:00:00.90]       Expected: False
[xUnit.net 00:00:00.90]       Actual:   True
[xUnit.net 00:00:00.90]       Stack Trace:
[xUnit.net 00:00:00.90]         /_/src/Controls/tests/Core.UnitTests/SwipeViewTests.cs(449,0): at Microsoft.Maui.Controls.Core.UnitTests.SwipeViewTests.SwipeItemViewCommandCanExecuteUpdatesIsEnabled()
[xUnit.net 00:00:00.90]            at System.Reflection.MethodBaseInvoker.InterpretedInvoke_Method(Object obj, IntPtr* args)
[xUnit.net 00:00:00.90]            at System.Reflection.MethodBaseInvoker.InvokeWithNoArgs(Object obj, BindingFlags invokeAttr)
  Passed TestConstructor [< 1 ms]
  Passed TestSwipeViewBindingContextChangedEvent [< 1 ms]
  Passed TestProgrammaticallyClose [< 1 ms]
  Passed TestSwipeItemsSwipeBehaviorOnInvoked [< 1 ms]
  Passed SwipeItemsRemainInLogicalTreeWhenContentIsSet [12 ms]
  Passed TestContentBindingContextChangedEvent [< 1 ms]
  Passed TestTemplatedContentBindingContextChangedEvent [< 1 ms]
  Failed SwipeItemViewCommandCanExecuteUpdatesIsEnabled [2 ms]
  Error Message:
   Assert.False() Failure
Expected: False
Actual:   True
  Stack Trace:
     at Microsoft.Maui.Controls.Core.UnitTests.SwipeViewTests.SwipeItemViewCommandCanExecuteUpdatesIsEnabled() in /_/src/Controls/tests/Core.UnitTests/SwipeViewTests.cs:line 449
   at System.Reflection.MethodBaseInvoker.InterpretedInvoke_Method(Object obj, IntPtr* args)
   at System.Reflection.MethodBaseInvoker.InvokeWithNoArgs(Object obj, BindingFlags invokeAttr)
[xUnit.net 00:00:00.90]     SwipeItemViewCommandCanExecuteUpdatesIsEnabled [FAIL]
  Passed TestProgrammaticallyOpen [< 1 ms]
[xUnit.net 00:00:00.90]   Finished:    Microsoft.Maui.Controls.Core.UnitTests
  Passed ClearRemovesLogicalChildren [< 1 ms]
  Passed TestLeftItems [< 1 ms]
  Passed TestContentBindingContextPropagatesToNewSwipeItems [< 1 ms]
  Passed TestSwipeItemsExecuteMode [< 1 ms]
  Passed TestTopItems [< 1 ms]
  Passed SwipeViewRediscoversScrollParentWhenTemplateRootIsReparented [< 1 ms]
  Passed SwipeViewFindsScrollParentAfterTemplateParentConnected [< 1 ms]

Test Run Failed.
Total tests: 25
     Passed: 24
     Failed: 1
 Total time: 1.2656 Seconds

🟢 With fix — 🧪 SwipeViewTests: PASS ✅ · 7s
  Determining projects to restore...
  All projects are up-to-date for restore.
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Graphics -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Graphics/Debug/net10.0/Microsoft.Maui.Graphics.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Essentials -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Essentials/Debug/net10.0/Microsoft.Maui.Essentials.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Core/Debug/net10.0/Microsoft.Maui.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Maps/Debug/net10.0/Microsoft.Maui.Maps.dll
  Controls.BindingSourceGen -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.BindingSourceGen/Debug/netstandard2.0/Microsoft.Maui.Controls.BindingSourceGen.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Controls.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core/Debug/net10.0/Microsoft.Maui.Controls.dll
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  ##vso[build.updatebuildnumber]10.0.80-ci+azdo.14135210
  Controls.Xaml -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Xaml/Debug/net10.0/Microsoft.Maui.Controls.Xaml.dll
  Controls.Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Maps/Debug/net10.0/Microsoft.Maui.Controls.Maps.dll
  TestUtils -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/TestUtils/Debug/netstandard2.0/Microsoft.Maui.TestUtils.dll
  Controls.Core.UnitTests -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core.UnitTests/Debug/net10.0/Microsoft.Maui.Controls.Core.UnitTests.dll
Test run for /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core.UnitTests/Debug/net10.0/Microsoft.Maui.Controls.Core.UnitTests.dll (.NETCoreApp,Version=v10.0)
VSTest version 18.0.1 (arm64)

Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
[xUnit.net 00:00:00.00] xUnit.net VSTest Adapter v2.8.2+699d445a1a (64-bit .NET 10.0.0)
[xUnit.net 00:00:00.09]   Discovering: Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:00.72]   Discovered:  Microsoft.Maui.Controls.Core.UnitTests
[xUnit.net 00:00:00.73]   Starting:    Microsoft.Maui.Controls.Core.UnitTests
  Passed TestSwipeItemView [17 ms]
  Passed TestContentBindingContextPropagatesToPassedInSwipeItem [5 ms]
  Passed BindingContextTransfersToNewSetOfSwipeItems [2 ms]
  Passed TestRightItems [< 1 ms]
  Passed TestContentBindingContextPropagatesToAddedSwipeItems [< 1 ms]
  Passed SwipeViewFindsScrollParentDirectlyWhenTreeIsConnected [1 ms]
  Passed TestBottomItems [< 1 ms]
  Passed TestDefaultSwipeItems [2 ms]
  Passed SwipeViewResubscribesToScrollParentAfterRemovalAndReaddition [< 1 ms]
  Passed TestConstructor [< 1 ms]
  Passed TestSwipeViewBindingContextChangedEvent [< 1 ms]
  Passed TestProgrammaticallyClose [1 ms]
  Passed TestSwipeItemsSwipeBehaviorOnInvoked [< 1 ms]
  Passed SwipeItemsRemainInLogicalTreeWhenContentIsSet [11 ms]
  Passed TestContentBindingContextChangedEvent [< 1 ms]
  Passed TestTemplatedContentBindingContextChangedEvent [< 1 ms]
  Passed SwipeItemViewCommandCanExecuteUpdatesIsEnabled [2 ms]
  Passed TestProgrammaticallyOpen [< 1 ms]
[xUnit.net 00:00:00.81]   Finished:    Microsoft.Maui.Controls.Core.UnitTests
  Passed ClearRemovesLogicalChildren [< 1 ms]
  Passed TestLeftItems [< 1 ms]
  Passed TestContentBindingContextPropagatesToNewSwipeItems [< 1 ms]
  Passed TestSwipeItemsExecuteMode [< 1 ms]
  Passed TestTopItems [< 1 ms]
  Passed SwipeViewRediscoversScrollParentWhenTemplateRootIsReparented [< 1 ms]
  Passed SwipeViewFindsScrollParentAfterTemplateParentConnected [< 1 ms]

Test Run Successful.
Total tests: 25
     Passed: 25
 Total time: 1.1910 Seconds

📁 Fix files reverted (6 files)
  • src/Controls/src/Core/PublicAPI/net-android/PublicAPI.Unshipped.txt
  • src/Controls/src/Core/PublicAPI/net-ios/PublicAPI.Unshipped.txt
  • src/Controls/src/Core/PublicAPI/net-maccatalyst/PublicAPI.Unshipped.txt
  • src/Controls/src/Core/PublicAPI/net/PublicAPI.Unshipped.txt
  • src/Controls/src/Core/PublicAPI/netstandard/PublicAPI.Unshipped.txt
  • src/Controls/src/Core/SwipeView/SwipeItemView.cs

🧪 UI Tests — SwipeView

Detected UI test categories: SwipeView

Deep UI tests — 69 passed, 0 failed across 1 category on platform-pool agent (replaces in-process counts above).

🧪 UI Test Execution Results (deep, platform pool)

Category Tests Snapshot diffs
controls-SwipeView 69/69 ✓
📎 Download drop-deep-uitests artifact (TRX + snapshot diffs)

🔍 Pre-Flight — Context & Validation

Pre-Flight: PR #35510 — Fix SwipeItemView command leak

Issue #35498 summary

SwipeItemView.Command subscribes directly to ICommand.CanExecuteChanged and unsubscribes only when the Command property changes. When the same long-lived ICommand instance is shared across many rows (e.g. from a page/shell/service VM), the command keeps each SwipeItemView (and its CommandParameter, Content, BindingContext, row VM, and per-row payload) alive forever.

Observed (issue repro):

  • iOS: 1,000 retained rows / ~125 MiB after closing the host page (vs. 0 / 0 MiB for the SwipeItem-based control path).
  • Android: 1,000 / ~125 MiB (vs. 160 / 20 MiB platform baseline).

Repro retains via:

long-lived ICommand
  -> CanExecuteChanged
  -> SwipeItemView.OnCommandCanExecuteChanged   // strong delegate -> instance
  -> SwipeItemView
  -> CommandParameter / Content / BindingContext

PR #35510 fix (canonical approach)

The PR makes SwipeItemView implement ICommandElement and reuses MAUI's existing weak-subscription infrastructure (the same one used by Button, MenuItem, CheckBox, RefreshView, SearchBar, SearchHandler, ImageButton, TextCell):

  1. CommandProperty / CommandParameterProperty wired to CommandElement.OnCommandChanging, OnCommandChanged, OnCommandParameterChanged.
  2. CommandProperty.DependsOn(CommandParameterProperty) so binding-order timing is handled (see CommandParameter TemplateBinding Lost During ControlTemplate Reparenting #31939).
  3. ICommandElement.CleanupTracker = new WeakCommandSubscription(...) — uses a DependentHandle(bindable, handler). The handler is reachable only while the bindable is reachable, so the command no longer keeps the view alive.
  4. IsEnabledCore override composes base.IsEnabledCore && Command.CanExecute(CommandParameter), so an explicit IsEnabled=false (or a disabled parent) is not stomped on the next CanExecuteChanged callback.
  5. ICommandElement.CanExecuteChangedRefreshIsEnabledProperty().

Tests added by the PR (gate)

  • CommandsSubscribedToCanExecuteCollect(SwipeItemView, true|false) (in CommandTests) — proves the command no longer retains the view: forces GC and asserts the WeakReference<SwipeItemView> is collected. This is the leak gate.
  • SwipeItemViewCommandCanExecuteUpdatesIsEnabled (in SwipeViewTests) — proves behavioral semantics:
    • CanExecute=falseIsEnabled=false.
    • CanExecuteChanged toggles IsEnabled.
    • Changing CommandParameter re-evaluates CanExecute.
    • IsEnabled=false overrides a CanExecute=true (sticky on next CanExecuteChanged). This last assertion can only be satisfied by composing through IsEnabledCore — a setter-based "IsEnabled = CanExecute" approach would flip it back to true on the next CanExecuteChanged.

Gate result

✅ PASSED (per separate gate phase): tests fail on base, pass with PR.

Implications for alternative candidates

The PR's approach is the canonical MAUI pattern for this exact problem. Meaningful alternatives must differ either in:

  • A. Subscription mechanism — hand-rolled weak (WeakReference<T> closure, WeakEventManager, etc.) instead of WeakCommandSubscription/DependentHandle. Functionally equivalent at best; more code; same outcome.
  • B. IsEnabled composition — set IsEnabled property directly (original design) instead of IsEnabledCore override. Demonstrably regresses the "explicit IsEnabled=false is sticky" assertion in the new test.
  • C. No subscription at all — lazy CanExecute only at OnInvoked time. Loses reactive IsEnabled updates → regresses the SwipeItemViewCommandCanExecuteUpdatesIsEnabled test.

Any alternative must (1) eliminate the strong reference from command -> view, (2) reactively update IsEnabled on CanExecuteChanged, (3) honor explicit IsEnabled=false. The PR satisfies all three with the least amount of bespoke code.

Base / branch

  • Base: main @ eb20ff6e
  • PR head: 5ef8007b (working branch pr-review-35510 @ 2236d7be — squashed PR diff).

🔧 Fix — Analysis & Comparison

Try-Fix Aggregate — PR #35510

Summary

Two alternative fix candidates were generated and tested against the PR's gate (CommandTests + SwipeViewTests, iOS-targeting unit tests). Both were chosen to explore meaningfully different axes from the PR's approach.

# Approach Tests passed Verdict
1 Hand-rolled WeakReference<SwipeItemView> closure + IsEnabledCore composition (replaces canonical ICommandElement/WeakCommandSubscription) 70/70 ✅ Functionally equivalent to PR; strictly worse (more bespoke code, no DependsOn/GetCanExecute binding-timing protection for #31939, divergent from other MAUI commandable controls)
2 No subscription at all — lazy CanExecute evaluation only on property changes 69/70 ❌ Invalid — regresses SwipeItemViewCommandCanExecuteUpdatesIsEnabled because IsEnabled no longer reactively tracks CanExecuteChanged events between property writes

Design constraints established by gate tests

  1. Weak subscription required. CommandsSubscribedToCanExecuteCollect(SwipeItemView, …) mandates that the command does not retain the view.
  2. Reactive subscription required. SwipeItemViewCommandCanExecuteUpdatesIsEnabled mandates that IsEnabled updates when ICommand.CanExecuteChanged fires (i.e., the lazy/no-subscription approach is ruled out).
  3. IsEnabledCore composition required. The last assertion of SwipeItemViewCommandCanExecuteUpdatesIsEnabled (explicit IsEnabled=false sticky across a subsequent ChangeCanExecute) is impossible to satisfy with the original "IsEnabled = Command.CanExecute(...)" setter pattern; only an IsEnabledCore override composes correctly.

The PR's fix is the only design that satisfies all three constraints and reuses the canonical ICommandElement pattern already used by Button, MenuItem, CheckBox, RefreshView, SearchBar, SearchHandler, ImageButton, and TextCell, and inherits the CommandElement.GetCanExecute binding-timing fix from #31939.

Recommendation

Approve PR #35510 as-is. No demonstrably better alternative exists in the explored design space. The PR is consistent with the established MAUI pattern, minimal in bespoke code, and passes all gate tests.

Details for each candidate are in try-fix-1/content.md and try-fix-2/content.md.

Phase metadata

  • Platform: iOS (unit tests, runs on macOS host, .NET 10)
  • Base: main @ eb20ff6e
  • PR head: 5ef8007b
  • Gate (separately verified): ✅ PASSED
  • Try-fix candidates: 2 (1 functionally-valid-but-worse, 1 invalid). Search stopped — no meaningfully different axes remain.

📋 Report — Final Recommendation

Comparative Report — PR #35510

Candidates compared

Candidate Approach Gate tests Verdict
pr ICommandElement + WeakCommandSubscription + IsEnabledCore (canonical MAUI pattern). ✅ 17/17 pass (per gate phase). Correct fix; 4 PublicAPI.Unshipped.txt files have an accidental BOM.
pr-plus-reviewer pr + strip accidental UTF-8 BOM from 4 PublicAPI.Unshipped.txt files. ✅ 17/17 pass (re-verified on candidate). Strictly better than pr — same correct fix + minor hygiene cleanup.
try-fix-1 Hand-rolled WeakReference<SwipeItemView> closure + IsEnabledCore (replaces canonical ICommandElement). ✅ 70/70 gate pass. Functionally equivalent to PR but ~3× bespoke code, loses DependsOn/GetCanExecute binding-timing protection from #31939, diverges from 8 other commandable controls. Worse than PR.
try-fix-2 No subscription at all — lazy CanExecute on property changes only. ❌ 69/70 — regresses SwipeItemViewCommandCanExecuteUpdatesIsEnabled (line 436: command.ChangeCanExecute() no longer reactively updates IsEnabled). Invalid — behavioral regression.

Ranking (failing-regression candidates ranked below passing ones, as required)

  1. 🥇 pr-plus-reviewer — passes gate, canonical fix, no hygiene issues.
  2. 🥈 pr — passes gate, canonical fix, minor BOM hygiene issue.
  3. 🥉 try-fix-1 — passes gate, but functionally redundant and strictly inferior on consistency/binding-timing/maintainability.
  4. try-fix-2fails gate (regresses behavioral test). Ranked below all passing candidates per scoring rule.

Dimensions

Dimension pr pr-plus-reviewer try-fix-1 try-fix-2
Leak fix (gate test) ✅ (trivially — no subscription)
Reactive CanExecuteChangedIsEnabled
Sticky explicit IsEnabled=false
CommandParameter binding-timing fix (#31939) ✅ (via GetCanExecute)
Consistency with 8 other MAUI commandable controls ❌ (bespoke) ❌ (bespoke)
Lines of bespoke code ~16 ~16 ~45 ~6 (but invalid)
PublicAPI.Unshipped.txt hygiene ❌ (BOM noise) n/a n/a

Winner: pr-plus-reviewer

The PR's substantive code change is already optimal — it is the canonical MAUI commandable-control pattern, reuses shared infrastructure (WeakCommandSubscription/DependentHandle), inherits the #31939 binding-timing fix, and satisfies all behavioral invariants the new tests pin (including the subtle "explicit IsEnabled=false is sticky" assertion that rules out a naive setter-based design). The only actionable reviewer feedback is the accidental UTF-8 BOM addition to 4 of the 5 PublicAPI.Unshipped.txt files. Stripping the BOM is a 40-line, text-file-only delta that re-verifies green against the full gate (17/17). Therefore pr-plus-reviewer is strictly better than pr (same behavior + cleaner artifact) and strictly better than every try-fix candidate explored.

Recommendation for the human reviewer: approve the PR, optionally request the BOM cleanup as a nit (or post it as inline review comments — captured in inline-findings.json).

Phase metadata

  • Platform: iOS (unit tests, .NET 10, macOS host)
  • Base: main @ eb20ff6eca
  • PR head: 5ef8007b
  • Gate (separately verified): ✅ PASSED
  • Candidates evaluated: 4 (pr, pr-plus-reviewer, try-fix-1, try-fix-2)

@MauiBot MauiBot added s/agent-review-incomplete s/agent-reviewed PR was reviewed by AI agent workflow (full 4-phase review) labels May 19, 2026
@kubaflo kubaflo changed the base branch from main to inflight/current May 19, 2026 13:57
@kubaflo

kubaflo commented May 19, 2026

Copy link
Copy Markdown
Contributor

/azp run maui-pr

@azure-pipelines

Copy link
Copy Markdown
Azure Pipelines successfully started running 1 pipeline(s).

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
@kubaflo

kubaflo commented May 19, 2026

Copy link
Copy Markdown
Contributor

/azp run maui-pr

@azure-pipelines

Copy link
Copy Markdown
Azure Pipelines successfully started running 1 pipeline(s).

@kubaflo

kubaflo commented May 19, 2026

Copy link
Copy Markdown
Contributor

/review -b feature/regression-check -p android

@kubaflo kubaflo merged commit 4d59186 into dotnet:inflight/current May 19, 2026
33 of 36 checks passed
@github-actions github-actions Bot added this to the .NET 10.0 SR8 milestone May 19, 2026
@AdamEssenmacher AdamEssenmacher deleted the issue-35498 branch May 19, 2026 21:04
@kubaflo kubaflo added s/agent-approved AI agent recommends approval - PR fix is correct and optimal s/agent-gate-passed AI verified tests catch the bug (fail without fix, pass with fix) and removed s/agent-review-incomplete labels May 20, 2026
PureWeen pushed a commit that referenced this pull request Jun 2, 2026
Fixes a memory leak in `SwipeItemView.Command` where assigning a
long-lived `ICommand` caused each `SwipeItemView` to be retained through
a direct `CanExecuteChanged` subscription.

The retained graph could keep row content, command parameters, binding
contexts, and row view models alive after the containing page was
closed.

This changes `SwipeItemView` to use the existing `ICommandElement` /
`WeakCommandSubscription` infrastructure used by other command-backed
controls. It also moves command enablement into `IsEnabledCore`, so
explicit `IsEnabled=false`, parent disabled state, and
`Command.CanExecute` are composed consistently.

Fixes #35498

## Testing

- `dotnet test
src/Controls/tests/Core.UnitTests/Controls.Core.UnitTests.csproj
--filter
"FullyQualifiedName~CommandTests|FullyQualifiedName~SwipeViewTests"`
- `dotnet build Microsoft.Maui.BuildTasks.slnf`
- Verified the #35498 iOS repro after forced rebuild:
- `SwipeItemViewCommand`: `commandSubscribers=0`, `aliveRows=0`,
`retainedPayloadBytes=0`
- Verified the #35498 Android repro after forced rebuild:
- `SwipeItemViewCommand` no longer retains all 1,000 rows / 125 MiB and
drops to the Android platform baseline

---------

Co-authored-by: Jakub Florkowski <42434498+kubaflo@users.noreply.github.com>
PureWeen pushed a commit that referenced this pull request Jun 11, 2026
Fixes a memory leak in `SwipeItemView.Command` where assigning a
long-lived `ICommand` caused each `SwipeItemView` to be retained through
a direct `CanExecuteChanged` subscription.

The retained graph could keep row content, command parameters, binding
contexts, and row view models alive after the containing page was
closed.

This changes `SwipeItemView` to use the existing `ICommandElement` /
`WeakCommandSubscription` infrastructure used by other command-backed
controls. It also moves command enablement into `IsEnabledCore`, so
explicit `IsEnabled=false`, parent disabled state, and
`Command.CanExecute` are composed consistently.

Fixes #35498

## Testing

- `dotnet test
src/Controls/tests/Core.UnitTests/Controls.Core.UnitTests.csproj
--filter
"FullyQualifiedName~CommandTests|FullyQualifiedName~SwipeViewTests"`
- `dotnet build Microsoft.Maui.BuildTasks.slnf`
- Verified the #35498 iOS repro after forced rebuild:
- `SwipeItemViewCommand`: `commandSubscribers=0`, `aliveRows=0`,
`retainedPayloadBytes=0`
- Verified the #35498 Android repro after forced rebuild:
- `SwipeItemViewCommand` no longer retains all 1,000 rows / 125 MiB and
drops to the Android platform baseline

---------

Co-authored-by: Jakub Florkowski <42434498+kubaflo@users.noreply.github.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

area-controls-swipeview SwipeView community ✨ Community Contribution s/agent-approved AI agent recommends approval - PR fix is correct and optimal s/agent-gate-passed AI verified tests catch the bug (fail without fix, pass with fix) s/agent-reviewed PR was reviewed by AI agent workflow (full 4-phase review)

Projects

None yet

Development

Successfully merging this pull request may close these issues.

SwipeItemView.Command leaks row views and command parameters through CanExecuteChanged

5 participants