AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Programmierung allgemein GUI-Design mit VCL / FireMonkey / Common Controls Delphi [StringGrid] berechnen d. Zellenbreite nach resize
Thema durchsuchen
Ansicht
Themen-Optionen

[StringGrid] berechnen d. Zellenbreite nach resize

Ein Thema von Buddelflink · begonnen am 6. Feb 2007 · letzter Beitrag vom 23. Aug 2012
Antwort Antwort
Seite 1 von 2  1 2      
Buddelflink

Registriert seit: 19. Mai 2003
7 Beiträge
 
Delphi 2006 Professional
 
#1

[StringGrid] berechnen d. Zellenbreite nach resize

  Alt 6. Feb 2007, 20:52
Hallo.
Bin sonst nur als aktiver Leser hier im Forum und konnte bisher alle meine Fragen mit der Suchfunktion lösen! Diesmal stellt sich mir aber ein Problem das scheinbar noch niemanden interessiert hat . Auch Google hat nix zu dem Thema.
Also, ich habe ein StringGrid und möchte bei resize die Spaltenbreite/höhe entsprechend berechnen.
Delphi-Quellcode:
procedure TForm_Jahrestest.FormCanResize(Sender: TObject; var NewWidth,
  NewHeight: Integer; var Resize: Boolean);
var iZelle, iHeader: Integer;
begin
  Resize:= true;
  JahresGrid.DoubleBuffered:= true;

  // Höhe d. StringGridZellen
  iHeader:= 15; // mindesthöhe
  iZelle:= (JahresGrid.Height -iHeader) div 12;
  iHeader:= (JahresGrid.Height - (iZelle * 12));
  JahresGrid.DefaultRowHeight:= iZelle; // iZelle -3 dann gehts
  JahresGrid.RowHeights[0]:= iHeader;

  // Breite d. Zellen
  iHeader:= 55; // mindestbreite
  iZelle:= ((JahresGrid.Width -iHeader) div 37);
  iHeader:= (JahresGrid.Width -(iZelle * 37));
  JahresGrid.DefaultColWidth:= iZelle;
  JahresGrid.ColWidths[0]:= iHeader;
end;
Leider wird ein Teil der letzten Zelle(n) verdeckt.
Was zum Geier mache ich falsch? -> siehe Bild

Ich hoffe jemand weiss wie man das logisch berechnet?
PS: Hab jetzt nur den relevanten Code rauskopiert aber bei Bedarf kann ich das ganze Projekt auch posten.

Danke, Cornel
Angehängte Grafiken
Dateityp: png stringgrid_159.png (27,5 KB, 68x aufgerufen)
  Mit Zitat antworten Zitat
Benutzerbild von Lannes
Lannes

Registriert seit: 30. Jan 2005
Ort: Münster
745 Beiträge
 
Delphi 3 Professional
 
#2

Re: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 6. Feb 2007, 22:50
Hallo,

Rahmen-, Srollbar- und die Gitterlinienbreite in die Berechnung einbeziehen, das Grid hat 5 Spalten.
ClientWidth ist gleich Width - Rand - Scrollbar
Delphi-Quellcode:
var iFixCol : Integer;
begin
  iFixCol := 50;
  StringGrid1.DefaultColWidth := ((StringGrid1.ClientWidth-iFixCol) div 4) - StringGrid1.GridLineWidth;
  StringGrid1.ColWidths[0] := iFixCol+(StringGrid1.ClientWidth-iFixCol) mod 4;
//..
showmessage('Achtung! Differenz: '+IntToStr(StringGrid1.Width-StringGrid1.ClientWidth));
MfG Lannes
(Nichts ist nicht Nichts) and ('' <> nil ) and (Pointer('') = nil ) and (@('') <> nil )
  Mit Zitat antworten Zitat
Buddelflink

Registriert seit: 19. Mai 2003
7 Beiträge
 
Delphi 2006 Professional
 
#3

Re: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 7. Feb 2007, 10:48
Hallo Lannes.
funktioniert Super. Danke
Mit StringGrid.ClientWidth hatte ich es auch schon versucht, allerdings hatte mich die Hilfe ein wenig verwirrt
Zitat:
OH
Mit ClientWidth können Sie die Breite des Client-Bereichs des Steuerelements abrufen oder ändern. ClientWidth ist mit ClientRect.Right identisch.

