-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathuyeargrid.pas
232 lines (200 loc) · 7.16 KB
/
uyeargrid.pas
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
(******************************************************************************)
(* uYearGrid ??.??.???? *)
(* *)
(* Version : 0.01 *)
(* *)
(* Author : Uwe Schächterle (Corpsman) *)
(* *)
(* Support : www.Corpsman.de *)
(* *)
(* Description : LCL-Component to visualise a year overview *)
(* *)
(* License : See the file license.md, located under: *)
(* https://github.com/PascalCorpsman/Software_Licenses/blob/main/license.md *)
(* for details about the license. *)
(* *)
(* It is not allowed to change or remove this text from any *)
(* source file of the project. *)
(* *)
(* Warranty : There is no warranty, neither in correctness of the *)
(* implementation, nor anything other that could happen *)
(* or go wrong, use at your own risk. *)
(* *)
(* Known Issues: none *)
(* *)
(* History : 0.01 - Initial version *)
(* *)
(******************************************************************************)
Unit uYearGrid;
{$MODE ObjFPC}{$H+}
Interface
Uses
Classes, SysUtils, Grids, Graphics;
Type
TOnDayClickEvent = Procedure(Sender: TObject; Year, Month, Day: integer) Of Object;
{ TYearGrid }
TYearGrid = Class(TStringGrid)
private
fYear: integer;
fUserPrepareCanvas: TOnPrepareCanvasEvent;
Function getPrepareCanvas: TOnPrepareCanvasEvent;
Procedure setPrepareCanvas(AValue: TOnPrepareCanvasEvent);
Procedure OnPrepareCanvasEvent(sender: TObject; aCol, aRow: Integer; aState: TGridDrawState);
Function DayExistsInYear(Month, Day: Integer): Boolean;
protected
Procedure Click; override;
Procedure DoOnResize; override;
public
// TODO: All die hier müssen noch mit settern ausgestatet werden und dann ein Invalidate in die Setter
WeekendDays: Set Of byte; // 1 = Sunday, 2 = Monday, .. 7 = Saturday
NoValidDateColor: TColor;
WeekendColor: TColor; // Disable by setting "WeekendDays" to []
WorkColor: TColor;
TodayColor: TColor; // if = clnone = not displayed.
OnDayClick: TOnDayClickEvent; // TODO: Den Onclick auch so schützen wie den Prepare Canvas
Property OnPrepareCanvas: TOnPrepareCanvasEvent read getPrepareCanvas write setPrepareCanvas;
Property Year: integer read fYear;
Constructor Create(AOwner: TComponent); override;
Function IsWeekendDay(Value: TDateTime): Boolean;
Procedure LoadYear(aYear: Integer);
End;
Implementation
Uses StdCtrls, forms;
{ TYearGrid }
Constructor TYearGrid.Create(AOwner: TComponent);
Var
d, m, y: word;
Begin
Inherited Create(AOwner);
Options := Options - [goRangeSelect];
fUserPrepareCanvas := Nil;
DecodeDate(now, y, m, d);
LoadYear(y);
WeekendDays := [1, 7];
NoValidDateColor := clGray;
WeekendColor := TColor($F0F0F0);
WorkColor := clWhite;
TodayColor := clBlue;
OnDayClick := Nil;
Inherited OnPrepareCanvas := @OnPrepareCanvasEvent;
ScrollBars := ssNone;
End;
Function TYearGrid.IsWeekendDay(Value: TDateTime): Boolean;
Begin
result := DayOfWeek(Value) In WeekendDays;
End;
Function TYearGrid.getPrepareCanvas: TOnPrepareCanvasEvent;
Begin
result := fUserPrepareCanvas;
End;
Procedure TYearGrid.setPrepareCanvas(AValue: TOnPrepareCanvasEvent);
Begin
fUserPrepareCanvas := AValue;
End;
Procedure TYearGrid.OnPrepareCanvasEvent(sender: TObject; aCol, aRow: Integer;
aState: TGridDrawState);
Var
m, d: Integer;
ad, am, ay: word;
Begin
If (aCol <= 0) Or (aRow <= 0) Then exit; // Die Fixed Sachen kommen erst gar net rein
m := aRow;
d := aCol;
If DayExistsInYear(m, d) Then Begin
// Dem User "Vor" auswählen was es so gibt
// Markieren der Wochenenden
If IsWeekendDay(EncodeDate(fYear, m, d)) Then Begin
Canvas.Brush.Color := WeekendColor;
End
Else Begin
Canvas.Brush.Color := WorkColor;
End;
// Heute
If TodayColor <> clNone Then Begin
DecodeDate(now, ay, am, ad);
If (fYear = ay) And (m = am) And (d = ad) Then Begin
Canvas.Brush.Color := TodayColor;
End;
End;
// Let the user overrule all decisions
If assigned(fUserPrepareCanvas) Then Begin
fUserPrepareCanvas(sender, aCol, aRow, aState);
End;
End
Else Begin
Canvas.Brush.Color := NoValidDateColor;
Canvas.Pen.Color := NoValidDateColor; // TODO: warum geht das nicht ??
End;
End;
Function TYearGrid.DayExistsInYear(Month, Day: Integer): Boolean;
Var
Leap: Boolean;
Begin
result := false;
If (Not (Month In [1..12])) Then exit;
If Day < 1 Then exit;
leap := IsLeapYear(fYear);
result := MonthDays[leap][Month] >= day;
End;
Procedure TYearGrid.Click;
Var
m, d: LongInt;
Begin
Inherited Click;
If assigned(OnDayClick) Then Begin
m := Selection.top;
d := Selection.Left;
If DayExistsInYear(m, d) Then Begin
OnDayClick(self, fYear, m, d);
End;
End;
End;
Procedure TYearGrid.DoOnResize;
Var
i, w, h, aborder: Integer;
Begin
Inherited DoOnResize;
If GetParentDesignControl(self) = Nil Then Begin
aborder := 5;
End
Else Begin
aborder := Scale96ToForm(5);
End;
h := ClientHeight - aborder;
For i := 0 To RowCount - 1 Do Begin
RowHeights[i] := h Div 13;
End;
w := ClientWidth - ColWidths[0] - aborder;
For i := 1 To ColCount - 1 Do Begin
ColWidths[i] := w Div 31;
End;
End;
Procedure TYearGrid.LoadYear(aYear: Integer);
Var
i: Integer;
Begin
fYear := aYear;
RowCount := 13;
ColCount := 32;
FixedCols := 1;
FixedRows := 1;
// TODO: make this configurable
Cells[0, 1] := 'Jan';
Cells[0, 2] := 'Feb';
Cells[0, 3] := 'Mar';
Cells[0, 4] := 'Apr';
Cells[0, 5] := 'May';
Cells[0, 6] := 'Jun';
Cells[0, 7] := 'Jul';
Cells[0, 8] := 'Aug';
Cells[0, 9] := 'Sep';
Cells[0, 10] := 'Oct';
Cells[0, 11] := 'Nov';
Cells[0, 12] := 'Dec';
For i := 1 To 31 Do Begin
Cells[i, 0] := inttostr(i);
End;
DoOnResize;
Invalidate;
End;
End.