Fix SwipeItemView command leak#35510
Conversation
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> [](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>
|
🚀 Dogfood this PR with:
curl -fsSL https://raw.githubusercontent.com/dotnet/maui/main/eng/scripts/get-maui-pr.sh | bash -s -- 35510Or
iex "& { $(irm https://raw.githubusercontent.com/dotnet/maui/main/eng/scripts/get-maui-pr.ps1) } 35510" |
|
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. |
|
/review -b feature/regression-check |
|
/review -b feature/regression-check -p android |
🤖 AI Summary
📊 Review Session —
|
| 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.txtsrc/Controls/src/Core/PublicAPI/net-ios/PublicAPI.Unshipped.txtsrc/Controls/src/Core/PublicAPI/net-maccatalyst/PublicAPI.Unshipped.txtsrc/Controls/src/Core/PublicAPI/net/PublicAPI.Unshipped.txtsrc/Controls/src/Core/PublicAPI/netstandard/PublicAPI.Unshipped.txtsrc/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):
CommandProperty/CommandParameterPropertywired toCommandElement.OnCommandChanging,OnCommandChanged,OnCommandParameterChanged.CommandProperty.DependsOn(CommandParameterProperty)so binding-order timing is handled (see CommandParameter TemplateBinding Lost During ControlTemplate Reparenting #31939).ICommandElement.CleanupTracker = new WeakCommandSubscription(...)— uses aDependentHandle(bindable, handler). The handler is reachable only while the bindable is reachable, so the command no longer keeps the view alive.IsEnabledCoreoverride composesbase.IsEnabledCore && Command.CanExecute(CommandParameter), so an explicitIsEnabled=false(or a disabled parent) is not stomped on the nextCanExecuteChangedcallback.ICommandElement.CanExecuteChanged→RefreshIsEnabledProperty().
Tests added by the PR (gate)
CommandsSubscribedToCanExecuteCollect(SwipeItemView, true|false)(inCommandTests) — proves the command no longer retains the view: forces GC and asserts theWeakReference<SwipeItemView>is collected. This is the leak gate.SwipeItemViewCommandCanExecuteUpdatesIsEnabled(inSwipeViewTests) — proves behavioral semantics:CanExecute=false⇒IsEnabled=false.CanExecuteChangedtogglesIsEnabled.- Changing
CommandParameterre-evaluatesCanExecute. IsEnabled=falseoverrides aCanExecute=true(sticky on nextCanExecuteChanged). This last assertion can only be satisfied by composing throughIsEnabledCore— a setter-based "IsEnabled = CanExecute" approach would flip it back to true on the nextCanExecuteChanged.
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 ofWeakCommandSubscription/DependentHandle. Functionally equivalent at best; more code; same outcome. - B. IsEnabled composition — set
IsEnabledproperty directly (original design) instead ofIsEnabledCoreoverride. Demonstrably regresses the "explicitIsEnabled=falseis sticky" assertion in the new test. - C. No subscription at all — lazy
CanExecuteonly atOnInvokedtime. Loses reactiveIsEnabledupdates → regresses theSwipeItemViewCommandCanExecuteUpdatesIsEnabledtest.
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 branchpr-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
- Weak subscription required.
CommandsSubscribedToCanExecuteCollect(SwipeItemView, …)mandates that the command does not retain the view. - Reactive subscription required.
SwipeItemViewCommandCanExecuteUpdatesIsEnabledmandates thatIsEnabledupdates whenICommand.CanExecuteChangedfires (i.e., the lazy/no-subscription approach is ruled out). IsEnabledCorecomposition required. The last assertion ofSwipeItemViewCommandCanExecuteUpdatesIsEnabled(explicitIsEnabled=falsesticky across a subsequentChangeCanExecute) is impossible to satisfy with the original "IsEnabled = Command.CanExecute(...)" setter pattern; only anIsEnabledCoreoverride 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)
- 🥇
pr-plus-reviewer— passes gate, canonical fix, no hygiene issues. - 🥈
pr— passes gate, canonical fix, minor BOM hygiene issue. - 🥉
try-fix-1— passes gate, but functionally redundant and strictly inferior on consistency/binding-timing/maintainability. - ❌
try-fix-2— fails 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 CanExecuteChanged → IsEnabled |
✅ | ✅ | ✅ | ❌ |
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)
|
/azp run maui-pr |
|
Azure Pipelines successfully started running 1 pipeline(s). |
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
|
/azp run maui-pr |
|
Azure Pipelines successfully started running 1 pipeline(s). |
|
/review -b feature/regression-check -p android |
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>
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>
Fixes a memory leak in
SwipeItemView.Commandwhere assigning a long-livedICommandcaused eachSwipeItemViewto be retained through a directCanExecuteChangedsubscription.The retained graph could keep row content, command parameters, binding contexts, and row view models alive after the containing page was closed.
This changes
SwipeItemViewto use the existingICommandElement/WeakCommandSubscriptioninfrastructure used by other command-backed controls. It also moves command enablement intoIsEnabledCore, so explicitIsEnabled=false, parent disabled state, andCommand.CanExecuteare 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.slnfSwipeItemView.Commandleaks row views and command parameters throughCanExecuteChanged#35498 iOS repro after forced rebuild:SwipeItemViewCommand:commandSubscribers=0,aliveRows=0,retainedPayloadBytes=0SwipeItemView.Commandleaks row views and command parameters throughCanExecuteChanged#35498 Android repro after forced rebuild:SwipeItemViewCommandno longer retains all 1,000 rows / 125 MiB and drops to the Android platform baseline