Bei TControl entspricht ClientWidth der Eigenschaft Width. In abgeleiteten Klassen kann eine ClientWidth-Eigenschaft implementiert werden, die sich von Width unterscheidet. So enthält beispielsweise die Eigenschaft ClientWidth eines Formulars den Wert der Eigenschaft Width minus der Breite des Rahmens und der Bildlaufleisten.
Bei Deinem Bsp. wurde die letzte Gitterlinie nicht angezeigt , sprich es fehlte ein Pixel in der Berechnung. -> siehe Bild
Habs jetzt so gemacht
Delphi-Quellcode:
with JahresGrid do begin
// Höhe d. StringGridZellen
  iFixCol:= 12;
  DefaultRowHeight:= ((ClientHeight -iFixCol) div 12) -GridLineWidth;
  Jahresgrid.RowHeights[0]:= (iFixCol -1 {warum?}) +(ClientHeight -iFixCol) mod 12;

// Breite d. Zellen
  iFixCol:= 55;
  DefaultColWidth:= ((ClientWidth -iFixCol) div 37) -GridLineWidth;
  ColWidths[0]:= (iFixCol -1) +(ClientWidth -iFixCol) mod 37;
end;
Angehängte Grafiken
Dateityp: png stringgrid_994.png (24,5 KB, 57x aufgerufen)
Dank & Gruss
  Mit Zitat antworten Zitat
phedon

Registriert seit: 13. Aug 2008
5 Beiträge
 
#4

AW: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 3. Aug 2012, 09:28
Bisl alt das Thema, aber:
danke Buddelfink für die Frage und Lannes für die Antwort.
Erspart ein neues Thema
  Mit Zitat antworten Zitat
omata

Registriert seit: 26. Aug 2004
Ort: Nebel auf Amrum
3.154 Beiträge
 
Delphi 7 Enterprise
 
#5

AW: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 3. Aug 2012, 22:13
Wenn man einfach nur mit DIV arbeitet, passen die Spalten natürlich in das Grid, ohne das der Scrollbalken erscheint. Je nach Gridgröße entsteht dann aber am rechten Rand eine mehr oder weniger große weiße Fläche, die nicht zu einer Zelle gehört.

Wenn man immer die gesamte Gridfläche verwenden möchte, ohne das der Scrollbaken erscheint, verwendet man z.B. folgenden Code...
Delphi-Quellcode:
procedure StringGridResize(Grid: TStringGrid);
var i, w, cols, line:integer;
begin
  cols:=Grid.ColCount;
  line:=Grid.GridLineWidth;
  w:=0;
  for i:=1 to cols do begin
    Grid.ColWidths[i-1]:=
      (Grid.Canvas.ClipRect.Right - w - (line * (cols - i))) div (cols - i + 1);
    w:=w + Grid.ColWidths[i-1] + line;
  end;
end;

Geändert von omata ( 3. Aug 2012 um 22:16 Uhr)
  Mit Zitat antworten Zitat
phedon

Registriert seit: 13. Aug 2008
5 Beiträge
 
#6

AW: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 4. Aug 2012, 14:06
omata - noch besser - danke dir!
  Mit Zitat antworten Zitat
phedon

Registriert seit: 13. Aug 2008
5 Beiträge
 
#7

AW: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 4. Aug 2012, 19:21
@Omata: zu früh gefreut, löscht leider den ganzen Inhalt..
  Mit Zitat antworten Zitat
spartakus

Registriert seit: 20. Feb 2005
22 Beiträge
 
#8

AW: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 17. Aug 2012, 23:06
Hallo Buddelflink,
wie hast du das gemacht dass einen Kalender in einem Stringgrid angezeigt wird?
Kann du mal bitte ein Beispiel zeigen!
danke.
  Mit Zitat antworten Zitat
Perlsau
(Gast)

n/a Beiträge
 
#9

AW: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 18. Aug 2012, 11:05
Hallo Buddelflink,
wie hast du das gemacht dass einen Kalender in einem Stringgrid angezeigt wird?
Kann du mal bitte ein Beispiel zeigen!
danke.
Einen Kalender in einem Stringgrid anzuzeigen ist nicht ganz einfach und daher auch nicht mit einem einfachen Beispiel zu illustrieren.

