Fundamental issue with many components in multi DPI setups.

Most of the VCL components like the TAdvTool* but also simple components as the
TAdvGlowButton are fundamentally broken with mixed DPI usage.

I'll elaborate the issues exemplarily using a TAdvGlowButton as this is a simpler one to fix,
but it perfectivly shows the root of all the DPI related flaws.

For starter to reproduce the issue, you'll have to have at least two screens (virtual should to too).
The important thing is that both need to be set to a different DPI setting (you can do this by changing screen scale to a different value than 100%).

Now take a VCL TButton and a TAdvGlowButton and place them onto a form.
Starting the application with either screen as primary seems to work, due to "fixes" being made.

We will see something like this:
grafik

Now as we move the form to the other screen it will look like this:

And now moving it back everything breaks down:
grafik

The fundamental flaw is that the components mutate the size values of the Fonts, which is inherently wrong.
You by no means cant ever do this for several reasons. giving few here:

  • It is a user (in the context the programmer) controlled value which only he sets and mutates and not a derived value. Here the relative font size he wishes to have.
  • You will always potentially have data loss! Think (x div 3) * 3 you'll lose the remainder!
  • You can have side effects down the line, think of the Font instance capping its further calculcated value.

If we want derived values, we have to either store and cache them seperatley (TFont can do that already for us btw!) or calculate it on the fly as needed (which you seemed to have tried at some locations).

Now let's dive a little deeper into TAdvGlowButton and fix this one exemplarily.

At around line 4.4k we find:

$IFDEF DELPHI_UNICODE}
{$IFDEF DELPHIXE10_LVL}
procedure TAdvCustomGlowButton.ChangeScale(M, D: Integer; isDpiChange: Boolean);
{$ELSE}
procedure TAdvCustomGlowButton.ChangeScale(M, D: Integer);
{$ENDIF}
begin
  inherited;
  {$IFDEF DELPHIXE15_LVL}
  if not (csDesigning in ComponentState) and not (csLoading in ComponentState) {and not ParentFont} then
  begin
    Font.Size := MulDiv(Font.Size, M, D);
  end;
  {$ENDIF}
  NotesFont.Size := MulDiv(NotesFont.Size,M,D);
end;
{$ENDIF}

Which exposes the mentioned data loss (leading to the catastrophic failure when moving forth and back).
Fortunateley, as we work with TFont, we have an easy solution here and ask it to take care of the calculating derived (scaled) values. We simple can do this:

{$IFDEF DELPHI_UNICODE}
{$IFDEF DELPHIXE10_LVL}
procedure TAdvCustomGlowButton.ChangeScale(M, D: Integer; isDpiChange: Boolean);
begin
  inherited;
  {$IFDEF DELPHIXE15_LVL}
  if not (csDesigning in ComponentState) and not (csLoading in ComponentState) {and not ParentFont} then
  begin
    Font.ChangeScale(M, D, isDpiChange);
  end;
  {$ENDIF}
  NotesFont.ChangeScale(M, D, isDpiChange);
end;
{$ENDIF}

(Note for simplicity only the DELPHIXE10_LVL path is given).

Furthermore, when mapping the font to a custom one we need to apply fixes and change:

    fh := -MulDiv(NotesFont.Size, GetDeviceCaps(Canvas.Handle, LOGPIXELSY), 72);
    fh := -MulDiv(AFont.Size, FormPPI, 72);

to

    fh := -MulDiv(NotesFont.Size, NotesFont.PixelsPerInch, 72);
    fh := -MulDiv(AFont.Size, AFont.PixelsPerInch, 72);

at the three locations it is used.

Tats all to fix the button!

Now unfortunateley fixing the TAdvToolBar* is not as straight forward as everything is quite scattered around, there have been attempts to "remedy" the effects of data loss by simply "add one to hopefully make it work again" etc.
Ontop of all comes the whole styling concept.

