[iOS, macOS] Shell: Fix RTL flow direction for flyout, menu cells, tab bar, and Locked flyout position#32701
[iOS, macOS] Shell: Fix RTL flow direction for flyout, menu cells, tab bar, and Locked flyout position#32701NanthiniMahalingam wants to merge 10 commits intodotnet:mainfrom
Conversation
|
/rebase |
a55bb9a to
2e1f117
Compare
🤖 AI Summary📊 Expand Full Review🔍 Pre-Flight — Context & Validation📝 Review Session — Added the output snapshots ·
|
| # | Source | Approach | Test Result | Files Changed | Notes |
|---|---|---|---|---|---|
| PR | PR #32701 | Propagate Shell FlowDirection to header/footer/tabbar/cells via UpdateFlowDirection; fix Locked flyout X PENDING (Gate) | 5 files (+53 lines) | Original PR, DRAFT status | pos |
🚦 Gate — Test Verification
📝 Review Session — Added the output snapshots · 8612998
Result PASSED:
Platform: ios
Mode: Full Verification
- Tests FAIL without fix
- Tests PASS with fix
Test Filter: Issue32419
Tests: VerifyShellFlyoutContentAlignedInRTL, VerifyShellMenuItemsAlignedInRTL
Blockers: None
🔧 Fix — Analysis & Comparison
📝 Review Session — Added the output snapshots · 8612998
Fix Candidates
| # | Source | Approach | Test Result | Files Changed | Notes |
|---|---|---|---|---|---|
| 1 | try-fix (claude-sonnet-4.5) | Set UISemanticContentAttribute directly on native UIViews (headers, footers, cells, FAIL | 4 files | Why: Bypassing MAUI FlowDirection system loses layout invalidation and recursive propagation | tabbar) |
| 2 | try-fix (claude-opus-4.6) | Container-level RTL propagation: SemanticContentAttribute on container views + FlowDirection on tracked pages/cell views + Locked flyout fix PASS | 5 files | Works but mixes MAUI and native approaches | |
| 3 | try-fix (gpt-5.2-codex) | Apply UpdateFlowDirection to root platform views + reload table without mutating FlowDirection FAIL | 3 files | Why: Missing key insight - MatchParent resolution fails because Shell items' MAUI parent chain doesn't connect to Shell's native view | values |
| PR | PR #32701 | Propagate Shell FlowDirection to header/footer/tabbar/cells via UpdateFlowDirection calls; fix Locked flyout X pos by positionX=shellWidth PASS (Gate) | 5 files | Original PR - idiomatic MAUI approach |
Skipped models (rate-limited): gpt-5.2, gemini-3-pro-preview, gpt-5.1-codex (429 Too Many Requests after retry)
Cross-Pollination Results
| Model | Round 2 Response |
|---|---|
| claude-sonnet-4.5 | NEW IDEA: Coordinate mirroring - intercept Frame/Bounds in renderers to mirror X coords (complex/risky, not pursued) |
| claude-opus-4.6 | NO NEW IDEAS - correctly identifies all approaches exhausted |
| gpt-5.2 | SKIPPED (rate-limited) |
| gpt-5.2-codex | SKIPPED (rate-limited) |
| gemini-3-pro-preview | SKIPPED (rate-limited) |
Exhausted: Yes (available models confirmed exhaustion; rate limits prevented others)
Selected Fix: PR's fix - It is simpler, cleaner, and uses the established UpdateFlowDirection pattern that is consistent with the existing codebase. Attempt #2 also passes but mixes MAUI's FlowDirection system with direct UISemanticContentAttribute manipulation, which is less maintainable. The PR's fix uses the proper MAUI abstraction.
Root Cause: Shell sub-elements (header, footer, cells, tracked pages) have a disconnected MAUI visual tree that prevents MatchParent FlowDirection resolution from propagating the Shell's RTL setting. Additionally, the SlideFlyoutTransition incorrectly calculated the flyout X position in Locked mode for RTL.
📋 Report — Final Recommendation
📝 Review Session — Added the output snapshots · 8612998
Final Recommendation: APPROVE (with comments)
Summary
PR #32701 fixes Shell RTL (Right-to-Left) layout propagation for iOS/macOS. When FlowDirection="RightToLeft" is set on a Shell, page content, flyout header/footer, tab bar items, and menu cells now correctly mirror to RTL. Additionally, the locked flyout X position calculation is fixed.
The PR's fix was validated by Gate (tests FAIL without fix, PASS with fix on iOS). An independent alternative (Attempt #2) also passed, but the PR's approach is more idiomatic to the MAUI codebase.
Root Cause
Shell sub-elements (header, footer, cells, tracked pages, tab bar) have a disconnected MAUI visual tree. When these elements have FlowDirection=MatchParent, the MatchParent resolution cannot traverse up to the Shell's flow direction because their parent chain is broken in the compatibility renderer layer. As a result, UpdateFlowDirection calls on these elements would not propagate the Shell's RTL setting.
Additionally, SlideFlyoutTransition.LayoutViews calculated positionX = shellWidth - openPixels for RTL, which is correct for the Flyout behavior but incorrect for Locked behavior where openPixels = flyoutWidth gives positionX = shellWidth - flyoutWidth (off by one "flyout width" unit).
Fix Quality Assessment
The PR's approach is clean and consistent with the existing UpdateFlowDirection pattern in the codebase:
- Extends existing
UpdateFlowDirection()calls to cover flyout header/footer, tab bar, and cells - Resolves the
MatchParentdisconnection by explicitly settingFlowDirectionbefore callingUpdateFlowDirection - Fixes the Locked flyout position with a minimal targeted check
Title/Description Review
Current Title: [iOS, macOS] Fixed Shell menu and flyout items do not update correctly in RTL mode
Recommended Title: [iOS, macOS] Shell: Fix RTL layout propagation for flyout, tab bar, and menu cells
Reasons:
- Use present-tense verb form ("Fix" not "Fixed")
- More descriptive of what changed
Description Needs improvementsAssessment:
- Missing NOTE block at (actually has it) top
- Good description of changes
- Missing root cause explanation
- PR is marked as DRAFT
Code Review Findings
1. FlowDirection mutation on MAUI elements (Medium)
In ShellFlyoutContentRenderer.cs, ShellSectionRootRenderer.cs, and ShellTableViewSource.cs, the code permanently mutates the element's FlowDirection from MatchParent to the Shell's direction:
if (header.FlowDirection == FlowDirection.MatchParent)
{
header.FlowDirection = _shellContext.Shell.FlowDirection;
}This is a one-way change. If Shell's FlowDirection later changes (e.g., dynamically switched from RTL to LTR), these elements won't revert to MatchParent and won't pick up the new direction. Consider whether this is intentional given the Shell architecture, or whether a non-destructive approach (e.g., storing original value) is needed.
2. Missing test to open flyout and verify menu items (Medium)
VerifyShellFlyoutContentAlignedInRTL takes a screenshot of the home page without opening the flyout. This verifies the navigation bar alignment but NOT the flyout menu item alignment. The issue specifically reports flyout item alignment problems. Consider adding a test that:
- Opens the flyout (taps the hamburger/flyout button)
- Verifies the flyout items are correctly right-aligned
3. SlideFlyoutTransition locked position comment missing (Low)
if (behavior == FlyoutBehavior.Locked)
{
positionX = shellWidth;
}The value shellWidth is not immediately obvious as the correct value for RTL locked mode. A comment explaining why this is correct (e.g., "In RTL with ForceRightToLeft, iOS mirrors coordinates so the flyout appears at the left edge") would help maintainers understand this.
4. Missing newline at end of file (Low)
TestCases.HostApp/Issues/Issue32419.cs is missing a newline at the end of file (the diff shows \ No newline at end of file).
Looks Good
- Implementation follows the established
UpdateFlowDirectionpattern consistently ShellSectionRenderer.cschange correctly guards withTabBarItem is not null && TabBarController?.TabBar is not nullShellTableViewSource.cscorrectly handles both the MAUI FlowDirection and the native platform view- Test structure uses
TestShell,[Category(UITestCategories.Shell)], andVerifyScreenshot()correctly - Snapshot baselines provided for iOS, Android, and Windows
Fix Candidates Summary
| # | Source | Approach | Test Result |
|---|---|---|---|
| 1 | try-fix | UISemanticContentAttribute directly on native FAIL | views |
| 2 | try-fix | Container-level SemanticContentAttribute + FlowDirection on pages/cells PASS | |
| 3 | try-fix | UpdateFlowDirection on root views without FlowDirection FAIL | mutation |
| PR | PR #32701 | UpdateFlowDirection extensions with MatchParent resolution PASS (Gate) |
Selected Fix: PR's fix - more idiomatic to MAUI codebase, uses established UpdateFlowDirection pattern
What NOT to Do (for future agents)
Don't set UISemanticContentAttribute only - Without MAUI's FlowDirection system, child views won't get proper layout invalidation-
Don't call UpdateFlowDirection on root views without resolving MatchParent - Shell sub-elements have disconnected visual trees; MatchParent resolution fails without explicit propagation-
Don't use bounds.Width in RTL flyout positioning - It gives the same result as shellWidth; the fix correctly uses shellWidth directly-
Draft Status
This PR is currently marked as DRAFT. Before merging, the author should address:
- The FlowDirection mutation concern ([Draft] Readme WIP #1 above)
- The missing flyout-open test (Update README.md #2 above)
- The trailing newline in test file (Aloha System.Maui! #4 above)
- Mark PR as ready for review
📋 Expand PR Finalization Review
Title: ⚠️ Needs Update
Current: [iOS, macOS] Fixed Shell menu and flyout items do not update correctly in RTL mode
Issues:
- Uses "Fixed" verb prefix — titles should describe the fix, not the bug
- Phrase "do not update correctly" describes the broken behavior rather than the solution
- Verbose; can be made more scannable
Recommended: [iOS, macOS] Shell: Fix RTL flow direction for flyout, menu cells, tab bar, and Locked flyout position
Description: ✅ Good
- Uses "Fixed" verb prefix — titles should describe the fix, not the bug
- Phrase "do not update correctly" describes the broken behavior rather than the solution
- Verbose; can be made more scannable
✨ Suggested PR Description
[!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 from this PR and let us know in a comment if this change resolves your issue. Thank you!
Root Cause
- Shell flyout headers, footers, and menu item cells (
UITableViewCell) were not propagating the Shell'sFlowDirectionto their platform views. WhenFlowDirectionwas set toMatchParent, the underlying UIKit views never received the resolved (Right-to-Left) direction. - The
TabBarwithin aShellSectionRendererwas not updated whenUpdateFlowDirectionwas called, so tab bar items remained in the default LTR layout. - In
ShellSectionRootRenderer, the tracked page'sFlowDirectionwas not resolved when set toMatchParent, leaving page content in LTR layout. SlideFlyoutTransition.LayoutViewscomputed the flyout's X position usingshellWidth - openPixels, which is correct for the sliding flyout case, but incorrect whenFlyoutBehavior.Lockedis used in RTL mode, resulting in a misaligned flyout.
Description of Change
ShellFlyoutContentRenderer.cs
- In
UpdateFlyoutHeader: when the header view'sFlowDirectionisMatchParent, it is now resolved to the Shell'sFlowDirectionbeforeUpdateFlowDirectionis called on the header's platform view. - In
UpdateFlyoutFooter: same fix applied to the footer view.
ShellSectionRenderer.cs
- In
UpdateFlowDirection: added propagation toTabBarController.TabBarwhen aTabBarItemis present, ensuring the tab bar is mirrored in RTL.
ShellSectionRootRenderer.cs
- In
UpdateFlowDirection: when the current section is active, if the tracked page'sFlowDirectionisMatchParent, it is resolved to the Shell's concrete flow direction before the view update, ensuring page content renders in the correct direction.
ShellTableViewSource.cs
- In
GetCell: when the cell'sView.FlowDirectionisMatchParent, it is resolved to the Shell'sFlowDirection. The resolved direction is then applied to the cell's platformUIViewviaUpdateFlowDirection.
SlideFlyoutTransition.cs
- In
LayoutViews: for RTL (ForceRightToLeft), whenFlyoutBehavior.Lockedis active, the flyout X position is set toshellWidthso the flyout is correctly anchored at the trailing edge in RTL layout.
Issues Fixed
Fixes #32419
Platforms Tested
- Android
- Windows
- iOS
- macOS (Mac Catalyst)
Output
⚠️ Before/After screenshots were not included in the original PR. Please add them to the Output section for reviewer context.
iOS
| Before | After |
|---|---|
| Add screenshot | Add screenshot |
macOS
| Before | After |
|---|---|
| Add screenshot | Add screenshot |
Code Review: ⚠️ Issues Found
Code Review — PR #32701
PR: #32701 - [iOS, macOS] Shell RTL Flow Direction Fix
🔴 Critical Issues
1. SlideFlyoutTransition.cs: Locked RTL flyout X position may place flyout off-screen
File: src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/SlideFlyoutTransition.cs
Problem:
// RTL branch
var positionX = shellWidth - openPixels;
if (behavior == FlyoutBehavior.Locked)
{
positionX = shellWidth; // ← places flyout at x = shellWidth (right edge / off-screen)
}
flyout.Frame = new CGRect(positionX, 0, flyoutWidth, flyoutHeight);Setting positionX = shellWidth positions the flyout's leading edge at shellWidth — entirely off-screen to the right in iOS screen coordinates (origin is top-left). The fix resolves the previously reported misalignment, and this is the author's stated intent, but the chosen value is surprising.
Expected behavior for RTL Locked:
In RTL mode, the flyout should appear anchored to the right side of the screen, visible to the user. The correct value depends on whether openPixels equals flyoutWidth in the Locked state. If openPercent is always 1.0 for Locked, then positionX = shellWidth - openPixels = shellWidth - flyoutWidth, which would correctly anchor the flyout at the right. Setting it to shellWidth (without subtracting flyoutWidth) would put it one flyout-width off-screen.
Recommendation:
Verify with a device/simulator that this value is correct (the author's before/after screenshots would confirm). If openPercent is something other than 1.0 for Locked, document why shellWidth is the right value. Consider adding a comment explaining the geometry.
🟡 Suggestions
1. Mutating FlowDirection from MatchParent to a concrete value is permanent and won't re-resolve on Shell FlowDirection change
Files:
ShellFlyoutContentRenderer.cs(header and footer)ShellSectionRootRenderer.csShellTableViewSource.cs
Pattern:
if (view.FlowDirection == FlowDirection.MatchParent)
{
view.FlowDirection = _shellContext.Shell.FlowDirection; // ← permanent mutation
}This overwrites the element's FlowDirection from MatchParent to a concrete value (e.g., RightToLeft). If the Shell's FlowDirection changes dynamically at runtime, these views won't re-resolve — they'll be stuck with the first resolved value. MatchParent is a sentinel/lazy value and should ideally be preserved so subsequent UpdateFlowDirection calls can re-resolve it.
Note: This pattern does appear in other places in the iOS Shell renderer codebase, so this PR is consistent with existing patterns. However, the underlying fragility should be acknowledged.
Recommendation: Consider whether this is acceptable given current usage patterns, or whether a non-mutating resolution (reading Shell's direction without writing it back to the element) would be safer.
2. ShellTableViewSource.GetCell: Flow direction applied on every GetCell call (including reused cells)
File: src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellTableViewSource.cs
Problem:
public override UITableViewCell GetCell(UITableView tableView, NSIndexPath indexPath)
{
// ...
if (cell.View.FlowDirection == FlowDirection.MatchParent)
{
cell.View.FlowDirection = _context.Shell.FlowDirection;
}
if (cell.View.Handler?.PlatformView is UIView platformView)
{
platformView.UpdateFlowDirection(cell.View);
}
// ...
}GetCell is called every time a cell is requested (including dequeued/reused cells). Since FlowDirection has already been overwritten from MatchParent to a concrete value on first use, the MatchParent guard won't fire on subsequent reuse — meaning UpdateFlowDirection will be called on every GetCell. This is not harmful (idempotent) but is a minor inefficiency.
Recommendation: No action required if behavior is correct. Low priority.
3. Issue32419.cs UI Test — Missing wait after tapping Locked button before screenshot
File: src/Controls/tests/TestCases.Shared.Tests/Tests/Issues/Issue32419.cs
Problem:
[Test, Order(2)]
[Category(UITestCategories.Shell)]
public void VerifyShellMenuItemsAlignedInRTL()
{
App.WaitForElement("homePageLabel");
App.Tap("FlyoutLockedButton");
VerifyScreenshot(); // ← no wait after layout change
}After tapping FlyoutLockedButton, Shell.SetFlyoutBehavior(this, FlyoutBehavior.Locked) triggers a layout change. If the screenshot is taken before the layout settles, this test may be flaky.
Recommendation: Use VerifyScreenshot(retryTimeout: TimeSpan.FromSeconds(2)) to allow the layout to settle:
App.Tap("FlyoutLockedButton");
VerifyScreenshot(retryTimeout: TimeSpan.FromSeconds(2));✅ Looks Good
ShellSectionRenderer.cs— TabBar flow direction propagation is guarded with bothTabBarItem is not nullandTabBarController?.TabBar is not nullnull checks. Clean and safe.ShellFlyoutContentRenderer.cs— The refactoring to add braces around the single-lineiffor theheader is not nullcase is a good readability improvement that avoids future single-lineifbugs.- Test coverage — New
Issue32419.csHostApp page and test class are well-structured, useTestShellbase class correctly, and snapshot files are provided for iOS, Android, and Windows platforms.
8612998 to
7acab73
Compare
I’ve updated PR title and addressed the valid suggestions by adding comments in the relevant places. There are no major concerns with the remaining comments. |
🚦 Gate - Test Before and After Fix📊 Expand Full Gate —
|
| Test | Without Fix (expect FAIL) | With Fix (expect PASS) |
|---|---|---|
🖥️ Issue32419 Issue32419 |
✅ FAIL — 281s | ✅ PASS — 93s |
🔴 Without fix — 🖥️ Issue32419: FAIL ✅ · 281s
Determining projects to restore...
Restored /Users/cloudtest/vss/_work/1/s/src/Graphics/src/Graphics/Graphics.csproj (in 655 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/BindingSourceGen/Controls.BindingSourceGen.csproj (in 655 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Essentials/src/Essentials.csproj (in 5.3 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Core/src/Core.csproj (in 5.78 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/Core/Controls.Core.csproj (in 6.49 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/Xaml/Controls.Xaml.csproj (in 6.49 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/tests/TestCases.HostApp/Controls.TestCases.HostApp.csproj (in 6.5 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/BlazorWebView/src/Maui/Microsoft.AspNetCore.Components.WebView.Maui.csproj (in 6.5 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/Foldable/src/Controls.Foldable.csproj (in 6.52 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/Maps/src/Controls.Maps.csproj (in 6.54 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Core/maps/src/Maps.csproj (in 6.55 sec).
/Users/cloudtest/vss/_work/1/s/.dotnet/packs/Microsoft.iOS.Sdk.net10.0_26.0/26.0.11017/targets/Xamarin.Shared.Sdk.targets(309,3): warning : RuntimeIdentifier was set on the command line, and will override the value for RuntimeIdentifiers set in the project file. [/Users/cloudtest/vss/_work/1/s/src/Controls/tests/TestCases.HostApp/Controls.TestCases.HostApp.csproj::TargetFramework=net10.0-ios]
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Graphics -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Graphics/Debug/net10.0-ios26.0/Microsoft.Maui.Graphics.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Essentials -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Essentials/Debug/net10.0-ios26.0/Microsoft.Maui.Essentials.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Core/Debug/net10.0-ios26.0/Microsoft.Maui.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Maps/Debug/net10.0-ios26.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.60-ci+azdo.13712679
Controls.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core/Debug/net10.0-ios26.0/Microsoft.Maui.Controls.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Microsoft.AspNetCore.Components.WebView.Maui -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Microsoft.AspNetCore.Components.WebView.Maui/Debug/net10.0-ios26.0/Microsoft.AspNetCore.Components.WebView.Maui.dll
Controls.Xaml -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Xaml/Debug/net10.0-ios26.0/Microsoft.Maui.Controls.Xaml.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Controls.Foldable -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Foldable/Debug/net10.0-ios26.0/Microsoft.Maui.Controls.Foldable.dll
Controls.Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Maps/Debug/net10.0-ios26.0/Microsoft.Maui.Controls.Maps.dll
Detected signing identity:
Code Signing Key: "" (-)
Provisioning Profile: "" () - no entitlements
Bundle Id: com.microsoft.maui.uitests
App Id: com.microsoft.maui.uitests
Controls.TestCases.HostApp -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.HostApp/Debug/net10.0-ios/iossimulator-arm64/Controls.TestCases.HostApp.dll
Optimizing assemblies for size may change the behavior of the app. Be sure to test after publishing. See: https://aka.ms/dotnet-illink
Optimizing assemblies for size. This process might take a while.
Build succeeded.
/Users/cloudtest/vss/_work/1/s/.dotnet/packs/Microsoft.iOS.Sdk.net10.0_26.0/26.0.11017/targets/Xamarin.Shared.Sdk.targets(309,3): warning : RuntimeIdentifier was set on the command line, and will override the value for RuntimeIdentifiers set in the project file. [/Users/cloudtest/vss/_work/1/s/src/Controls/tests/TestCases.HostApp/Controls.TestCases.HostApp.csproj::TargetFramework=net10.0-ios]
1 Warning(s)
0 Error(s)
Time Elapsed 00:02:32.05
Determining projects to restore...
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/tests/CustomAttributes/Controls.CustomAttributes.csproj (in 1.22 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/BindingSourceGen/Controls.BindingSourceGen.csproj (in 1.22 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/TestUtils/src/VisualTestUtils/VisualTestUtils.csproj (in 2 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/TestUtils/src/UITest.Core/UITest.Core.csproj (in 2 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Graphics/src/Graphics/Graphics.csproj (in 1.59 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Essentials/src/Essentials.csproj (in 358 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/Core/Controls.Core.csproj (in 1.77 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/TestUtils/src/UITest.NUnit/UITest.NUnit.csproj (in 1.79 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Core/src/Core.csproj (in 498 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/TestUtils/src/UITest.Appium/UITest.Appium.csproj (in 2.56 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/TestUtils/src/VisualTestUtils.MagickNet/VisualTestUtils.MagickNet.csproj (in 4.16 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/tests/TestCases.iOS.Tests/Controls.TestCases.iOS.Tests.csproj (in 4.31 sec).
Restored /Users/cloudtest/vss/_work/1/s/src/TestUtils/src/UITest.Analyzers/UITest.Analyzers.csproj (in 4.48 sec).
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Controls.CustomAttributes -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.CustomAttributes/Debug/net10.0/Controls.CustomAttributes.dll
Graphics -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Graphics/Debug/net10.0/Microsoft.Maui.Graphics.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Essentials -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Essentials/Debug/net10.0/Microsoft.Maui.Essentials.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
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.60-ci+azdo.13712679
Controls.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core/Debug/net10.0/Microsoft.Maui.Controls.dll
UITest.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/UITest.Core/Debug/net10.0/UITest.Core.dll
VisualTestUtils -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/VisualTestUtils/Debug/netstandard2.0/VisualTestUtils.dll
VisualTestUtils.MagickNet -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/VisualTestUtils.MagickNet/Debug/netstandard2.0/VisualTestUtils.MagickNet.dll
UITest.NUnit -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/UITest.NUnit/Debug/net10.0/UITest.NUnit.dll
UITest.Appium -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/UITest.Appium/Debug/net10.0/UITest.Appium.dll
UITest.Analyzers -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/UITest.Analyzers/Debug/netstandard2.0/UITest.Analyzers.dll
Controls.TestCases.iOS.Tests -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.iOS.Tests/Debug/net10.0/Controls.TestCases.iOS.Tests.dll
Test run for /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.iOS.Tests/Debug/net10.0/Controls.TestCases.iOS.Tests.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.
/Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.iOS.Tests/Debug/net10.0/Controls.TestCases.iOS.Tests.dll
[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.05] Discovering: Controls.TestCases.iOS.Tests
[xUnit.net 00:00:00.13] Discovered: Controls.TestCases.iOS.Tests
NUnit Adapter 4.5.0.0: Test execution started
Running selected tests in /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.iOS.Tests/Debug/net10.0/Controls.TestCases.iOS.Tests.dll
NUnit3TestExecutor discovered 2 of 2 NUnit test cases using Current Discovery mode, Non-Explicit run
>>>>> 4/1/2026 7:23:10 AM FixtureSetup for Issue32419(iOS)
>>>>> 4/1/2026 7:23:13 AM VerifyShellFlyoutContentAlignedInRTL Start
>>>>> 4/1/2026 7:23:15 AM VerifyShellFlyoutContentAlignedInRTL Stop
>>>>> 4/1/2026 7:23:15 AM Log types: syslog, crashlog, performance, safariConsole, safariNetwork, server
Failed VerifyShellFlyoutContentAlignedInRTL [1 s]
Error Message:
VisualTestUtils.VisualTestFailedException :
Snapshot different than baseline: VerifyShellFlyoutContentAlignedInRTL.png (3.60% difference)
If the correct baseline has changed (this isn't a a bug), then update the baseline image.
See test attachment or download the build artifacts to get the new snapshot file.
More info: https://aka.ms/visual-test-workflow
Stack Trace:
at VisualTestUtils.VisualRegressionTester.Fail(String message) in /_/src/TestUtils/src/VisualTestUtils/VisualRegressionTester.cs:line 162
at VisualTestUtils.VisualRegressionTester.VerifyMatchesSnapshot(String name, ImageSnapshot actualImage, String environmentName, ITestContext testContext) in /_/src/TestUtils/src/VisualTestUtils/VisualRegressionTester.cs:line 123
at Microsoft.Maui.TestCases.Tests.UITest.<VerifyScreenshot>g__Verify|13_0(String name, <>c__DisplayClass13_0&) in /_/src/Controls/tests/TestCases.Shared.Tests/UITest.cs:line 477
at Microsoft.Maui.TestCases.Tests.UITest.VerifyScreenshot(String name, Nullable`1 retryDelay, Nullable`1 retryTimeout, Int32 cropLeft, Int32 cropRight, Int32 cropTop, Int32 cropBottom, Double tolerance) in /_/src/Controls/tests/TestCases.Shared.Tests/UITest.cs:line 309
at Microsoft.Maui.TestCases.Tests.Issues.Issue32419.VerifyShellFlyoutContentAlignedInRTL() in /_/src/Controls/tests/TestCases.Shared.Tests/Tests/Issues/Issue32419.cs:line 20
at System.Reflection.MethodBaseInvoker.InterpretedInvoke_Method(Object obj, IntPtr* args)
at System.Reflection.MethodBaseInvoker.InvokeWithNoArgs(Object obj, BindingFlags invokeAttr)
>>>>> 4/1/2026 7:23:15 AM VerifyShellMenuItemsAlignedInRTL Start
>>>>> 4/1/2026 7:23:18 AM VerifyShellMenuItemsAlignedInRTL Stop
>>>>> 4/1/2026 7:23:18 AM Log types: syslog, crashlog, performance, safariConsole, safariNetwork, server
Failed VerifyShellMenuItemsAlignedInRTL [3 s]
Error Message:
VisualTestUtils.VisualTestFailedException :
Snapshot different than baseline: VerifyShellMenuItemsAlignedInRTL.png (8.28% difference)
If the correct baseline has changed (this isn't a a bug), then update the baseline image.
See test attachment or download the build artifacts to get the new snapshot file.
More info: https://aka.ms/visual-test-workflow
Stack Trace:
at Microsoft.Maui.TestCases.Tests.UITest.VerifyScreenshot(String name, Nullable`1 retryDelay, Nullable`1 retryTimeout, Int32 cropLeft, Int32 cropRight, Int32 cropTop, Int32 cropBottom, Double tolerance) in /_/src/Controls/tests/TestCases.Shared.Tests/UITest.cs:line 296
at Microsoft.Maui.TestCases.Tests.Issues.Issue32419.VerifyShellMenuItemsAlignedInRTL() in /_/src/Controls/tests/TestCases.Shared.Tests/Tests/Issues/Issue32419.cs:line 30
at System.Reflection.MethodBaseInvoker.InterpretedInvoke_Method(Object obj, IntPtr* args)
at System.Reflection.MethodBaseInvoker.InvokeWithNoArgs(Object obj, BindingFlags invokeAttr)
NUnit Adapter 4.5.0.0: Test execution complete
Total tests: 2
Failed: 2
Test Run Failed.
Total time: 1.2760 Minutes
🟢 With fix — 🖥️ Issue32419: PASS ✅ · 93s
Determining projects to restore...
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/BindingSourceGen/Controls.BindingSourceGen.csproj (in 520 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Graphics/src/Graphics/Graphics.csproj (in 530 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Essentials/src/Essentials.csproj (in 483 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/Core/Controls.Core.csproj (in 529 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Core/src/Core.csproj (in 591 ms).
6 of 11 projects are up-to-date for restore.
/Users/cloudtest/vss/_work/1/s/.dotnet/packs/Microsoft.iOS.Sdk.net10.0_26.0/26.0.11017/targets/Xamarin.Shared.Sdk.targets(309,3): warning : RuntimeIdentifier was set on the command line, and will override the value for RuntimeIdentifiers set in the project file. [/Users/cloudtest/vss/_work/1/s/src/Controls/tests/TestCases.HostApp/Controls.TestCases.HostApp.csproj::TargetFramework=net10.0-ios]
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Graphics -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Graphics/Debug/net10.0-ios26.0/Microsoft.Maui.Graphics.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Essentials -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Essentials/Debug/net10.0-ios26.0/Microsoft.Maui.Essentials.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Core/Debug/net10.0-ios26.0/Microsoft.Maui.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Maps/Debug/net10.0-ios26.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.60-ci+azdo.13712679
Controls.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core/Debug/net10.0-ios26.0/Microsoft.Maui.Controls.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Microsoft.AspNetCore.Components.WebView.Maui -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Microsoft.AspNetCore.Components.WebView.Maui/Debug/net10.0-ios26.0/Microsoft.AspNetCore.Components.WebView.Maui.dll
Controls.Foldable -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Foldable/Debug/net10.0-ios26.0/Microsoft.Maui.Controls.Foldable.dll
Controls.Xaml -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Xaml/Debug/net10.0-ios26.0/Microsoft.Maui.Controls.Xaml.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Controls.Maps -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Maps/Debug/net10.0-ios26.0/Microsoft.Maui.Controls.Maps.dll
Detected signing identity:
Code Signing Key: "" (-)
Provisioning Profile: "" () - no entitlements
Bundle Id: com.microsoft.maui.uitests
App Id: com.microsoft.maui.uitests
Controls.TestCases.HostApp -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.HostApp/Debug/net10.0-ios/iossimulator-arm64/Controls.TestCases.HostApp.dll
Optimizing assemblies for size may change the behavior of the app. Be sure to test after publishing. See: https://aka.ms/dotnet-illink
Optimizing assemblies for size. This process might take a while.
Build succeeded.
/Users/cloudtest/vss/_work/1/s/.dotnet/packs/Microsoft.iOS.Sdk.net10.0_26.0/26.0.11017/targets/Xamarin.Shared.Sdk.targets(309,3): warning : RuntimeIdentifier was set on the command line, and will override the value for RuntimeIdentifiers set in the project file. [/Users/cloudtest/vss/_work/1/s/src/Controls/tests/TestCases.HostApp/Controls.TestCases.HostApp.csproj::TargetFramework=net10.0-ios]
1 Warning(s)
0 Error(s)
Time Elapsed 00:00:48.15
Determining projects to restore...
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/BindingSourceGen/Controls.BindingSourceGen.csproj (in 582 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Essentials/src/Essentials.csproj (in 582 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Graphics/src/Graphics/Graphics.csproj (in 582 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Controls/src/Core/Controls.Core.csproj (in 554 ms).
Restored /Users/cloudtest/vss/_work/1/s/src/Core/src/Core.csproj (in 621 ms).
8 of 13 projects are up-to-date for restore.
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Controls.CustomAttributes -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.CustomAttributes/Debug/net10.0/Controls.CustomAttributes.dll
Graphics -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Graphics/Debug/net10.0/Microsoft.Maui.Graphics.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
Essentials -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Essentials/Debug/net10.0/Microsoft.Maui.Essentials.dll
##vso[build.updatebuildnumber]10.0.60-ci+azdo.13712679
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.60-ci+azdo.13712679
Controls.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.Core/Debug/net10.0/Microsoft.Maui.Controls.dll
VisualTestUtils -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/VisualTestUtils/Debug/netstandard2.0/VisualTestUtils.dll
UITest.Core -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/UITest.Core/Debug/net10.0/UITest.Core.dll
VisualTestUtils.MagickNet -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/VisualTestUtils.MagickNet/Debug/netstandard2.0/VisualTestUtils.MagickNet.dll
UITest.Appium -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/UITest.Appium/Debug/net10.0/UITest.Appium.dll
UITest.NUnit -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/UITest.NUnit/Debug/net10.0/UITest.NUnit.dll
UITest.Analyzers -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/UITest.Analyzers/Debug/netstandard2.0/UITest.Analyzers.dll
Controls.TestCases.iOS.Tests -> /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.iOS.Tests/Debug/net10.0/Controls.TestCases.iOS.Tests.dll
Test run for /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.iOS.Tests/Debug/net10.0/Controls.TestCases.iOS.Tests.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.
/Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.iOS.Tests/Debug/net10.0/Controls.TestCases.iOS.Tests.dll
[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.07] Discovering: Controls.TestCases.iOS.Tests
[xUnit.net 00:00:00.16] Discovered: Controls.TestCases.iOS.Tests
NUnit Adapter 4.5.0.0: Test execution started
Running selected tests in /Users/cloudtest/vss/_work/1/s/artifacts/bin/Controls.TestCases.iOS.Tests/Debug/net10.0/Controls.TestCases.iOS.Tests.dll
NUnit3TestExecutor discovered 2 of 2 NUnit test cases using Current Discovery mode, Non-Explicit run
>>>>> 4/1/2026 7:24:48 AM FixtureSetup for Issue32419(iOS)
>>>>> 4/1/2026 7:24:52 AM VerifyShellFlyoutContentAlignedInRTL Start
>>>>> 4/1/2026 7:24:52 AM VerifyShellFlyoutContentAlignedInRTL Stop
>>>>> 4/1/2026 7:24:52 AM VerifyShellMenuItemsAlignedInRTL Start
>>>>> 4/1/2026 7:24:53 AM VerifyShellMenuItemsAlignedInRTL Stop
Passed VerifyShellFlyoutContentAlignedInRTL [278 ms]
Passed VerifyShellMenuItemsAlignedInRTL [926 ms]
NUnit Adapter 4.5.0.0: Test execution complete
Test Run Successful.
Total tests: 2
Passed: 2
Total time: 17.6116 Seconds
📁 Fix files reverted (6 files)
eng/pipelines/ci-copilot.ymlsrc/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellFlyoutContentRenderer.cssrc/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellSectionRenderer.cssrc/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellSectionRootRenderer.cssrc/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellTableViewSource.cssrc/Controls/src/Core/Compatibility/Handlers/Shell/iOS/SlideFlyoutTransition.cs
New files (not reverted):
github-merge-flow-release-11.jsonc
🤖 AI Summary📊 Expand Full Review —
|
| # | Source | Approach | Test Result | Files Changed | Notes |
|---|---|---|---|---|---|
| PR | PR #32701 | Resolve MatchParent → UpdateFlowDirection on header/footer/cells/tabbar; fix Locked flyout positionX=shellWidth | ✅ PASSED (Gate) | 5 files | Original PR |
🔧 Fix — Analysis & Comparison
Fix Candidates
| # | Source | Approach | Test Result | Files Changed | Notes |
|---|---|---|---|---|---|
| 1 | try-fix (claude-opus-4.6) | Centralized ShellFlowDirectionCoordinator static helper — no FlowDirection mutation, direct SemanticContentAttribute + child propagation |
✅ PASS | 6 files | Avoids MatchParent mutation; needed 2 iterations to add child propagation |
| 2 | try-fix (claude-sonnet-4.6) | UpdateFlowDirection(view, effectiveParent) overload in ViewExtensions — resolves MatchParent via Shell without mutation, minimal infrastructure extension |
✅ PASS | 6 files | Cleanest no-mutation approach; touches core ViewExtensions.cs |
| 3 | try-fix (gpt-5.3-codex) | ShellRenderer-centric propagation through existing renderer hierarchy on FlowDirectionProperty changed | ❌ FAIL | 8 files | Menu-items screenshot 8.45% diff; cells not fully covered by hierarchy traversal |
| 4 | try-fix (gpt-5.4, gemini unavailable) | Pass shell as context to existing UpdateFlowDirection() calls on disconnected surfaces — no mutation, no new types |
❌ FAIL | 5 files | 3.42%/8.45% diffs; UpdateFlowDirection reads element.FlowDirection first — MatchParent+broken chain = LTR fallback |
| PR | PR #32701 | Resolve MatchParent → UpdateFlowDirection on header/footer/cells/tabbar; fix Locked flyout positionX=shellWidth | ✅ PASSED (Gate) | 5 files | Original PR |
Cross-Pollination
| Model | Round | New Ideas? | Details |
|---|---|---|---|
| claude-opus-4.6 | 2 | NO NEW IDEAS | Solution space fully covered |
| claude-sonnet-4.6 | 2 | NO NEW IDEAS | All viable strategies explored |
| gpt-5.3-codex | 2 | NEW IDEA | FlowDirection resolution fallback in GetEffectiveFlowDirection via IMauiContext handler root — variation of already-passing Attempt 2 |
| gpt-5.4 | 2 | NO NEW IDEAS | Fix space sufficiently covered with 3 passing alternatives |
Exhausted: Yes — 3/4 models NO NEW IDEAS; new idea is a variation of passing Attempt 2
Selected Fix: PR's fix — most idiomatic to codebase, minimal (5 files, ~53 lines), well-documented comments, all prior review comments addressed, no new helper classes or core infrastructure changes
📋 Report — Final Recommendation
✅ Final Recommendation: APPROVE
Phase Status
| Phase | Status | Notes |
|---|---|---|
| Pre-Flight | ✅ COMPLETE | Issue #32419, 5 iOS fix files, 2 test files, 9 snapshots |
| Gate | ✅ PASSED | iOS — tests FAIL without fix, PASS with fix |
| Try-Fix | ✅ COMPLETE | 4 attempts (2 ✅, 2 ❌); 3 additional passing alternatives confirmed across 2 review sessions |
| Report | ✅ COMPLETE |
Summary
PR #32701 fixes Shell RTL (Right-to-Left) layout for iOS/macOS. When FlowDirection="RightToLeft" is set on a Shell, page content, flyout header/footer, tab bar items, and menu cells now correctly mirror to RTL. The Locked flyout X position calculation is also fixed for RTL mode. All prior Copilot inline review comments have been fully addressed in commit 584278b.
Root Cause
Shell sub-elements (header, footer, cells, tracked pages, tab bar) in the iOS compatibility renderer layer have a disconnected MAUI visual tree. When these elements have FlowDirection=MatchParent, the MatchParent resolution cannot traverse up to the Shell because the parent chain is broken in the compatibility renderer layer — IVisualTreeElement.GetVisualParent() for these elements yields BaseShellItem, not the Shell itself.
For the flyout position bug: SlideFlyoutTransition.LayoutViews computed positionX = shellWidth - openPixels for RTL, where openPixels = flyoutWidth in Locked mode, giving positionX = shellWidth - flyoutWidth. This is off by one flyout width, placing the flyout incorrectly.
Fix Quality
The PR's fix is clean, minimal, and consistent with the existing UpdateFlowDirection pattern:
- 5 targeted files, ~53 lines added — minimal and focused
- Extends existing
UpdateFlowDirection()call sites to cover previously missing elements: flyout header/footer, tab bar, cells, tracked page - Resolves the
MatchParentdisconnection by explicitly settingFlowDirectionbefore callingUpdateFlowDirection— a one-way mutation fully documented in code comments with architectural rationale - Footer specifically uses Shell as context without mutating
_footer.FlowDirection(cleaner approach, correctly addressed from prior review) TabBarController?.TabBar is not nullguard (withoutTabBarItemcheck) is correct per prior review comment — ensures TabBar is always updated when it exists regardless of asyncTabBarItemstate- Fixes the Locked flyout position with a minimal targeted
if (behavior == FlyoutBehavior.Locked)check settingpositionX = shellWidth - The MatchParent mutation is appropriate: it runs at renderer initialization/property-change time; subsequent
Shell.FlowDirectionchanges continue to update native UIViews correctly via the existingUpdateFlowDirection()wiring
Try-Fix Results Summary
4 independent approaches attempted (2 passing):
- ✅ Attempt 1 (claude-opus-4.6):
ShellFlowDirectionCoordinator— no mutation, direct SemanticContentAttribute + child propagation - ✅ Attempt 2 (claude-sonnet-4.6):
UpdateFlowDirection(view, effectiveParent)overload — no mutation, extends core infrastructure - ❌ Attempt 3 (gpt-5.3-codex): Renderer hierarchy traversal — cells not fully reached, 8.45% screenshot diff
- ❌ Attempt 4 (gpt-5.4): Pass
shellas context directly — UpdateFlowDirection reads element.FlowDirection first; MatchParent+broken chain = LTR fallback
Cross-pollination (Round 2): 3/4 models NO NEW IDEAS; 1 idea is a variation of already-passing Attempt 2. Exhausted.
Selected Fix: PR's fix — most idiomatic to the MAUI codebase, uses existing UpdateFlowDirection pattern, minimal surface area (5 files vs 6), no new helper classes or core infrastructure changes. Passing alternatives (1, 2) either add a new class (Attempt 1) or touch core ViewExtensions.cs (Attempt 2) which has broader impact. The PR's one-way MatchParent mutation is explicitly documented and acceptable for the static RTL setup scenario.
Code Review Findings
1. One-Way FlowDirection Mutation (Low Risk — Documented)
The MatchParent→concrete mutation in ShellFlyoutContentRenderer.cs (header), ShellSectionRootRenderer.cs (tracked page), and ShellTableViewSource.cs (cells) is a one-way change. However:
- Explicitly documented in code comments with full architectural rationale
- Subsequent
Shell.FlowDirectionchanges still update native UIViews correctly viaUpdateFlowDirection()which uses Shell as context - Footer correctly avoids mutation (uses Shell as context directly) — this is the preferred pattern where applicable
- The issue reports a static RTL setup; this trade-off is acceptable
2. VerifyShellFlyoutContentAlignedInRTL Test Design
Test waits for homePageLabel but never explicitly opens the flyout — it captures the Shell content page in RTL. Test name suggests flyout content, but it's actually validating the page content RTL alignment. This was flagged in prior review; the PR author indicated VerifyShellMenuItemsAlignedInRTL handles the flyout. Acceptable, but test name could be more precise (low risk).
3. Snapshot Coverage
PR includes Android and Windows snapshots for the two new tests. No fix code for those platforms — snapshots appear to be visual validation only to confirm the fix is non-regressive on those platforms. Appropriate.
4. DRAFT Status
PR is no longer marked DRAFT (confirmed from current state).
|
/azp run maui-pr-uitests , maui-pr-devicetests |
|
Azure Pipelines successfully started running 2 pipeline(s). |
There was a problem hiding this comment.
Pull request overview
Fixes Shell RTL (Right-to-Left) behavior on iOS/macOS by ensuring flow direction is applied consistently across flyout/header/footer/menu cells/tab bar and correcting the flyout’s locked positioning in RTL, with new UI-test coverage and updated visual snapshots.
Changes:
- Correct RTL flyout positioning when
FlyoutBehavior.Lockedon iOS. - Apply/propagate flow direction updates to Shell flyout header/footer, flyout menu item cells, section root views, and tab bar.
- Add Issue32419 HostApp repro + Appium/NUnit UI tests and update baseline screenshots across platforms.
Reviewed changes
Copilot reviewed 7 out of 15 changed files in this pull request and generated 7 comments.
Show a summary per file
| File | Description |
|---|---|
| src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/SlideFlyoutTransition.cs | Adjusts RTL frame calculation for locked flyout positioning. |
| src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellTableViewSource.cs | Applies flow direction handling to flyout menu item cells. |
| src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellSectionRootRenderer.cs | Updates section root flow direction and tracked-page flow direction handling. |
| src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellSectionRenderer.cs | Ensures tab bar flow direction is updated alongside section/nav bar updates. |
| src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellFlyoutContentRenderer.cs | Ensures flyout header/footer use Shell flow direction and applies it to native views. |
| src/Controls/tests/TestCases.HostApp/Issues/Issue32419.cs | Adds HostApp repro Shell page for RTL flyout/menu alignment + locked behavior. |
| src/Controls/tests/TestCases.Shared.Tests/Tests/Issues/Issue32419.cs | Adds Appium UI tests and screenshot verification for Issue32419. |
| src/Controls/tests/TestCases.iOS.Tests/snapshots/ios/VerifyShellFlyoutContentAlignedInRTL.png | Updates iOS baseline snapshot for RTL alignment. |
| src/Controls/tests/TestCases.iOS.Tests/snapshots/ios/VerifyShellMenuItemsAlignedInRTL.png | Updates iOS baseline snapshot for RTL menu alignment in locked mode. |
| src/Controls/tests/TestCases.iOS.Tests/snapshots/ios/ShellFlowDirectionUpdate.png | Updates iOS baseline snapshot for flow-direction update scenario. |
| src/Controls/tests/TestCases.Android.Tests/snapshots/android/VerifyShellMenuItemsAlignedInRTL.png | Updates Android baseline snapshot for the new test scenario. |
| src/Controls/tests/TestCases.WinUI.Tests/snapshots/windows/VerifyShellFlyoutContentAlignedInRTL.png | Updates WinUI baseline snapshot for RTL alignment. |
| src/Controls/tests/TestCases.WinUI.Tests/snapshots/windows/VerifyShellMenuItemsAlignedInRTL.png | Updates WinUI baseline snapshot for RTL menu alignment. |
| src/Controls/tests/TestCases.WinUI.Tests/snapshots/windows/ShellFlowDirectionUpdate.png | Updates WinUI baseline snapshot for flow-direction update scenario. |
src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellFlyoutContentRenderer.cs
Outdated
Show resolved
Hide resolved
src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellFlyoutContentRenderer.cs
Show resolved
Hide resolved
src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellTableViewSource.cs
Show resolved
Hide resolved
src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellSectionRootRenderer.cs
Show resolved
Hide resolved
src/Controls/src/Core/Compatibility/Handlers/Shell/iOS/ShellSectionRenderer.cs
Outdated
Show resolved
Hide resolved
2d58996 to
aad5c3a
Compare
|
🚀 Dogfood this PR with:
curl -fsSL https://raw.githubusercontent.com/dotnet/maui/main/eng/scripts/get-maui-pr.sh | bash -s -- 32701Or
iex "& { $(irm https://raw.githubusercontent.com/dotnet/maui/main/eng/scripts/get-maui-pr.ps1) } 32701" |
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 from this PR and let us know in a comment if this change resolves your issue. Thank you!
Issues
Root Cause
Description of Changes
Validated the behaviour in the following platforms
Issues Fixed
Fixes #32419
Output
iOS
Issue_32419_iOS_before.mov
Issue_32419_iOS_after.mov
macOS
Issue_32419_mac_before.mov
Issue_32419_mac_after.mov