In der Unit DateUtils findest du zahlreiche nützliche Funktionen wie z.B. DayOfTheWeek, DaysInMonth und WeekOfTheYear.

Es gibt mehrere Wege, die nach Rom (zum Ziel) führen, daher möchte ich deiner Lernfähigkeit nicht im Wege stehen, indem ich hier den vollständigen Code reinstelle. Mit anderen Worten: Selber ausprobieren heißt die Devise. Für die Kalenderdarstellung einer Terminverwaltung hab ich einige Tage aus- und rumprobiert, auch mal die Hilfe eines Delphi-Forums in Anspruch genommen bei ganz gezielten Fragen. Dagegen wird dir die pauschale Frage, wie man einen Kalender programmiert, wohl kaum einer beantworten wollen.
Angehängte Grafiken
Dateityp: jpg Kalender.jpg (101,1 KB, 26x aufgerufen)

Geändert von Perlsau (18. Aug 2012 um 11:11 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Sir Rufo
Sir Rufo

Registriert seit: 5. Jan 2005
Ort: Stadthagen
9.454 Beiträge
 
Delphi 10 Seattle Enterprise
 
#10

AW: [StringGrid] berechnen d. Zellenbreite nach resize

  Alt 18. Aug 2012, 12:33
Hallo Buddelflink,
wie hast du das gemacht dass einen Kalender in einem Stringgrid angezeigt wird?
Kann du mal bitte ein Beispiel zeigen!
danke.
Nur mal so für Spaß, wie man in ein StringGrid - mit wiederverwendbarem Code - einen Kalender reinschmeißt
Form:
Delphi-Quellcode:
unit MainView;

interface

uses
  Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
  Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.Grids, uStringGridCalendar,
  Vcl.StdCtrls, Vcl.ExtCtrls;

type
  TForm1 = class( TForm )
    StringGrid1 : TStringGrid;
    Button1 : TButton;
    Button2 : TButton;
    Panel1: TPanel;
    procedure FormCreate( Sender : TObject );
    procedure FormDestroy( Sender : TObject );
    procedure Button1Click( Sender : TObject );
    procedure Button2Click( Sender : TObject );
  private
    { Private-Deklarationen }
    FCal : TStringGridCalendar;
  public
    { Public-Deklarationen }
  end;

var
  Form1 : TForm1;

implementation

{$R *.dfm}

procedure TForm1.Button1Click( Sender : TObject );
begin
  FCal.DoIncYear( 1 );
end;

procedure TForm1.Button2Click( Sender : TObject );
begin
  FCal.DoIncYear( -1 );
end;

procedure TForm1.FormCreate( Sender : TObject );
begin
  FCal := TStringGridCalendar.Create( StringGrid1 );
end;

procedure TForm1.FormDestroy( Sender : TObject );
begin
  FCal.Free;
end;

end.
und die Klasse, die das gesamte Handling übernimmt:
Delphi-Quellcode:
unit uStringGridCalendar;

interface

uses
  Vcl.Grids;

type
  TStringGridCalendar = class
  private
    FGrid : TStringGrid;
    FSelDate : TDate;
    procedure SetSelDate( const Value : TDate );
  protected
    procedure GridSelectCell( Sender : TObject; ACol, ARow : Integer; var CanSelect : Boolean );
    procedure DoGridDataFill;
    function TryGetDateFromGrid( ACol, ARow : Integer; out aDate : TDate ) : Boolean;
    function GetDateFromGrid( ACol, ARow : Integer ) : TDate;
  public
    constructor Create( aStringGrid : TStringGrid );
    property SelDate : TDate read FSelDate write SetSelDate;
    procedure DoIncYear( aValue : Integer );
  end;

implementation

uses
  System.SysUtils, System.DateUtils;

{ TStringGridCalendar }

constructor TStringGridCalendar.Create( aStringGrid : TStringGrid );
begin
  inherited Create;
  FGrid := aStringGrid;
  FGrid.Options := FGrid.Options - [goRangeSelect];
  FGrid.OnSelectCell := GridSelectCell;
  FSelDate := Date;

  DoGridDataFill;
end;

procedure TStringGridCalendar.DoGridDataFill;
var
  idx : Integer;
  DateIdx : TDate;
  lRow, lCol : Integer;
begin
  FGrid.FixedCols := 1;
  FGrid.FixedRows := 1;

  FGrid.ColCount := 38;
  FGrid.RowCount := MonthsPerYear + FGrid.FixedRows;

  // Jahreszahl in die erste Zelle
  FGrid.Cells[0, 0] := IntToStr( YearOf( SelDate ) );

  // Tagesnamen in die erste Zeile
  for idx := FGrid.FixedCols to Pred( FGrid.ColCount ) do
    begin
      FGrid.Cells[idx, 0] := ShortDayNames[( idx - FGrid.FixedCols + 1 ) mod DaysPerWeek + 1];
    end;

  // Monatsnamen in die erste Spalte
  for idx := FGrid.FixedRows to Pred( FGrid.RowCount ) do
    begin
      FGrid.Cells[0, idx] := LongMonthNames[idx - FGrid.FixedRows + 1];
    end;

  for lRow := FGrid.FixedRows to Pred( FGrid.RowCount ) do
    begin
      for lCol := FGrid.FixedCols to Pred( FGrid.ColCount ) do
        begin
          if TryGetDateFromGrid( lCol, lRow, DateIdx )
          then
            FGrid.Cells[lCol, lRow] := IntToStr( DayOf( DateIdx ) )
          else
            FGrid.Cells[lCol, lRow] := '';
        end;
    end;

end;

function TStringGridCalendar.GetDateFromGrid( ACol, ARow : Integer ) : TDate;
begin
  if not TryGetDateFromGrid( ACol, ARow, Result )
  then
    raise Exception.Create( 'Fehlermeldung' );
end;

procedure TStringGridCalendar.GridSelectCell( Sender : TObject; ACol, ARow : Integer; var CanSelect : Boolean );
var
  lDate : TDate;
begin
  if TryGetDateFromGrid( ACol, ARow, lDate )
  then
    begin
      CanSelect := True;
      SelDate := lDate;
    end
  else
    CanSelect := False;
end;

procedure TStringGridCalendar.DoIncYear( aValue : Integer );
begin
  SelDate := IncYear( SelDate, aValue )
end;

function TStringGridCalendar.TryGetDateFromGrid( ACol, ARow : Integer; out aDate : TDate ) : Boolean;
var
  lMonth, lDay, lYear : Word;
  lMinCol, lMaxCol : Integer;
begin
  Result := False;
  if ( ACol >= FGrid.FixedCols ) and ( ARow >= FGrid.FixedRows )
  then
    begin
      lYear := YearOf( SelDate );
      lMonth := ARow - FGrid.FixedRows + 1;
      aDate := EncodeDate( lYear, lMonth, 1 );
      lMinCol := DayOfTheWeek( aDate ) + FGrid.FixedCols - 1;
      lMaxCol := lMinCol + DaysInMonth( aDate ) - 1;
      if ( ACol >= lMinCol ) and ( ACol <= lMaxCol )
      then
        begin
          Result := True;
          lDay := ACol - lMinCol + 1;
          aDate := EncodeDate( lYear, lMonth, lDay );
        end;
    end;
end;

procedure TStringGridCalendar.SetSelDate( const Value : TDate );
var
  lYearChange : Boolean;
begin
  if ( Value <> SelDate )
  then
    begin
      lYearChange := YearOf( Value ) <> YearOf( SelDate );
      FSelDate := Value;
      if lYearChange
      then
        DoGridDataFill;
    end;
end;

end.
Diese Klasse kann man nun noch um beliebige Features erweitern um die Anzeige aufzuhübschen oder weitere Funktionalitäten anzupassen.
Kaum macht man's richtig - schon funktioniert's
Zertifikat: Sir Rufo (Fingerprint: ‎ea 0a 4c 14 0d b6 3a a4 c1 c5 b9 dc 90 9d f0 e9 de 13 da 60)
  Mit Zitat antworten Zitat
Antwort Antwort
Seite 1 von 2  1 2      


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 05: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