![]() |
Formulargröße auf HighDPI-Monitor speichern
Mir ist das gerade zum ersten mal aufgefallen.
Verschiebe ich mein Formular auf meinen 150% DPI-Monitor und beende das Programm, wird die Formulargröße viel zu groß gespeichert. Was vorher width 1000 war ist jetzt 1500. Scheint normal zu sein. Aber wenn ich das Programm jetzt erneut starte, ist das Fenster nochmal größer als 1500. Muss man beim Speichern unter HighDPI irgendwas beachten? |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Zitat:
Zitat:
Schwieriger wird es da schon bei der Position. Andererseits sollte man beim Wiederherstellen von Position und Größe immer noch auf die aktuellen Bereiche achten, sonst ist das Form irgendwann mal halb oder ganz verschwunden. |
AW: Formulargröße auf HighDPI-Monitor speichern
Da wird wohl dein Code zum Speichern und wiederherstellen der Fenstergrößen falsch sein.
Ich würde die Speicherung normiert auf 96 dpi durchführen und beim Laden prüfen was du vor dir hast. |
AW: Formulargröße auf HighDPI-Monitor speichern
Wie führe ich eine solche normierte 96 DPI Speicherung aus?
Ich übergebe beim Speichern der Formulargröße Form123 an die entsprechende Prozedur, die anschließend AForm.Width und AForm.Height in eine Datei schreibt. |
AW: Formulargröße auf HighDPI-Monitor speichern
Ich denke übrigens, dass Delphi z.B. aus 125% (120dpi/96dpi) 146% macht... Warum ist das so?
|
AW: Formulargröße auf HighDPI-Monitor speichern
Prüfe auf welchen Monitor Formular liegt (Property Monitor) und welche DPI-Einstellung dieser hat.
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Zitat:
Ich habe schon Dinge versucht wie NeueBreite = Prozentwert aus (AktuelleBreite, (AktuelleDPI minus 96) ) aber das klappt nicht und gibt einen Wert von rund 700 zurück statt 1000. Ganz genau genommen ist mein Formular bei 96 DPI 1050 Pixel breit. Nach dem Rüberziehen und Speichern auf einen 144 DPI (150%) Monitor werden daraus 1575. Das sind 33,4% mehr. Wie ich da wieder auf 1050 komme ist mir ein Geheimnis. |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Und lade sie so ein Windows Skaliert dann automatisch. oder? Wenn du die größe inklusive DPI abspeicherst nun ja dann skaliert Windows immer höher. gruss |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Ich würde die gerne umrechnen zurück auf 96 DPI aber ich weiß leider nicht wie. Versuche ergeben (wie oben geschrieben) sehr komische Werte die nicht korrekt sein können. Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Hallo DieDolly von einer Länge 1050 auf 1575 sind's nicht 33,4% mehr sondern 50%. Genau so ist es bei den DPI Werten: Von 96 auf 144 sind's auch 50% mehr. Wenn du auf System 1 Auflösung DPI1 und auf System 2 DPI2 hast, und ein Formular auf System 1 l1 lang ist, dann hat es auf System 2 eine Länge von l2 := (DPI2/DPI1)*l1. In deinem Fall l2=144/96*1050=1.5*1050=1575. |
AW: Formulargröße auf HighDPI-Monitor speichern
Ich gebe mal ein kleines Beispiel wie ich das in meinem KVPlayer mache (c++) keine Ahnung ob du das dann so auf VCL umsetzen kannst.
Code:
CLIENT_WIDTH und CLIENT_HEIGHT ist die normale größe ohne DPI.
SetRect(&rc, 0, 0, dpi(CLIENT_WIDTH), dpi(CLIENT_HEIGHT));
AdjustWindowRectEx(&rc, dwStyle, FALSE, 0);
Code:
Ich rechne DPI hoch wie folgt.
#define CLIENT_WIDTH 960
#define CLIENT_HEIGHT 658
Code:
Du musst nun lediglich die DPI in umgekehrter Reihenfolge zurück rechnen das wars schon.
long dpi(IN long nPix) {
static float ratio; if (ratio == 0) { HDC hDC = GetDC(0); ratio = (GetDeviceCaps(hDC, LOGPIXELSX) / 96.0f); ReleaseDC(0, hDC); } return (long)(nPix * ratio); } Ich denke damit kommst du zurecht. gruss |
AW: Formulargröße auf HighDPI-Monitor speichern
Und wie rechne ich zurück (auf 96 DPI), dass ich so abspeichern kann, dass Windows nicht bei jedem Start neu hochskaliert? :(
Leider verstehe ich das da oben nicht. Was funktioniert ist, das Formular kurz vorher zurück auf 96 DPI zu skalieren ABER ob das schön ist ... ich bezweifle es
Delphi-Quellcode:
Das ergibt dann zwar 1057 Pixel statt 1050 aber damit kann man leben.
// Routine zum beenden und speichern des Programms ...
ZuSchließendeUndSpeicherndeForm.ScaleBy(96, Aktuelle DPI des Monitors auf dem das Form liegt); |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Code:
so wird die neue gesetzt das ins Gegenteil umlegen und fertig.
(GetDeviceCaps(hDC, LOGPIXELSX) / 96.0f);
Code:
nPix * Ratio
Code:
Oder?
nPix div Ratio
gruss |
AW: Formulargröße auf HighDPI-Monitor speichern
Leider kenne ich mich mit C++ kein bisschen aus. Deswegen sagen mir weder diese Funktionen was noch die Variablen.
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Zitat:
Delphi-Quellcode:
<zu speichernder Wert> := MulDiv(<aktueller Wert>, 96, <aktuelle DPI>);
|
AW: Formulargröße auf HighDPI-Monitor speichern
Hallo Dolly, was an der Formel verstehst du nicht?
Nimm ein System 1 mit DPI1 und ein System 2 mit DPI2. Formularlänge auf System 1 sei l1 und auf System 2 l2. Wenn DPI1 und DPI2 bekannt sind und das Formular mit Länge l1 momentan auf System 1 liegt, dann gilt für die entsprechende Länge l2 auf System 2 (zum Beispiel die Breite oder die Höhe oder die Diagonale ;-)): l2 := DPI2/DPI1*l1 (1) und umgekehrt kannst du natürlich auch nach l1 auflösen: 11 := DPI1/DPI2*l2 (2) In deinem Fall: Dein Formular befindet sich auf System 2 mit DPI2=144 und hat eine Breite von w=1575. Nun willst du die Breite auf System 1 bestimmen. Es gilt Formel 2 von oben: Du berechnest l1 := 96/144*1575 = 1050 Das Formular hat also auf System 1 eine Breite von l1=1050. Was nebenbei und nicht ganz zufällig genau dem Wert entspricht, den du gemessen hast ;-). |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Delphi-Quellcode:
function dpi(nPix: Integer): Integer; (oder(LongInt ))
var ratio: Single; DC: HDC; begin if (ratio = 0) then begin DC := GetDC(0); ratio := (GetDeviceCaps(DC, LOGPIXELSX) / 96.0); ReleaseDC(0, hDC); end; result := nPix * ratio; end;
Delphi-Quellcode:
ratio: Single;
global definieren. Dafür brauche ich nicht mal ein Manifest.. muss halt nur drauf achten das ich jedes Control durch diese Funktion jage. Zitat:
GetDC(0) ist das DC des Desktop. Von der Auflösung des Desktop wird dann über GetDeviceCaps(GetDeviceCaps(DC, LOGPIXELSX)) die DPI zurückgegeben. Diese noch durch 96.0 teilen und fertig. gruss |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Ich hatte es so verstanden, daß es genau darum ging: Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
![]() Zitat:
Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Ich habe mir eure Funktionen jetzt nochmal genau angeguckt, selber nachgerechnet und ja, ich bin dumm wie Stroh. ich bin halt ein mathematisches Anti-Wunderkind im erhöhten Alter :thumb:
Ich versuche hier die ganze Zeit Prozentrechnung und wundere mich, warum ich nicht zurück auf 1050 komme. MultiDiv scheint genau das zu tun, was Michael schrieb. Leider kann man in die Quelltexte von MultiDiv nicht reingucken. Wenn ich das richtig verstehe, ist MultiDiv auch plattformspezifisch. Meine zwei neuen Funktionen, die mir sicher oft noch weiterhelfen werden
Delphi-Quellcode:
***
// Plattformunabhängiges MultiDiv, Uwe Raabe, Michael II@ DelphiPraxis.net
function TMathUtils.MulDiv(nNumber, nNumerator, nDenominator: Integer): Integer; begin Result := Round((nNumber / nNumerator) * nDenominator); end; // Diese Funktion nutze ich aktuell dort, wo ich vorher nur ScaleBy nutzte procedure TFormUtils.ScaleForm(const Form: TForm; const NewDPI, CurrentDPI: Integer); begin if NewDPI <> CurrentDPI then begin SendMessage(Form.Handle, WM_SETREDRAW, WPARAM(False), 0); // LockWindow nutze ich nicht wegen, siehe *** try aForm.ScaleBy(NewDPI, CurrentDPI); finally SendMessage(Form.Handle, WM_SETREDRAW, WPARAM(True), 0); end; end; end; Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Versteh nicht was du da tust aber gut..
Was hat das verhindern von Zeichnungs Operationen nun mit dem Skalieren der Form zu tun? Ok bin raus das driftet ab. gruss |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Ich habe kein Problem da ich mich NUR mit einem rumschlage. Ich möchte nicht im Elektrosmog untergehen. gruss |
AW: Formulargröße auf HighDPI-Monitor speichern
Früher hatte ich auch nur einen Monitor. Mittlerweile einen 24" und einen 21". Den zweiten möchte ich nicht mehr missen.
1x Delphi links, 1x rechts oder auf dem anderen Sublime oder irgendwas anderes. |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
Die MulDiv Funktion muss ich korrigieren. Sie war leider falsch. Nicht, dass einer die falsche kopiert. Richtig lautet sie
Delphi-Quellcode:
class function TMathUtils.MulDiv(nNumber, nNumerator, nDenominator: Integer): Integer;
begin Result := Round((nNumber * nNumerator) / nDenominator); end; |
AW: Formulargröße auf HighDPI-Monitor speichern
Ja mir war aufgefallen, dass deine nicht 100%ig jener von Delphi entspricht, aber meistens wohl genügt ;-).
Delphi setzt dies um: ![]() D.h. deine nachgebaute Funktion sollte zum Beispiel in gewissen Fällen -1 ausgeben. Fürs Runden solltest du nicht round verwenden. round(x) rundet .5 immer auf die gerade Zahl, welche x am nächsten liegt. Beispiel 0.5 wird auf 0 gerundet, 1.5 auf 2, 2.5 auf 2, 3.5 auf 4. MulDiv macht's anders. Schau dir zum Beispiel trunc an. Zum Runden in Delphi gibt's ![]() |
AW: Formulargröße auf HighDPI-Monitor speichern
Verbesserung
Delphi-Quellcode:
class function TMathUtils.MathRound(X: Extended): Int64; // Bei stackoverflow gefunden, für die Sammlung. Obwohl Trunc dasselbe macht.
begin if X >= 0 then Result := Trunc(X + 0.5) else Result := Trunc(X - 0.5); end; class function TMathUtils.MulDiv(nNumber, nNumerator, nDenominator: Integer): Integer; begin Result := Trunc((nNumber * nNumerator) / nDenominator); end; |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
gruss |
AW: Formulargröße auf HighDPI-Monitor speichern
Ja du musst aber natürlich in deiner MulDiv() deine Rundungsfunktion MathRound() verwenden - so wie von dir gepostet verwendest du trunc().
Also so:
Delphi-Quellcode:
Weiter musst du den Fall nDenominator=0 abfangen (Result := -1) und auch mögliche Überläufe ((nNumber * nNumerator) / nDenominator kann so gross werden, dass das Resultat nicht in einem integer Platz hat).
Result := MathRound( (nNumber * nNumerator) / nDenominator);
Weiteres Problem: Möglicher integer Überlauf bei der Multiplikation nNumber * nNumerator Deine MulDiv Funktion liefert zum Beispiel für
Delphi-Quellcode:
x = -2, müsste aber natürlich maxint div 2 zurückgeben.
var x : integer;
begin x := MulDiv( maxint div 2, maxint div 2, maxint div 2 ); Schreibe deshalb int64(nNumber) * int64(nNumerator) statt nNumber * nNumerator, sonst hast du wenig Freude, wenn nNumber * nNumerator > maxint. OK für deinen "Fenster-Fall" wirst du kaum je solche Werte als Argumente haben. Aber wenn du MulDiv nachbilden willst, dann musst du darauf achten. |
AW: Formulargröße auf HighDPI-Monitor speichern
Ich hätte nicht gedacht, dass die windowseigene Muldiv() das alles berücksichtigt. Was berücksichtigt die denn noch? Denn die ist scheinbar plattformspezifisch.
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
MulDivInt64(.....) aus der System Unit. Die ist crossplatform angelegt sollte also keine Problem machen. |
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Zitat:
|
AW: Formulargröße auf HighDPI-Monitor speichern
Habe kurz in den Code reingeguckt und der scheint grob überschlagen 10x länger zu sein als der aus diesem Thread hier.
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 16:45 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz