Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixed HorizontalStackLayout Crashes Debugger on Negative Spacing #26927

Open
wants to merge 4 commits into
base: main
Choose a base branch
from

Conversation

devanathan-vaithiyanathan
Copy link
Contributor

@devanathan-vaithiyanathan devanathan-vaithiyanathan commented Jan 2, 2025

Issue Details:

When negative spacing value is given for StackLayout, System.ArgumentOutOfRangeException is thrown in WinUI.

The issue occurs when an Image control is placed inside a StackLayout without explicitly setting the HeightRequest and WidthRequest properties.

Root Cause:

The measured value (Width/Height)of the elements like image has calculated as 0 during first measure call in WinUI platform when negative spacing is passed for Layouts. When deducting the negative spacing from measured value(Width / height) in Measure() of LayoutManager.cs it will return negative value in Measure override of MauiPanel.cs. So when converting this negative value to ToPlatform(), System.ArgumentOutOfRangeException is thrown.

Description of Change:

As Measured value of Image is negative during first call from native, the negative value has been constrained and the measured value is set to 0 using Math.Max() and converting 0 to ToPlatform() will not throw exception.
On the subsequent call, the Image is measured correctly, and the negative spacing is appropriately deducted from the measured value. This adjustment ensures that the spacing is applied accurately and aligns with the intended layout behavior.

Issues Fixed

Fixes #19513

Tested the behavior in the following platforms.

  • Android
  • Windows
  • iOS
  • Mac

#Output Screenshot

Before After

@dotnet-policy-service dotnet-policy-service bot added the community ✨ Community Contribution label Jan 2, 2025
Copy link
Contributor

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

@@ -62,7 +63,8 @@ protected override WSize ArrangeOverride(WSize finalSize)
var height = finalSize.Height;

var actual = CrossPlatformArrange(new Rect(0, 0, width, height));
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wonder if CrossPlatformArrange is allowed to return negative widths and heights. Does it make sense in some cases or should it return always strictly >=0 values?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I initially added the Math.Max logic as a safeguard to ensure the layout wouldn't encounter issues due to negative dimensions. However, upon reviewing the source and testing the behavior without this change, I found that CrossPlatformArrange works as expected and does not produce negative widths or heights under normal circumstances.

Since the framework inherently handles this case correctly, clamping the values to zero is unnecessary.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for the reply. I still struggle a bit to understand why the last two lines:

var measure = CrossPlatformMeasure(availableSize.Width, availableSize.Height);

measure.Width = Math.Max(measure.Width, 0);
measure.Height = Math.Max(measure.Height, 0);

are necessary. Why CrossPlatformMeasure can return negative widths and/or heights? I mean you fix it in MauiPanel but is it OK that CrossPlatformMeasure returns negative values at all?

(I'm mostly asking because one can put there an assert or log a warning if it is not desirable. I fail to see why a measurement should be negative but my knowledge of this topic is not great.)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In the WinUI platform, when negative spacing is applied to a layout, the initial measured values (Width and Height) of certain elements, such as an Image, are calculated as 0 during the first measure pass. This occurs because, during the first measure call, the PlatformView is measured as 0 when GetDesiredSize() is invoked.

During the measurement process in the Measure() method of LayoutManager.cs, deducting the negative spacing from the measured values (Width /Height) results in negative dimensions being returned to the Measure override of MauiPanel.cs.
Please find below code section where spacing calculation is included:
For VerticalStack layout: https://github.com/dotnet/maui/blob/main/src/Core/src/Layouts/VerticalStackLayoutManager.cs#L36
HorizontalStackLayout: https://github.com/dotnet/maui/blob/main/src/Core/src/Layouts/HorizontalStackLayoutManager.cs#L35
Attempting to convert these negative dimensions to platform-specific units using ToPlatform() leads to a System.ArgumentOutOfRangeException.

To address this issue, a fix has been implemented where negative values are constrained using Math.Max(), setting the measured value to 0 if it is negative. This ensures that converting a value of 0 to platform-specific units does not throw an exception.
On the subsequent call, the Image is measured correctly, and the negative spacing is appropriately deducted from the measured value. This adjustment ensures that the spacing is applied accurately and aligns with the intended layout behavior.

@karthikraja-arumugam karthikraja-arumugam added the partner/syncfusion Issues / PR's with Syncfusion collaboration label Jan 3, 2025
@devanathan-vaithiyanathan devanathan-vaithiyanathan marked this pull request as ready for review January 3, 2025 14:11
@devanathan-vaithiyanathan devanathan-vaithiyanathan requested a review from a team as a code owner January 3, 2025 14:11
@rmarinho
Copy link
Member

rmarinho commented Jan 7, 2025

/azp run

Copy link

Azure Pipelines successfully started running 3 pipeline(s).

@jfversluis jfversluis added the area-layout StackLayout, GridLayout, ContentView, AbsoluteLayout, FlexLayout, ContentPresenter label Jan 9, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area-layout StackLayout, GridLayout, ContentView, AbsoluteLayout, FlexLayout, ContentPresenter community ✨ Community Contribution partner/syncfusion Issues / PR's with Syncfusion collaboration platform/windows 🪟
Projects
None yet
Development

Successfully merging this pull request may close these issues.

HorizontalStackLayout Crashes Debugger on Negative Spacing
6 participants