Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Algorithmen, Datenstrukturen und Klassendesign (https://www.delphipraxis.net/78-algorithmen-datenstrukturen-und-klassendesign/)
-   -   PaintBox.Paint startet immer wieder neu (https://www.delphipraxis.net/207754-paintbox-paint-startet-immer-wieder-neu.html)

SteffenSchm 29. Apr 2021 12:19

PaintBox.Paint startet immer wieder neu
 
Hallo,

ich habe folgendes aktuelle Problem:

Im Hauptfenster meine Programms wird eine PaintBox angezeigt, welche in der Methode PaintBox.Paint neu gezeichnet wird. Das klappte bisher sehr gut (ist ja auch nichts besonderes). In einer aktuellen Anwendung des Programms soll jetzt aber eine sehr große Anzahl von Polylines dargestellt werden. Dies dauert lange, so dass man beim Aufbau des Bildes zuschauen kann. Nicht schön - wäre aber auch kein Problem.

Das Problem ist, dass die Zeichnung nie fertiggestellt wird. Nach einigen Sekunden wird das Bild jeweils verworfen und der Aufbau beginnt von vorn.

Kann mir jemand erklären, woran das liegt bzw. wie man das verhindern kann?

Besten dank im voraus!
Steffen

Uwe Raabe 29. Apr 2021 12:21

AW: PaintBox.Paint startet immer wieder neu
 
Kannst du das mit einem einfachen Beispielprogramm reproduzieren?

SteffenSchm 29. Apr 2021 12:35

AW: PaintBox.Paint startet immer wieder neu
 
Muss ich mal versuchen.
Alles drumherum rausschmeissen und nur das Problem übrig lassen - Ich melde mich!

hoika 29. Apr 2021 13:05

AW: PaintBox.Paint startet immer wieder neu
 
Hallo,
warum benutzt du nicht Invalidate bzw. InvalidateRect?

Das Paint wird ja auch Windows selbst aufgerufen.

himitsu 29. Apr 2021 13:54

AW: PaintBox.Paint startet immer wieder neu
 
gab es nicht einen "lock"-Befehl (an PaintBox oder eher am Canvas), mit dem du vor/während deinem Zeichnen die Ereignisse der Paitbox deaktivieren kannst, damit es beim Zeichnen nicht neu startet?

Ansonsten bleibt noch die Möglichkeit in ein Bitmap zu zeichnen und jenes dann im OnPaint anzuzeigen.
Und statt .Paint besser ein Refresh bzw. Invalidate aufrufen.
(alternativ ein TImage)

KodeZwerg 29. Apr 2021 13:55

AW: PaintBox.Paint startet immer wieder neu
 
PaintBox in ein Panel, Panel caption = 'painting...', PaintBox visible = false, wenn fertig visible = true ?

himitsu 29. Apr 2021 14:07

AW: PaintBox.Paint startet immer wieder neu
 
Nein.
TPaintBox hat keinen Speicher, es muß also sichtbar bleiben, sonst ist das Gemalte weg und beim nächsten Show/OnPaint wird neu gezeichnet.
Bloß was auf den sichtbaren Canvas gemalt wird, das ist auch sichtbar.

(aber gerade das ist auch ein Vorteil, z.B. gegenüber TImage ... wenn schnell genug gezeichnet werden kann, dann fällt quasi der Cache/Hintergrundbitmap/DoubleBuffered/... weg und Zeichenoperationen landen direkt in der Anzeige, ohne Umwege)

KodeZwerg 29. Apr 2021 14:27

AW: PaintBox.Paint startet immer wieder neu
 
Zitat:

Zitat von himitsu (Beitrag 1488184)
Nein.

Mist :) War ein Versuch wert

SteffenSchm 29. Apr 2021 16:22

AW: PaintBox.Paint startet immer wieder neu
 
Ich habe das Programm jetzt mit Debugger laufen lassen und dabei tritt das Problem nicht auf.
Kann es sein, dass ich innerhalb der langen Schleife in der Methode PaintBox.Paint die Steuerung mal an das Programm zurückgeben muss?

himitsu 29. Apr 2021 16:32

AW: PaintBox.Paint startet immer wieder neu
 
Nein,
jedenfalls nicht wenn du die Aktualisierung der PaintBox/Canvas während des Zeichnens nicht deaktivierst.

Sonst kann es passieren, dass mitten in deinem OnPaint ein neues OnPaint ausgelöst wird und du in einer Endlosschleife landen könntest.


Wie gesagt ihr hättet die Wahl auf ein TBitmap zu zeichnen und anschließend ein Invalidate/Refresh der PaintBox auszulösen. (bzw. im OnChange des Bitmap das Invalidate der PaintBox auslösen)
Die PaintBox zeichnet dann im OnPaint dann nur noch das Bitmap.

Oder eben TImage ... arbeitet dann so ähnlich, wie grade für die PaintBox mit TBitMap beschrieben.

SteffenSchm 29. Apr 2021 16:38

AW: PaintBox.Paint startet immer wieder neu
 
Aber ist es nicht so, dass der Debugger immer mal zwischendurch abfragt, ob irgendwelche Ereignisse eingetreten sind (z.B. Drücken der Pausentaste), welche bearbeitet werden müssen. Und kann das nicht der Grund dafür sein, dass mit Debugger die Zeichnung (wenn auch langsam) fertig gezeichnet wird, während ohne Debugger immer wieder von vorn angefangen wird.

Es muss doch einen Grund geben, warum das mit Debugger funktioniert und ohne nicht!

jfheins 29. Apr 2021 16:52

AW: PaintBox.Paint startet immer wieder neu
 
Zitat:

Zitat von SteffenSchm (Beitrag 1488207)
Ich habe das Programm jetzt mit Debugger laufen lassen und dabei tritt das Problem nicht auf.
Kann es sein, dass ich innerhalb der langen Schleife in der Methode PaintBox.Paint die Steuerung mal an das Programm zurückgeben muss?

Neee. Aber es sollte halt auch keine lange Schleife sein. Falls die Grafik zu aufwändig ist, evtl. auf ein Bitmap malen und im Paintbox OnPaint nur das Bitmap kopieren.

Oder die Daten runter samplen. Es bringt ja nix, pro Pixel 3 Datenpunkte einzuzeichnen ;-)

