![]() |
Für CopyRect passendes Ereignis?
Hallo,
ich habe folgende Procedure (ohne Fehler):
Delphi-Quellcode:
Es wird also ein Stück Bild kopiert und in ein anderes gefüllt. Nun soll das sofort beim Start der Anwendung passieren, aber ich finde kein passendes Ereignis dafür. OnCreate, OnShow, OnPaint oder OnResize von Form1 bring da nichts, das Ziel wird schwarz.
procedure TMainForm.BackgroundToSpectrumImage();
var Quelle, Ziel: TRect; begin Quelle := Rect(SpectrumBackgroundImage.Left{=13}, SpectrumBackgroundImage.Top{=112}, SpectrumBackgroundImage.Left + SpectrumBackgroundImage.Width{=42}, SpectrumBackgroundImage.Top + SpectrumBackgroundImage.Height{=184}); Ziel := Rect(0, 0, SpectrumBackgroundImage.Width{=30}, SpectrumBackgroundImage.Height{=73}); SpectrumBackgroundImage.Canvas.CopyRect(Ziel, MainBackgroundImage.Canvas, Quelle); end; MainBackgroundImagePictureChanged ruft einen Fehler aus... jah... bitte hier also um Hilfe. P.S. ich programmiere mit Lazarus. |
AW: Für CopyRect passendes Ereignis?
"Sofort beim Start" heisst eigentlich OnCreate. Wann füllst du denn das Quellbild mit Inhalt? Das muss natürlich vor dem Umkopieren geschehen. Im Zweifel schickst du dir selbst halt eine Message von der Methode, die das Quellbild malt ans MainForm und reagierst darauf.
|
AW: Für CopyRect passendes Ereignis?
Zitat:
|
AW: Für CopyRect passendes Ereignis?
Was spricht dagegen, die bewußte Methode dann einfach am Ende von OnShow aufzurufen? Nach dem OnShow passiert erstmal nichts mehr, bis der Anwender oder sonstwert ein Ereignis auslöst. Was bringt dich zu der Annahme, daß das Bild erst nach OnShow aus der Ressourcendatei in das TImage geladen wird?
|
AW: Für CopyRect passendes Ereignis?
In der Regel erfolgt das Malen im OnPaint Event.
Wenn man jetzt im OnPaint des MainBackgroundImages ein SpectrumBackgroundImage.Invalidate aufruft, sorgt man dafür, dass das neu gezeichnet wird. Und im OnPaint des SpectrumBackgroundImage holt man sich das darunterliegende Bild. Um die Koordinaten herauszufinden gibt es im Übrigen ClientToScreen und ScreenToClient Funktionen. ![]() (Sollte mit FPC analog funktionieren) @Perlsau Das Bild ist beim ersten OnShow schon geladen aber noch nicht gezeichnet, denn der Event kommt direkt vor dem Anzeigen. Wenn man jetzt vom Canvas etwas kopiert, auf dem noch nichts gezeichnet wurde, dann kopiert man nichts ;) |
AW: Für CopyRect passendes Ereignis?
@Sir Rufo
Würde da nicht ein Application.ProcessMessages das Zeichnen des geladenen Bildes anstoßen? Und erst danach kopiert man. |
AW: Für CopyRect passendes Ereignis?
Nicht zwangsläufig. OnShow beschreibt die Transition der Visible Eigenschaft von False nach True.
Ich bin immer skeptisch bei Application.ProcessMessages und innerhalb eines (nicht von mir definiertem) Event sowieso. Das kann eben auch nach hinten losgehen, weil sich z.B. die Implementierung auf eine bestimmte Reihenfolge velässt und man die damit etwas durcheinander bringen kann. Zudem kann die Message zum Zeichnen der Form auch erst nach dem OnShow geschickt werden, dann bringt es nichts die Nachrichten-Warteschlange abzuarbeiten. Die VCL-Form ist gesichert komplett gezeichnet beim ersten Application.OnIdle Event nach dem Form.OnShow Event. |
AW: Für CopyRect passendes Ereignis?
Der sauberste Weg wäre es in dem Fall doch, das Bitmap nicht über den Form-Designer zu laden, sondern es selbst in eine Ressource zu packen. Dann hat man den Ladezeitpunkt definitiv in der Hand, kann das im OnCreate erledigen, und direkt danach gut leserlich und nachvollziehbar die Kopie machen.
|
AW: Für CopyRect passendes Ereignis?
Zitat:
|
AW: Für CopyRect passendes Ereignis?
Im Kompilat gibt es einen eigenen Ressourcen-Abschnitt, darin landen u.a. alle eingebundenen Bilder. Du musst sie also nicht extern mitliefern.
|
AW: Für CopyRect passendes Ereignis?
Zitat:
|
AW: Für CopyRect passendes Ereignis?
Zitat:
Delphi-Quellcode:
procedure TForm1.FormShow(Sender: TObject);
var Quelle, Ziel: TRect; begin Image1.Picture.LoadFromLazarusResource('bg'); Image2.Picture.LoadFromLazarusResource('5ge'); Quelle := Rect(0, 0, 100, 100); Ziel := Rect(0, 0, 100, 100); Image2.Canvas.CopyRect(Ziel, Image1.Canvas, Quelle); end; procedure TForm1.Button2Click(Sender: TObject); var Quelle, Ziel: TRect; begin Quelle := Rect(0, 0, 100, 100); Ziel := Rect(0, 0, 100, 100); Image2.Canvas.CopyRect(Ziel, Image1.Canvas, Quelle); end; |
AW: Für CopyRect passendes Ereignis?
Was passiert denn, wenn Du auf den Button klickst und anschließend mal ein anderes Fenster über Deins legst?
|
AW: Für CopyRect passendes Ereignis?
Zitat:
Delphi-Quellcode:
.
OnShow
|
AW: Für CopyRect passendes Ereignis?
Zitat:
|
AW: Für CopyRect passendes Ereignis?
Zitat:
|
AW: Für CopyRect passendes Ereignis?
Im oder nach OnPaint?
|
AW: Für CopyRect passendes Ereignis?
Zitat:
|
AW: Für CopyRect passendes Ereignis?
Guck mal hier
![]() |
AW: Für CopyRect passendes Ereignis?
Oben hatte SirRufo doch einen Hinweis auf OnIdle geliefert:
Die VCL-Form ist gesichert komplett gezeichnet beim ersten Application.OnIdle Event nach dem Form.OnShow Event. Ist dieser Hinweis denn nicht brauchbar? Könnte man denn nicht einfach in der OnIdle-Ereignisbehandlung eine globale Boolean-Variable, die als Schalter fungiert, abfragen, so daß die Anweisungen nur dieses eine Mal ausgeführt werden? Z.B. so:
Delphi-Quellcode:
Nachtrag:
...
private Var FirstIdlePassed : Boolean; public procedure MyOnIdleHandler(Sender: TObject; var Done: Boolean); end; procedure TForm1.MyOnIdleHandler(Sender: TObject; var Done: Boolean); begin Done := False; If FirstIdlePassed Then Exit; FirstIdlePassed := True; ShowMessage('Das erste OnIdle-Event!'); ... end; procedure TForm1.FormCreate(Sender: TObject); begin FirstIdlePassed := False; Application.OnIdle := MyOnIdleHandler; end; Ob das dann auch in Lazarus funktioniert, muß man gegebenenfalls ausprobieren ... Ach ja, bei Lazarus muß man das @ davor setzen:
Delphi-Quellcode:
Application.OnIdle := @MyOnIdleHandler;
|
AW: Für CopyRect passendes Ereignis?
@Perlsau danke Dir... Application.OnIdle Event war mir noch nicht vertraut.
An sich funktioniert es auch, die ShowMessage wird ausgegeben, aber das Bild wird nicht kopiert. Ich würde für noch ein Paar Ideen sehr dankbar... |
AW: Für CopyRect passendes Ereignis?
Wobei ist das Problem mit dem Laden aus einer Ressource?
|
AW: Für CopyRect passendes Ereignis?
Zitat:
|
AW: Für CopyRect passendes Ereignis?
Zitat:
Bleibt also zu 99,9995% ein Fehler in deinem Code. :stupid: Image.Canvas ist die "sichtbare" Zeichenfläche und Image.Picture.Canvas?, Image.Picture.Graphics.Canvas oder z.B. Image.Picture.Bitmap.Canvas ist der Canvas des internen Bildes. PS: Rate mal was TImage mit dem Canvas der Oberfläche macht, wenn es das WM_PAINT bekommt. :roll: In neueren Windowsen hat man zwar oft den Vorteil, daß sich der DesktopWindowManager (DWM) dazwischen klemmt, aber normaler Weise ist der Canvas des Fensters (und allem darauf) flüchtig. |
AW: Für CopyRect passendes Ereignis?
Zitat:
Hier wird nicht das Image1 kopiert, sondern der Desktop noch bevor meine Anwendung starten. Kann also nach dem Start im Image2 die Icons sehen die auf meinem Desktop liegen.
Delphi-Quellcode:
unit Unit1;
{$mode objfpc}{$H+} interface uses Classes, SysUtils, FileUtil, Forms, Controls, Graphics, Dialogs, StdCtrls, ExtCtrls, Windows; const MY_MESSAGE = WM_USER + 1; type { TForm1 } TForm1 = class(TForm) Image1: TImage; Image2: TImage; Label1: TLabel; procedure FormShow(Sender: TObject); private { private declarations } procedure MyMessage(Var MSG: TMessage); message MY_MESSAGE; public { public declarations } end; var Form1: TForm1; implementation {$R *.lfm} { TForm1 } procedure TForm1.FormShow(Sender: TObject); begin PostMessage(Self.Handle, MY_MESSAGE, 0, 0); end; procedure TForm1.MyMessage(var MSG: TMessage); var Quelle, Ziel: TRect; begin Label1.Caption := 'Luluu'; Quelle := Classes.Rect(0, 0, 100, 100); Ziel := Classes.Rect(0, 0, 100, 100); Image2.Canvas.CopyRect(Ziel, Image1.Canvas, Quelle); end; end. |
AW: Für CopyRect passendes Ereignis?
NOCHMAL zum MITLESEN.
Das Ändern des Labels löst ein Neuzeichnen dessen Canvas aus (TLabel und TImage haben aber kein eigenes Canvas, sondern leiten die Operationen an ihren Parent weiter) Fazit: Die Form wird neu gezeichnet und schmeißt die Zeichenausgabe deines FEHLERHAFTEN Codes weg. (übermalt das wieder) Label1.Caption:=... zeichnet nicht sofort, sondern verzögert, also nach deinem Code. (quasi PostMessage WM_PAINT) Zitat:
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 08:43 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