![]() |
Repaint beim Form-Resize beschränken
Liste der Anhänge anzeigen (Anzahl: 1)
Heyho,
ich habe eine Form, welcher ich mit der Free-Version der Alpha Controls einen Skin zugewiesen habe, außerdem sind u.A. noch eine PageControl und mehrere Buttons auf der Form. Beim Resize der Form kommt es nun zu extremem Flackern bei den Buttons, der PageControl (der TabHeader bewegt sich deutlich sichtbar bestimmt 2mm rauf und runter) sowie den Rändern (man sieht z.B. deutlich, das erst eine Vergrößerung ausgeführt wird, der Bereich dann schwarz ist und anschließend übermalt wird). Auch wenn ich den Skin deaktiviere besteht das Problem, wenn auch weniger stark. Leider hab ich davon nicht so viel Ahnung, deswegen frag ich jetzt einfach mal ganz naiv ... Kann ich das Neuzeichnen der Form auf bestimmte Bereiche beschränken (z.B. nur den Rand, der grade resized wird), oder sonst irgendwas machen, so das das Resizen der Form vernünftig und sauber aussieht, so wie bei allen anderen Anwendungen auch ? ;) Meine Recherchen dazu haben mich bis jetzt kaum weiter gebracht, DoubleBuffered auf der Form und / oder einzelnen Komponenten ändert genau garnix. Ich hab gedacht, vlt könnte man ne Region erstellen und das Update auf diese beschränken, ob (und wenn ja vor allem wie) das geht hab ich aber keinen Plan ... vielleicht gibts auch noch bessere Lösungsansätze. Hoffe jemand kann mir helfen, danke im voraus =) Hinweis: Ich hab die Exe mal in nem Verzeichnis angehängt, damit ihr seht was ich meine. Bitte nicht auf die Buttons klicken (dann stürzt das Programm ab, da ich gewisse Pfade statisch gesetzt habe bzw. auf andere Programme zugreife), es geht nur um das resize. Hoffe das Ding läuft so wies is überhaupt auf anderen Rechnern ... |
AW: Repaint beim Form-Resize beschränken
Hast Du es mal mit DoubleBuffered probiert?
|
AW: Repaint beim Form-Resize beschränken
Ich denke, das Problem wirst Du nicht abstellen können.
Es gab mal eine Unit FlickerReduce.pas (o.ä.), die das Problem eingeschränkt hat, aber andere Probleme verursachen konnte. Unter D2009 waren die Optionsformulare z.B. teilweise nicht mehr nutzbar. Dass es in geskinten Forms stärker flackert, ist nachvollziehbar, da dort natürlich aufwendiger gezeichnet wird und das länger dauert. Welche "alle anderen Anwendungen" meinst Du denn, die das Problem nicht haben? (Delphianwendungen mit Skins?) |
AW: Repaint beim Form-Resize beschränken
Das Problem, dass es ab und an ein wenig flackert, haben meiner Meinung nach im gewissen Maß leider alle "geskinnten" Anwendungen.
(habe schon einige geskinnte Anwendungen erstellt...gerade bei vielen, etwas älteren OnBoard-Grafikchips mit schwindligen Treibern flackert es fast immer ein wenig...) Dagegen kann man generell, bis auf DoubleBuffering wie es bereits gesagt wurde, nicht viel machen. Um das Flackern einer geskinnten Anwendung unter Allen Umständen komplett abzustellen, vergiss AlphaControls, VCLSkin und Konsorten, dafür gibt es dann nur einen vernünftigen Weg: VCL wegschmeissen, SDL nehmen, eigene GUI-Library auf DirectX oder OpenGL bauen und damit dann die Anwendung erstellen....was aber zugegebenermaßen horrormäßig viel Arbeit ist und zudem die Grafikkarte dann DX/OpenGL-Treiber besitzen muss :-) |
AW: Repaint beim Form-Resize beschränken
Zitat:
Das eigentliche Problem liegt darin, daß generell in zwei Schritten aktualisiert wird: Auf WM_ERASEBKGND wird das Element mit der Hintergrundfarbe übermalt und in WM_PAINT neu dargestellt. Zwischen beiden Ereignissen liegt logischerweise etwas Zeit. Und genau die bewirkt das Flackern. Dazu kommt noch, daß Steuerelemte - wenn diese nicht per WS_CLIPCHILDREN aus dem DC des Parents herausgenommen wurden - auch erst einmal gelöscht werden. Eine ähnliche Unsitte ist es, Controls bei der Darstellung erst einmal komplett zu löschen. Ein unrühmliches Beispiel dafür sind Grids. Anstatt zuerst die Zellen zu zeichnen, deren Bereich aus dem DC per Clipping herauszunehmen und dann die verbliebene Update-Region mit der Hintergrundfarbe zu füllen, wird meist erst einmal alles gelöscht. Wurden solche Dinge beachtet, war es auch schon unter Windows 3.1 und einem 220 MHz-Prozessor möglich, flackerfreie Anwendungen zu entwickeln - und das völlig ohne Hardwarebeschleunigung. Auch mit VCL! Wenn das Komponentenhersteller vergessen, ist das einfach nur traurig und zeugt nicht gerade von Kompetenz. |
AW: Repaint beim Form-Resize beschränken
Yoyo,
Zitat:
Zitat:
Zitat:
Zitat:
@Thom: Deine Erklärung klingt logisch, ich hab mich schon immer gefragt wieso immer alles neu gezeichnet werden muss, und ob man das nicht irgendwie effizienter gestalten könnte ... aber kann ich da als "Anwender" im Nachinein noch irgendwas machen ? Zusammenfassend: Eure Antworten sehen ja ziemlich duster aus, trotzdem hatte ich eigentlich nicht vor so schnell aufzugeben *g* Um nochmal auf meinen Ausgangsvorgang zurückzukommen: kann man nicht irgendwie ne Region definieren, und diese dann für die Dauer des Resizens vom Update ausschließen? Also ganz naiv einfach eine Methode finden, Rect(0,0,500,500) [oder wie viel man halt braucht, nur so als Beispiel] in nen Handle zu transferieren und den dann mit LockWindowUpdate oder der entsprechenden Message während des Resizens einfach in Ruhe zu lassen ? Wie gesagt, ich hab davon nicht viel Ahnung, weiß der Geier ob das irgendwie geht, ob das Sinn machen würde oder obs da was besseres gibt ... Aber danke schonmal für eure Antworten, freut mich das es so viele Leute gibt die bereit sind, sich mit den Problemen anderer auseinander zu setzen =) |
AW: Repaint beim Form-Resize beschränken
Ich weis nicht, ob es bei Dir zutrifft (wegen Skins), aber ich hab' festgestellt, das ParentBackground:=False bei mir oft hilft, wenn DoublePuffered versagt.
|
AW: Repaint beim Form-Resize beschränken
Das gibt es aber nicht bei Buttons (und m.E. sowiso nicht unter D7).
Die Optimierung der VCL-Controls-Darstellungen wäre wohl auch ein Punkt pro VCL-Überarbeitung. Damit würde man allerdings die Kompatibilität alter Formulare gefährden. Vielleicht soll ja auch FireMonkey einen entsprechenden Wechsel einläuten, aber diesbezüglich bin ich noch etwas skeptisch. Mit D7 kannst Du mal die FlickerReduce.Pas testen (wenn Du sie noch irgendwo findest). Ich fand sie damals hilfreich (allerdings ohne Skin-Controls). Unter D2009 allerdings dann nicht mehr. |
AW: Repaint beim Form-Resize beschränken
@Thom:
Du hast schon recht mit deinen Worten. Ich meinte aber mit "VCL wegschmeissen" eher, dass man kaum drumherum kommt, das ganze selbst von Grund auf zu bauen, wenn man es richtig machen will. Die ganzen SkinControl-Komponenten, die es käuflich gibt, sind leider meist "All-In-Wonder" Tüten, die in allen erdenklichen Fällen funktionieren müssen, und deswegen haben sie oft lahmen Code drin und sind kaum optimiert. (Wenn man sich den Sourcecode von so manchen SkinControls ansieht, ich will hier keine Namen nennen..hust...sieht man das schon...) Man muss die VCL natürlich nicht wegschmeissen, um es auch vernünftig machen zu können, allerdings würde ich dann gleich auf SDL+DirectX oder SDL+OpenGL gehen, wenn ich mir alles selbst bauen würde und die VCL hinter mir lassen. Das meinte ich damit. Zwar mega viel Arbeit, aber durchaus interessant und man lernt viel, ohne den Ballast der VCL mitzuschleppen, bei der man eh jedes Control modifizieren müsste. |
AW: Repaint beim Form-Resize beschränken
Die VCLFlickerReduce.pas muss ich doch nur bei uses mit aufführen, oder ?
Wenn ja, dann bringt es leider auch garnix ... Auch das mit dem ParentBackground leider keine Veränderung. Grr. Das Ding will nich ;) |
AW: Repaint beim Form-Resize beschränken
Ja, die FlickerReduce musste nur in die uses aufgenommen werden.
Vermutlich wird das dann mit den AlphaControls nicht funktionieren. Mit vertretbarem Aufwand wirst Du wohl keine Lösung finden. Ich würde das dann so hinnehmen und nicht übermäßig dramatisieren ;-) So oft wird der User die Formulargröße sicher nicht ändern bzw. sich nicht über etwas Flackern dabei aufregen. |
AW: Repaint beim Form-Resize beschränken
Auf die Gefahr hin damit zu nerven, wie siehts mit diesem Vorschlag aus ? ;)
Zitat:
Trotzdem, finde es verwunderlich, das es für sowas keine Lösung gibt, da muss der programmiertechnische "Murks" der da in dem Kram drin ist ja wirklich tief sitzen, wenn das selbst die Experten hier nicht umgehen können ... ;) |
AW: Repaint beim Form-Resize beschränken
Zitat:
Zitat:
Mir fällt gerade ein: Ich habe mal ein Demovideo gesehen, in dem ein FireMonkey-Formular stufenlos vergrößert und dabei flackerfrei gezoomt wurde. Also die VCL ist wohl einfach in der Beziehung (und nicht nur da) recht veraltet. |
AW: Repaint beim Form-Resize beschränken
Hallo,
gab es bei D7 schon Anchors ? Wenn ja, nimm die mal komplett beim PageControl raus setze die Werte direkt im FormResize. Heiko |
AW: Repaint beim Form-Resize beschränken
Ja, Anchors gabs in D7 schon. Aber nein, hilft leider auch nich, weder wenn ich sie bei der PageControl oder bei der Form rausnehme und dann dynamisch oder auch garnicht setzte, keinerlei Effekt ...
|
AW: Repaint beim Form-Resize beschränken
Hast Du es schon mit dem VCLFixPack von Andy versucht? (
![]() Zitat:
|
AW: Repaint beim Form-Resize beschränken
Auch das mit dem Fix Pack wird von meinen flackernden Komponenten völlig ignoriert ...
Das Übel scheint aber wirklich großteils auf der PageControl zu liegen. Buttons die da drauf sind flackern wie nen Blinklicht beim Resizen, die Buttons die weiter unten auf dem Panel sind verhalten sich völlig ruhig ... |
AW: Repaint beim Form-Resize beschränken
Pagecontrol war jetzt das Stichwort...
Wir hatten bei einem Projekt mit GDI+-Darstellung von Maschinendaten durch eigen Komponenten auch ein massives Flickerproblem mit PageControls. Wir haben es an den Stellen auf ein Panel mit einem eigenen Tabcontrol umgestellt, in welches Rahmenlose Forms (alClient) geladen wurden, beim Tabwechsel mit BringToFront ensprechend angezeigt. Das geflickere war damit Geschichte. |
AW: Repaint beim Form-Resize beschränken
Das kann ich bestätigen...PageControls haben bei uns auch immer die meisten Schwierigkeiten gemacht, wenn es ums Skinning geht..von Geflackere bis hin zu völligem Verschwinden der darauf liegenden Controls..also am Besten beim Skinning PageControls vermeiden, entweder selbst eine ähnliche Komponente schreiben oder eine elegante Lösung wie Bummi sie beschrieben hat verwenden :-)
|
AW: Repaint beim Form-Resize beschränken
Ich hab jetzt nicht jeden Beitrag hier gelesen, aber weil eigenes GUI-Framework angesprochen wurde: Afair hat Andorra 2d sowas (wenn ich mich nicht irre sogar mit graphischem Form-Designer). Wäre vielleicht einen Blick wert.
|
AW: Repaint beim Form-Resize beschränken
Heyhey,
Zitat:
Danke für den Tip, das werde ich jetzt so machen ! Funktioniert soweit auch super, ein kleines Problem habe ich aber noch: Die einzelnen Forms habe ich einfach per CreateParented integriert, ist auch alles in Ordnung. Allerdings habe ich ihren Borderstyle natürlich auf bsNone gesetzt, was dazu führt, dass sie keine Scrollbars mehr anzeigen ... Also: jemand ne Idee wie ich die Form trotz bsNone zum anzeigen von ScrollBars bewege, wenn der Inhalt größer als die Form ist ? Eine Scrollbox möchte ich explizit nicht verwenden, da die Scrollbars der Form seltsamerweise die einzigen sind die mitgeskinnt werden & ich mir das überlagern von Windows-Scrollbars mit geskinneten um das Verhalten zu simulieren sparen möchte. Man kann doch bestimmt über CreateParams der Childform irgendwie einstellen, das sie trotzdem Scrollbars anzeigt, oder ? Zitat:
Danke, ebenfalls interessanter Hinweis. Wenn ich Zeit & Lust habe werden ich mal versuchen mich da einzuarbeiten, für den Moment werde ich es aber weiter ohne probieren ;) |
AW: Repaint beim Form-Resize beschränken
wir arbeiten grundsätzlich mit Templates, deren überladene Konstruktoren deutlich umfangreicher als das folgende Beispiel sind (unter anderem wird ein Zeiger auf die Variable (kann nil sein) der zugewiesen wird, mit übergeben die im Destruktor wenn gesetzt "genilt" wird)
Delphi-Quellcode:
Constructor Create(AOwner:TComponent;UseAutoScroll:Boolean=false);Overload;
{ Public-Deklarationen } end; var Form2: TForm2; implementation {$R *.dfm} { TForm2 } constructor TTemplate.Create(AOwner: TComponent; AParent: TwinControl;UseAutoScroll:Boolean=false); begin inherited Create(AOwner); if Assigned(Aparent) then begin borderStyle := bsNone; Align := alClient; Parent := AParent; Show; AutoScroll := UseAutoScroll; end; end; |
AW: Repaint beim Form-Resize beschränken
Das Problem besteht ja bei mir nicht in der Tatsache, dass ich die Form nicht eingebunden bekomme, sondern darin das ich trotz bsNone die Scrollbars der Form brauche (Autoscroll wird bei bsNone ignoriert).
|
AW: Repaint beim Form-Resize beschränken
Mit embedded Forms würde es wohl funktionieren:
Delphi-Quellcode:
FormXyz.ManualDock(TabSheetXyz, nil, alNone);
FormXyz.Align := alClient; FormXyz.Show; (Im Beispiel setze ich das Formular übrigens in ein TabSheet und da flackern die Controls auch beim Resizen - ebenso wie GroupBoxen. Ich nutze hier embedded Forms aus Gründen der Übersichtlichkeit und Projektaufteilung.) Embedded Forms in TabControls flackern widerum nicht. Allerdings muss man dann dynamisch beim umblättern das passende Form einblenden. Das habe ich so getan:
Delphi-Quellcode:
procedure TFormPlan.odTabControlTournamentsPlanShowRegister(Sender: TObject);
procedure ShowForm(F: TForm); procedure DoShowForm(SF, F: TForm; FC: TodFormCtrl); begin if SF = F then begin if (SF.Parent <> PanelTournamentsPlan) or (not SF.Visible) then begin SF.ManualDock(PanelTournamentsPlan, nil, alNone); SF.Align := alClient; SF.Show; end; FC.od := odTabControlTournamentsPlan.ActiveOd; if Assigned(SF.OnShow) then SF.OnShow(Self); end else begin FC.od := nil; SF.Close; end; end; begin CourtTournament := odTabControlTournamentsPlan.ActiveOd as TodTournament; DoShowForm(FormPlanMelee, F, FormPlanMelee.odFormCtrl); DoShowForm(FormPlanSwiss, F, FormPlanSwiss.odFormCtrl); DoShowForm(FormPlanGroup, F, FormPlanGroup.odFormCtrl); DoShowForm(FormPlanKo, F, FormPlanKo.odFormCtrl); end; begin if odTournament = nil then ComboBoxTournamentSystem.ItemIndex := -1 else if odTournament.Data_Is_Melee then ComboBoxTournamentSystem.ItemIndex := 0 else if odTournament.Data_Is_Swiss then ComboBoxTournamentSystem.ItemIndex := 1 else if odTournament.Data_Is_Group then ComboBoxTournamentSystem.ItemIndex := 2 else if odTournament.Data_Is_Ko then ComboBoxTournamentSystem.ItemIndex := 3 else ComboBoxTournamentSystem.ItemIndex := -1; case ComboBoxTournamentSystem.ItemIndex of 0: ShowForm(FormPlanMelee); 1: ShowForm(FormPlanSwiss); 2: ShowForm(FormPlanGroup); 3: ShowForm(FormPlanKo); else ShowForm(nil); end; ComboBoxTournamentSystem.Enabled := ComboBoxTournamentSystem.Text = ''; end; |
AW: Repaint beim Form-Resize beschränken
Delphi-Quellcode:
ich habe leider kein D7 zu Testen zur Hand, der Code den ich eingestellt habe funktioniert unter unter D2010 + problemlos ....
sondern darin das ich trotz bsNone die Scrollbars der Form brauche (Autoscroll wird bei bsNone ignoriert).
|
AW: Repaint beim Form-Resize beschränken
Vielleicht hab ich das auch einfach nur falsch umgesetzt, aber auch wenn ich die Form nach der beschriebenen Methode andocke bleibt das Problem, das die Form bei bsNone keine Scrollbars erzeugt bestehen ...
Edit: Menschliches Versagen beim Umsetzen von Bummis Code, funktioniert natürlich. Aber: Das Ding macht mich trotzdem fertig ... Wenn ich das genau so mache skinnt er die Scrollbars einfach nicht mit, ändere ich zum testen bsNone zu bsSizeable habe ich stattdessen ne Form mit Header ABER dafür mit geskinnten Scrollbars ... |
AW: Repaint beim Form-Resize beschränken
wenn Du den Code aus #22 erfolglos getestet hast muss ich Dich auf Montag vertrösten, im Büro habe ich noch ein paar VM's mit D7 ....
|
AW: Repaint beim Form-Resize beschränken
Kann natürlich auch sein, dass die AlphaControls (vielleicht auch nur in Verbindung mit D7) da nicht korrekt mitspielen. Ist ja auch ein seltener Sonderfall, solch eine Anwendung.
|
AW: Repaint beim Form-Resize beschränken
Hi,
habe nicht viel Ahnung vom Thema. In einem Richedit mit vielen Schreibaktionen half dies:
Delphi-Quellcode:
Mfg Uwe
LockWindowUpdate(self.Handle);
//hier alle Zeichenaktionen LockWindowUpdate(0); |
AW: Repaint beim Form-Resize beschränken
Mhh ... was mir jetzt noch einfallen würde, wäre
1) Über CreateParams (Style) zu sagen, das auch bei bsNone Scrollbars gezeichnet werden sollen, oder 2) Den Header durch irgendeinen Trick trotz bsSizeable nicht anzeigen 3) Entweder mit den Windows-Scrollbars leben, mit dem Formheader in der anderen Form oder geskinnte Scrollbars drüberzeichnen und das Verhalten imitieren. Wobei ich natürlich glücklicher wäre wenn 1 oder 2 irgendwie machbar sind ... ^^ Edit: Wüsste nicht wie mir LockWindowUpdate helfen sollte, wenn meine Form entweder die falschen oder gakeine Scrollbars malt ... |
AW: Repaint beim Form-Resize beschränken
LockWindowUpdate soll jedoch genau für soetwas nicht verwendet werden.
Zitat:
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 01:17 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