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

Problem with WPF's Slider style (and a few thoughts on porting DX11 rendering engine to Core3) #521

Closed
ab4d opened this issue Apr 7, 2019 · 14 comments
Assignees
Labels
Bug Product bug (most likely) tenet-compatibility Incompatibility with previous versions or with WPF for .NET Framework
Milestone

Comments

@ab4d
Copy link

ab4d commented Apr 7, 2019

Issue Title

WPF's Slider control is not correctly rendered - probably because of a problem with its style.

General

If the following xaml is used to define the slider:
<Slider Minimum="0" Maximum="5" Value="4" IsSnapToTickEnabled="True" TickFrequency="1" TickPlacement="BottomRight" Width="100" HorizontalAlignment="Left" VerticalAlignment="Top" />

Then the shown slider is cut on the right side and does not align with the ticks (the same happens when setting TickPlacement to Both):

image

This happens on a screen without any DPI scale (I was not able to try it on high DPI screen). This issue exists from the preview1 version.

Otherwise, I would like to congratulate you on a great job! I have ported my DirectX 11 rendering engine (Ab3d.DXEngine) and WPF 3D toolkit (Ab3d.PowerToys) to .Net Core 3 and they work without any problems from Preview1. Except changing the csproj files there were no changes in the code needed. The code is quite complex and uses SharpDX to call DirectX 11 API and it runs without any problems. I also tested the WPF 3D part and this also runs very well. One of the first things that I did, was to check the performance. I have expected a slight performance improvement, but just recompiling the code for .Net Core 3 brings no performance benefits to my rendering engine. Anyway, in the future, I plan to use new options like Slice and SIMD vectors and this should have more impact.

@karelz karelz transferred this issue from dotnet/core Apr 7, 2019
@karelz
Copy link
Member

karelz commented Apr 7, 2019

Thanks for the feedback, it is really great to hear that your migration to .NET Core was smooth!

Is the bug above .NET Core specific? (i.e. It does not happen on .NET Framework.)

@karelz karelz added the tenet-compatibility Incompatibility with previous versions or with WPF for .NET Framework label Apr 7, 2019
@ab4d
Copy link
Author

ab4d commented Apr 8, 2019

Yes, Slider control is correctly rendered in .Net Framework (from 3.5 to 4.7.2):
image

@karelz
Copy link
Member

karelz commented Apr 8, 2019

Thanks for confirmation @ab4d!

@karelz karelz added the Bug Product bug (most likely) label Apr 8, 2019
@Berrysoft
Copy link
Contributor

Berrysoft commented Apr 8, 2019

It is correctly rendered in .Net Framework 4.8, too.

And it is NOT correctly rendered in .Net Core 3.0 with DPI scale, either.

@KodamaSakuno
Copy link
Contributor

KodamaSakuno commented Apr 16, 2019

Here's the origin of this problem.

double viewportSize = Math.Max(0.0, ViewportSize);
 
// If viewport is NaN, compute thumb's size based on its desired size,
// otherwise compute the thumb base on the viewport and extent properties
if (double.IsNaN(viewportSize))
{
    ComputeSliderLengths(arrangeSize, isVertical, out decreaseButtonLength, out thumbLength, out increaseButtonLength);
}
else
{
    // Don't arrange if there's not enough content or the track is too small
    if (!ComputeScrollBarLengths(arrangeSize, viewportSize, isVertical, out decreaseButtonLength, out thumbLength, out increaseButtonLength))
    {
        return arrangeSize;
    }
}

Source: https://referencesource.microsoft.com/#PresentationFramework/src/Framework/System/Windows/Controls/Primitives/Track.cs,484

The ViewportSize property is always NaN when the thumb is in a Slider. However, the logic of Math.Max has been changed to match MSVC (dotnet/coreclr#20912), so viewportSize is 0 and the program goes to ComputeScrollBarLengths method. As the result, the thumb is arranged with a wrong size.

UPDATE: Replace decompiled code with the one from Reference Source.

@karelz
Copy link
Member

karelz commented Apr 16, 2019

FYI @tannergooding - fallout from some of our alignments in Math area ...

@karelz
Copy link
Member

karelz commented Apr 16, 2019

@KodamaSakuno thanks for root causing it!
Do you know where in the code it lives? Is that open source yet?

@tannergooding
Copy link
Member

However, the logic of Math.Max has been changed to match MSVC (dotnet/coreclr#20912)

Correct. This was updated to be IEEE 754:2008 compliant and to match the behavior defined by the C/C++ language standards in Annex F - IEC 60559 floating-point arithmetic (which all of MSVC, GCC, and Clang currently follow for their implementations).

However, it is worth noting that IEEE 754:2019 (which is currently in draft form) looks to be changing this slightly. Namely, they are removing minNum/maxNum from the list of "required" operations and replacing them with new minimum/maximum and minimumNumber/maximumNumber operations which are recommended by not required and which more clearly specify various behaviors.

  • minimumNumber/maximumNumber are largely compatible with the existing minNum/maxNum functions It clarifies that +0 is greater than -0 for the purpose of this function and clarifies the behavior of signalling NaN
  • minimum/maximum are new and would propagate the NaN as expected here.

Once the IEEE 754 draft is officially published, there will likely be proposals going up to implement the new recommended operations; which would involve exposing the new minimum/maximum functions under System.Math/System.MathF. The names under which they would be exposed likely need more thought.

  • I'll talk to some people today and determine if, for 3.0, we should revert Math.Min/Math.Max to their previous behavior (which propagates NaN) and instead expose Math.MinNum/Math.MaxNum to handle the case where the NaN should not be propagated.

@KodamaSakuno
Copy link
Contributor

KodamaSakuno commented Apr 16, 2019

@karelz I got it with a decompiler.

UPDATE: I replaced it with the one from Reference Source.

@tannergooding
Copy link
Member

I've opened https://github.com/dotnet/corefx/issues/36925 proposing we partially revert the behavior of Math.Min/Math.Max and expose new functions for the non-propagating functionality.

Ultimately, this would mean we keep the change that considers +0 to be gerater than -0 (for the purposes of these functions only); but we continue propagating NaN so that we don't break people dependent on that behavior. We then provide new functions that don't propagate for people who need/prefer that behavior.

@tannergooding
Copy link
Member

This should be unblocked now, once WPF can consume a new coreclr build.

@jongfeel
Copy link

jongfeel commented Apr 25, 2019

Still reproduce at .NET Core SDK 3.0.100-preview4-011223 (Released 2019-04-18)
I hope fix it next .NET Core release.

dotnetcore_wpf_slider_thumb_bug

@karelz
Copy link
Member

karelz commented Apr 25, 2019

@jongfeel if you follow the links, you can see that the fix is in Math in dotnet/coreclr#24039, commit dotnet/coreclr@c5b2e71. It did not get into Preview 4, but it should be part of Preview 5.

@grubioe grubioe added this to the 3.0 milestone May 24, 2019
@miguep
Copy link
Contributor

miguep commented Jun 4, 2019

This issue has been verified fixed in Preview6. Both by the change made in coreclr and by a change in WPF to remove the dependency on Math.Max/Min NaN behavior.

@miguep miguep closed this as completed Jun 4, 2019
@ghost ghost locked as resolved and limited conversation to collaborators Apr 17, 2022
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
Bug Product bug (most likely) tenet-compatibility Incompatibility with previous versions or with WPF for .NET Framework
Projects
None yet
Development

No branches or pull requests

7 participants