![]() |
Zeilen eines TDBGrid "Blockweise" färben
Hallo,
ich weiß, wie man Zeilen eines Grids abwechselnd färbt, aber wie mach ich das Blockweise? Ich habe ein Grid, wo immer mehrere Zeilen untereinander zu einem Vorgang gehören, jede Zeile hat eine eigene ID und mehrere Zusammen / Hintereinander die selbe VorgangsID. Unterschiedlich viele Zeilen gehören so zusammen. Dementsprechend ist die Datenmenge sortiert nach "Order by VorgangsID, ID". Jetzt möchte ich, dass das Grid nicht nach jeder Zeile sondern nach jedem Vorgang die ID wechselt. Aber wie mach ich das? Was nicht klappt
Delphi-Quellcode:
Irgendwie hab ich naiv angenommen, dass beim Zeichnen des Grids die Query einmal durchlaufen wird und das dann so funzt. Aber dem ist wahrscheinlich nicht so.
// im OnOwnerDrawCell
if GridIDChangeDetected then SwitchGridColor; SetLastGridID; // usw. procedure TCheckdaten.SwitchGridColor; begin if CurrentGridColor=Color1 then CurrentGridColor:=Color2 else CurrentGridColor:=Color1; end; procedure TCheckdaten.SetLastGridID; begin Last_VG_ID:=Quelle.FieldByName('VG_ID').AsInteger; end; function TCheckdaten.GridIDChangeDetected:Boolean; begin Result:=Last_VG_ID<>Quelle.FieldByName('VG_ID').AsInteger; end; |
AW: Zeilen eines TDBGrid "Blockweise" färben
Das Malen passiert erst bei der Anzeige (nur der sichtbare Bereich) und nicht beim Laden der Daten.
Habe Deine Idee vermutlich noch nicht so ganz verstanden, aber hier müsste irgendwo eine Änderung im Quelltext stattfinden:
Delphi-Quellcode:
procedure TCheckdaten.SwitchGridColor;
begin If GridIDChangeDetected then begin if CurrentGridColor = Color1 then CurrentGridColor := Color2 else CurrentGridColor := Color1; end; end; |
AW: Zeilen eines TDBGrid "Blockweise" färben
Wenn ich Zeilenweise Färbe, dann kann ich ja so Dinge machen wie:
Wenn (Query.FieldByName('ID').AsInterger mod 2)=0 dann Farbwechsel. Das hilft mir ja nicht, denn ich muss die Zeile der jetzt gezeichneten Celle mir der Zeile davor vergleichen, ob sich bei der VG_ID eine Änderunge ergeben hat, dann Farbwechsel. Wenn beim Zeichnen halt die Query nicht von oben nach unten durchgegangen wird hilft das ganze ja auch nicht. |
AW: Zeilen eines TDBGrid "Blockweise" färben
Ich hoffe ich habe es richtig verstanden.
Ich mache es so:
Delphi-Quellcode:
procedure TRechnungen.DBGrid1DrawColumnCell(Sender: TObject;
const Rect: TRect; DataCol: Integer; Column: TColumn; State: TGridDrawState); begin if (gdSelected in State) or (gdFocused in State) then DBGrid1.Canvas.Font.Color := clBlue; if DataModule2.Rechnung.FieldByName('Bezahlt').AsBoolean=True then Begin DBGrid1.Canvas.Brush.Color := clGray; end; if DataModule2.Rechnung.FieldByName('Bezahlt').AsBoolean=False then Begin DBGrid1.Canvas.Brush.Color := clMoneyGreen; end; DBGrid1.DefaultDrawColumnCell(Rect, DataCol, Column, State); end; |
AW: Zeilen eines TDBGrid "Blockweise" färben
Hallo,
ich würde mir eine Hilfsklasse bauen, z.B. eine StringList. Die bekommt so viele Einträge wie Grid-Zeilen (DBGrid.Dataet.RowCount) Du füllst du mit einfach mit passenden Zahlen (0,1, oder sowas) für die einzelnen Blöcke, und benutzt die StringList dann im OnDrawCell. Viel einfacher hättest Du mit einem normalen TStringGrid, dort könntest du mit Objects oder eine versteckten Spalte. |
AW: Zeilen eines TDBGrid "Blockweise" färben
Hallo,
sorry die späte Meldung, war halt Wochenende, und da bleibt der Rechner aus :-D. Also prinzipiell funktioniert das schon, wie ich das oben beschrieben habe, ich hatte nur einen Denkfehler und hab die falsche ID verglichen. Das heißt ich kann die Zeilen Blockweise alternierend einfärben, zumindest beim ersten mal! Das neue Problem ist nämlich jetzt, dass die ausgewählte Zeile (gdSelected in State) wie üblich anders gefärbt ist, gehighlighted. Das ist natürlich für diese Zeile kein Problem. Aber wenn ich nun in eine andere Zeile klicke, so wird diese nun gehighlighted, aber die alte Zeile muss nun ja wieder neu gezeichnet werden und da fehlt dann halt der Kontext, da eben nur einzelne Zeilen neu gezeichnet werden und nicht das ganze Grid. Ich müsste vielleicht sowas machen wie OnSelectionChange->Grid.Repaint, aber ich hab da im Grid kein Event für gefunden. |
AW: Zeilen eines TDBGrid "Blockweise" färben
welches Grid Ereignis benutzt du DrawColumnCell ?
|
AW: Zeilen eines TDBGrid "Blockweise" färben
Zitat:
|
AW: Zeilen eines TDBGrid "Blockweise" färben
Liste der Anhänge anzeigen (Anzahl: 1)
OK, hast du mein Beispiel aus #4 probiert
Delphi-Quellcode:
So färbe ich Zeilen. Die Sortierung (Gruppen) macht ja eigentlich ein SQL-Script.
procedure TRechnungen.DBGrid1DrawColumnCell(Sender: TObject;
const Rect: TRect; DataCol: Integer; Column: TColumn; State: TGridDrawState); begin if (gdSelected in State) or (gdFocused in State) then DBGrid1.Canvas.Font.Color := clBlue; if DataModule2.Rechnung.FieldByName('Bezahlt').AsBoolean=True then Begin DBGrid1.Canvas.Brush.Color := clGray; end; if DataModule2.Rechnung.FieldByName('Bezahlt').AsBoolean=False then Begin DBGrid1.Canvas.Brush.Color := clMoneyGreen; end; DBGrid1.DefaultDrawColumnCell(Rect, DataCol, Column, State); end; Das geht nicht nur einmal am Anfang auch beim durchklicken. |
AW: Zeilen eines TDBGrid "Blockweise" färben
Hast Du für das Programm zufällig die JVCL zur Verfügung?
Dann könnte sich ein Blick auf TJvDBGrid lohnen. Der Austausch ist einfach: Altes Grid raus, neues Grid rein, solange der Name gleich bleibt, sollte das erstmal ohne Nebenwirkungen gehen. Die Methoden für die farblische Gestaltung erscheinen mir aber deutlich leistungsfähiger. Ein Grundproblem wird aber bleiben: Beim Scrollen durch die Datenmenge, der Auswahl einer Zeile bzw. dem Aufheben der Auswahl ... muss man immer auf den Satz davor schauen, um auf die richtige Farbgestaltung zu kommen. Das wird vermutlich zu einem wilden "Gehopse" mit Dataset.Next und Dataset.Prev führen. Hab's gestern mal versucht, mir ist da keine sinnvolle und immer funktionierende Lösung eingefallen. Beim Öffnen der Datenmenge und dem vorwärtsscrollen hat's ja noch geklappt, aber rückwärts oder bei der Auswahl eines Datensatzes und folgendem Wechsel vorwärts oder rückwärts bin ich gescheitert. Was man eventuell noch versuchen könnte: Neben dem DBGrid noch ein unsichtbares Stringgrid "mitführen". Beim Lesen des Daten wird in das StringGrid je Datensatz dessen RecNo und seine Farbe als Zeile eingefügt. Scrollt man nun durch die Datenmenge, so schaut man dort nach, ob die RecNo schon vorhanden ist. Wenn ja, wird dem StringGrid der Wert für die Farbe entnommen, andernfalls muss die Farbe ermittelt werden und dann im StringGrid eine entsprechende Zeile eingefügt werden. Oder immer dann, wenn eine neue VorgangsID auftaucht wird die im StringGrid mit der entsprechenden Farbe vermerkt. Bei einem Datensatzwechsel wird dort dann nachgeschaut, welche Farbe zu vergeben ist. Das könnte im Ereignis OnDrawColumnCell realisierbar sein. Bei kleinen Datenmengen mag das insgesamt so angehen, für große Datenmengen dürfte es aber eher ungeeignet sein. Oder im DataSet eine zusätzliche Spalte einfügen für den Farbwert. Die Spalte darf halt nur nicht angezeigt werden. Dann könnte man beim Scrollen die Farbe dort entnehmen. Nur bei der erstmaligen Anzeige eines Datensatzes muss halt die Farbe erst aus dem vorherigen Datensatz ermittelt werden und dann im DataSet eingefügt werden. Bei einer ReadOnly-Datenmenge dürfte das aber auch schon wieder scheitern. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 13:57 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