Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   GUI-Design mit VCL / FireMonkey / Common Controls (https://www.delphipraxis.net/18-gui-design-mit-vcl-firemonkey-common-controls/)
-   -   Tastaturpuffer löschen? (https://www.delphipraxis.net/168273-tastaturpuffer-loeschen.html)

Popov 12. Mai 2012 12:53

Tastaturpuffer löschen?
 
Ein kleines Problem: mein Tool reagiert auf zwei Tasten in OnKeyDown, auf Escape und Return. Beim drücken der Return Taste wird eine Funktion ausgeführt die etwa 1 Sekunde dauert. Klicke ich 10 mal auf Return, wird das anscheinend nacheinander abgearbeitet. Drücke ich 10 mal Return und dann Escape, dauert es somit 10 Sekunden bis das Programm beendet wird.

So wie ich das beobachte wird erst die Funktion abgearbeitet, bevor OnKeyDown wieder eine Taste empfängt.

Bevor ich etwas unnötiges proge, gibt es eine Möglichkeit am ende der Funktion den Tastaturpuffer zu löschen, so dass erst danach Tasten aktuell sind?

Luckie 12. Mai 2012 13:10

AW: Tastaturpuffer löschen?
 
Soll die Funktion nur einmal ausgeführt werden, auch wenn du 10 mal Return drückst?

Popov 12. Mai 2012 13:29

AW: Tastaturpuffer löschen?
 
Muß nicht sein. Eigentlich soll sie nur ausgeführt werden wenn Return gedrückt wird. Theoretisch kommt es nicht vor, dass man Return mehrmals nacheinander drückt. Nur habe ich zufällig festgestellt, dass man es machen könnte und dann wird die Funktion mehrmals ausgeführt. Wäre an für sich keine Katastrophe, wenn es die Esc Taste nicht blockieren würde.

Oder anders ausgedrückt: wenn man Return drei mal nacheinander geklickt hat, dann zwei mal zuviel.

EWeiss 12. Mai 2012 14:28

AW: Tastaturpuffer löschen?
 
Tastaturpuffer leeren
Selbst nicht getestet.

gruss

Namenloser 12. Mai 2012 16:16

AW: Tastaturpuffer löschen?
 
Ist das wirklich der Tastaturpuffer? Oder nicht vielleicht eher der Message-Queue?

Das hier sollte alle Tastatur-Messages im Queue löschen:
Delphi-Quellcode:
procedure TfrmMain.ClearKeyboardMessages;
var
  Msg: TMsg;
begin
  while PeekMessage(Msg, 0, 0, 0, PM_REMOVE or (QS_KEY shl 16)) do
  begin
// Normalerweise:
//    TranslateMessage(Msg);
//    DispatchMessage(Msg);
// Aber da wir die Messages ignorieren, tun wir einfach nix
  end;
end;
Ist ungetestet, aber der Code ist nur minimal aus einer ähnlichen, funktionierenden Prozedur abgewandelt.

Mit PeekMessage gehen aber auch noch andere coole Sachen, z.B. kannst du gucken, ob irgendwo später im Message-Queue die Escape-Taste auftaucht, und dann abbrechen – ohne, dass der ganze Message-Queue dabei draufgeht. MSDN hat ein Beispiel, das ziemlich genau dein Problem abbildet.
Zitat:

Occasionally, an application needs to examine the contents of a thread's message queue from outside the thread's message loop. For example, if an application's window procedure performs a lengthy drawing operation, you may want the user to be able to interrupt the operation.
(Ich bin mir gerade nicht sicher, ob du nicht der warst, der in einem anderen Thread geschrieben hatte, dass er kein Englisch kann, daher mal kurz eine deutsche Übersetzung: „Manchmal muss eine Anwendung von außerhalb der Message-Loop eines Threads die Inhalte von dessen Message-Queue untersuchen. Wenn zum Beispiel die Fenster-Routine einer Anwendung eine längere Zeichen-Operation ausführt, willst du dem User u.u. die Möglichkeit geben, die Operation unterbrechen.“)

Popov 14. Mai 2012 06:19

AW: Tastaturpuffer löschen?
 
Ich habe beides noch nicht zum laufen gekriegt, oder zumindest nicht so wie ich es mir vorstelle.

Ich habe das Problem vorläufig simpel gelöst:
Delphi-Quellcode:
  if IsBusy then Exit;
  IsBusy := True;
  MachWas; //inc. ProcessMessages
  IsBusy := False;
Das, aber nur in Verbindung mit ProcessMessages, funktioniert sogar brauchbar. Neue Anweisungen werden solange abgewiesen bis die Funktion fertig ist.

Trotzdem interessiert mich das Leeren des Puffers, nur jetzt eher allgemein. Ich gucke mal wieso es nicht so geklappt hat, vielleicht lag der Fehler bei mir.

Ich hab es getestet, es ist tatsächlich so, dass, wenn eine Funktion eine Sekunde in Anspruch nimmt, der zweite Tastendruck erst eine Sekunde später OnKeyDown erreicht.


Alle Zeitangaben in WEZ +1. Es ist jetzt 06:22 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-2025 by Thomas Breitkreuz