Andreas13 29. Apr 2021 16:55

AW: PaintBox.Paint startet immer wieder neu
 
Zitat:

Zitat von SteffenSchm (Beitrag 1488162)
Das Problem ist, dass die Zeichnung nie fertiggestellt wird. Nach einigen Sekunden wird das Bild jeweils verworfen und der Aufbau beginnt von vorn.

Hallo Steffen,
ich hatte vor einigen Monaten ein ähnliches Problem mit einer DLL für Excel. Nach dem Schließen von Excel, wurde dieses immer wieder neu gestartet, aber nur, wenn ich die DLL als Release kompiliert hatte. Im Falle von Debug trat der Fehler merkwürdigerweise nicht auf, so ähnlich wie bei Dir.
Der Fehler lag bei mir daran, daß ich an einer Stelle die Längen von Excel-LongInt und Dalphi-Integer falsch angepaßt hatte.
Es könnte sein, daß dies Dir vielleicht etwas weiterhilft, vermutlich nicht...
Gruß, Andreas

SteffenSchm 29. Apr 2021 17:03

AW: PaintBox.Paint startet immer wieder neu
 
Ich habe jetzt in die Schleife die innerhalb PaintBox.Paint aufgerufen wird die folgende Zeile eingefügt:

Delphi-Quellcode:
Application.ProcessMessages;


Jetzt schafft es das Programm die Zeichnung zu Ende zu zeichen ohne immer wieder von vorn anzufangen.
Auch wenn ich nicht weiß warum, so scheint es doch etwas damit zu tun zu haben.

Schönen Abend und Danke für die Hinweise und Eure Zeit!

Steffen

Uwe Raabe 29. Apr 2021 17:13

AW: PaintBox.Paint startet immer wieder neu
 
Damit hast du das Problem aber nur kaschiert ohne es zu verstehen und zu lösen. Beim nächsten Fehler sind deine Probleme dann nur noch größer.

Michael II 29. Apr 2021 17:15

AW: PaintBox.Paint startet immer wieder neu
 
Ohne Code von dir kann nur spekuliert werden. Zeig doch mal ein Minimalbeispiel.

Wie bereits erwähnt wurde: Wenn du den User nicht beim Zeichnen der Polyline zuschauen lassen willst, dann male deine Grafik auf eine Bitmap und gib diese am Ende der Berechnung aus.

Da die Berechnung der Grafik länger dauert: Nutze einen separaten Thread für die Berechnung, damit dein Programm während der Berechnung (zum Beispiel Verschieben des Fensters etc.) reagiert.

Wenn du eine Bitmap verwendest, kannst du im Ereignis "neu malen" (z.Bsp. wenn ein User ein anderes Fenster über dein Fenster zieht - oder wenn dein Fenster über den Bildschirmrand verschoben wird und zurück) nur die bereits berechnete Bitmap neu ausgeben. Sonst musst du wieder alles rechnen und deine Kundinnen und Kunden nutzlos warten lassen...

Wenn du in deine PaintBox ein Image legst und die berechnete Bitmap dort rein legst, dann malt sich das Ding sogar selber.

Wenn du in deinem Programm nach dem Grund (IDE so, nicht IDE anders) suchen willst, dann könntest du zum Beispiel ein TApplicationsEvents auf deine Form setzen und aufzeichnen was unter OnMessage rein kommt. Sehr wahrscheinlich würdest du rasch sehen, was anders ist. (Du könntest nat. auch ein "externes" Tool verwenden.)


In deinem Ereignis Handler Application.ProcessMessages; aufzurufen ist nicht mal für ein Hobby Programm... aber das ist ein zu weites Feld.

himitsu 29. Apr 2021 17:19

AW: PaintBox.Paint startet immer wieder neu
 
Aber wie gesagt, damit kann es passieren, dass dein Zeichnen (OnPaint) während der Arbeit neu gestartet wird und es so auch in einer Eindlosschleife einem Stacküberlauf endet.