Theres basically the following ways to deal about this:

  • Use the builtin TFont capabilities. This requires however that no TFont instances are shared between two different screens. This might break the Styler approach?
  • Cache data yourself (store the derived = scaled values)
  • Derive the data right where it is needed ie. do the scaled font size calc right when it is needed.

As it is now it's definitley in an totally unusable state in multi monitor environments with different scales. You'll figure really much bigger issues on the office sample demo depending how bad the different monitor settings mismatch.

This is also the root cause of issues like: Issue with changing screen dpi TADVGlowButton

Kind regards.

What exact Delphi version do you use and is this the latest TMS VCL UI Pack release? As sadly, we have seen different Delphi releases with subtle & slightly different high DPI handling that make it extremely difficult to write component code for different Delphi versions that takes all these subtle differences in account.
We therefore primarily focus on the last versions of Delphi (12.3)

It is the latest Delphi 12.3 with latest patch and latest TMS.

If you had read the elaboration above, you'd have seen this issue is a fundament design flaw and I've also provided a proper fix for TAdvGlowButton.
However I cannot really fix all of the Components.

Before allocating time to go into details, I wanted to simply check missing basic version information.

Hello, any news about this topic? I had the same issue. And some more problems regarding Ribbons and High DPI.

I have created a simple demo with only a few components. (Delphi 12.2; TMS VCL UI Pack v13.5.4.2)

My Setting: Laptop as main monitor with 150% scaling + external display with 100% scaling.
Start the program and move the form from one to the other and vice versa. There are (sometime) different bad behavior.
TMSRibbon_HighDPI.zip (7.8 KB)

See also Screenshots:



Regards

I can't see this font size issue on TAdvGlowButton here with Delphi 12.3 and latest TMS VCL UI Pack.
There are subtle differences between Delphi 12.1, 12.2 and 12.3 in high DPI handling unfortunately. We're investigating how we can best deal with this.

We skipped 12.3. But we will update to Delphi 13 if it is released.
Can You recompile my sample with Delphi 12.3 and latest TMS components, please? Then I can test if it is working in my setting.
Note: The problem is not only related to TAdvGlowButton. The CompactCaption and the tilte of the pages or TAdvPreviewMenu has similar problems. (You can see all this in the example)

Hello,

I installed Delphi 13 and latest VCL UI Pack. But there are still issues.

The problem with the AdvGlowButton from the first post is also not OK in my setting (primary display 250% 3840 x 2400px and 2nd 100% 1920 x 1080 px) If I start the program on the 100% display first, then the Caption of the GlowButton is much too small.

Also, my demo application is not working in all cases. It depends on which screen the form starts first. But moving from one to the other display does not work, too.

Regards

We will reopen this case and will report.

We revisited this and have seen changed behavior in control high DPI handling for Delphi 12 and Delphi 13.
We did adaptions to make this more compatible with changes that were done in Delphi 12 and Delphi 13 and these adaptions will be included in the next release.

1 Like

I've been updating my multi-monitor app to HighDPI and I've come across all these issues as well. I'm using tms.vcl.uipack (13.5.7.1) and Delphi 12.1.

Many thanks @Schroedter_Ralf for your TAdvGlowButton fix, it works great! This issue has been plaguing me for a long time. I also use quite a lot of TAdvPolyList instances containing TImageItem instances and I've noticed that the caption text on these classes also suffers from the same font scaling problems.

You really need a mix of monitors with different resolutions and scaling settings (>100%) in order to reproduce the problems. My app uses 3 monitors and I need everything to scale properly no matter what monitor each form starts on (this is another important variable in reproducing the problems) and also if forms are moved back and forth between any monitors.

Here's one example of the kind of thing I'm experiencing:

I've also discovered that the routine AdvGDIP.CalculateDPIScale() doesn't seem to produce correct DPI scaling values, and so I've had to fix this in some of my derived classes e.g. at 144 DPI (150% scaling) it returns a value of 1.666.

It will be of tremendous help if the issue(s) you encounter could be isolated and a sample source project with clear steps, multimonitor configuration, IDE(s) used, IDE form designer DPI setting, .. provided.