Schlimmer wird es dann, wenn nicht nur mit lokalen Variablen gearbeitet wird, womit es dann zwischen den verschiedenen Ausführungen Überschneidungen geben kann. (z.B. das eingebettete Zeichnen ändert einen globalen Zähler/Liste und wenn es in den ersten Aufruf zurück kommt hat sich dort mitten drin in der Schleife der Zustand geändert und z.B. ein Index oder Abbruchbedingung raucht ab oder wird nie erreicht)

Mit ProcessMessages muß man echt aufpassen, da es Doppelausführungen geben kann.
z.B. in einem Button eine Schleife, die Zahlen in ein Memo ausgibt. und mitten im ProcessMessages kannst du ja nochmals auf den Button drücken, während das erste OnClick noch nicht fertig war.

himitsu 29. Apr 2021 17:26

AW: PaintBox.Paint startet immer wieder neu
 
OnMessage sind aber ausschließlich von PostMessage, was in der MessageQueue landete.
SendMessage kommt dort nicht vorbei. (dafür bräuchte man einen andren Hook)

Ja, für Vieles kann es dennoch einen Hinweis geben.
Aber beachte, dass wenn man z.B. die Messages in ein Memo loggt, das wiederrum Messages auslöst, was das Ergebnis verfälschen kann.

Michael II 29. Apr 2021 19:52

AW: PaintBox.Paint startet immer wieder neu
 
Zitat:

Zitat von himitsu (Beitrag 1488223)
OnMessage sind aber ausschließlich von PostMessage, was in der MessageQueue landete.
SendMessage kommt dort nicht vorbei. (dafür bräuchte man einen andren Hook)

Ja, für Vieles kann es dennoch einen Hinweis geben.
Aber beachte, dass wenn man z.B. die Messages in ein Memo loggt, das wiederrum Messages auslöst, was das Ergebnis verfälschen kann.

Er braucht's ja nicht für ein Memo.
Ich bin ziemlich sicher, dass es reicht und er den Unterschied dort abgreifen kann.

Und nun schreib ich's halt doch ;-): Zu deinem #7: Es ist ganz sicher auch auf deinen Kisten nicht schneller, wenn du direkt Linie für Linie in die Paintbox (Grafikkarte) malst, als wenn du zuerst im RAM alles in eine Bitmap zeichnest und dann einmal raus auf die Grafikarte schreibst. Auf meinem Uraltnotebook benötigt die direkte Paintboxkmalerei auf eine HD-Form (PaintBox alClient) für 1 Mio Zufallslinien 30 Sekunden. Wenn ich die 1 Mio Zufallslinien in die Bitmap zeichne und dann raus in die Paintbox schreibe "kostet" das 3 Sekunden.

himitsu 29. Apr 2021 20:07

AW: PaintBox.Paint startet immer wieder neu
 
War nur als Hinweis, falls er es sich "anzeigen" will, um es im laufenden Programm sehen zu können. :angle:

In ein Log oder OutputDebugStrings (wenn im Debugger, in dessen MessagesFenster, aber aufpassen, ist bissl langsamer) ginge natürlich auch.


Wer (zu)viel Misst, misst manchmal auch Mist.

Michael II 29. Apr 2021 20:17

AW: PaintBox.Paint startet immer wieder neu
 
Zitat:

Zitat von himitsu (Beitrag 1488234)
Wer (zu)viel Misst, misst manchmal auch Mist.

Denken reicht hier vollauf. Die Messung sollte das nur bestätigen ;-) ;-).

SteffenSchm 30. Apr 2021 07:50

AW: PaintBox.Paint startet immer wieder neu
 
Guten Morgen!

Besten Dank für Eure zahlreichen Hinweise.

Ich war gestern einfach froh, dass ich überhaupt eine Möglichkeit gefunden hatte, mit meinem Delphi-Programm arbeiten zu können. Ich muss nämlich heute ein Projekt abschliessen (kein IT-Projekt), für welches ich das Programm brauche. Ich bin sozusagen mein eigener (und fast einziger) Anwender dieses Programms.

Das Programm liest Polylines aus einer dxf-Datei und stellt sie in einer von Paintbox abgeleiteten WorldBox dar. Auf der Worldbox kann man in Weltkoordinaten zeichnen. Diese werden dann skaliert und gedreht, also auf Pixel umgerechnet.

Die Polylines der dxf-Datei werden in einer Liste (TObjectList) gespeichert, deren Elemente wiederum Listen (TObjectList) von Punkten sind. In PaintBoxPaint (bzw. eigentlich WorldBoxPaint) werden über eine Schleife die einzelnen Polylines gezeichnet. In dem Fall, in dem die Probleme auftraten hatte die Liste ca. 153.000 Elemente, wobei jedes Polyline mindesten zwei Punkte hat aber natürlich auch sehr viel mehr Punkte haben kann.

Ich weiß natürlich auch, dass Euch diese Erläuterungen nicht viel weiterhelfen und Quelltext viel besser wäre, aber dazu müsste ich das halbe Programm hier posten.

Beste Grüße
Steffen


Alle Zeitangaben in WEZ +1. Es ist jetzt 21:31 Uhr.

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz