lazarus-ccr/components/orpheus/ovccal.pas
2018-01-09 19:39:55 +00:00

1803 lines
48 KiB
ObjectPascal

{*********************************************************}
{* OVCCAL.PAS 4.06 *}
{*********************************************************}
{* ***** BEGIN LICENSE BLOCK ***** *}
{* Version: MPL 1.1 *}
{* *}
{* The contents of this file are subject to the Mozilla Public License *}
{* Version 1.1 (the "License"); you may not use this file except in *}
{* compliance with the License. You may obtain a copy of the License at *}
{* http://www.mozilla.org/MPL/ *}
{* *}
{* Software distributed under the License is distributed on an "AS IS" basis, *}
{* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License *}
{* for the specific language governing rights and limitations under the *}
{* License. *}
{* *}
{* The Original Code is TurboPower Orpheus *}
{* *}
{* The Initial Developer of the Original Code is TurboPower Software *}
{* *}
{* Portions created by TurboPower Software Inc. are Copyright (C)1995-2002 *}
{* TurboPower Software Inc. All Rights Reserved. *}
{* *}
{* Contributor(s): *}
{* *}
{* ***** END LICENSE BLOCK ***** *}
{$I OVC.INC}
{$B-} {Complete Boolean Evaluation}
{$I+} {Input/Output-Checking}
{$P+} {Open Parameters}
{$T-} {Typed @ Operator}
{.W-} {Windows Stack Frame}
{$X+} {Extended Syntax}
unit ovccal;
{-Calendar component}
interface
uses
{$IFNDEF LCL} Windows, Messages, {$ELSE} LclIntf, LMessages, LclType, MyMisc, FileUtil,{$ENDIF}
Buttons, Classes, Controls, Forms, Graphics, Menus,
SysUtils, OvcBase, OvcConst, OvcData, OvcIntl,
OvcMisc, OvcDate;
type
TOvcDateFormat = (dfShort, dfLong);
TOvcDayNameWidth = 1..3;
TOvcDayType = (dtSunday, dtMonday, dtTuesday, dtWednesday,
dtThursday, dtFriday, dtSaturday);
TOvcCalDisplayOption = (cdoShortNames, cdoShowYear, cdoShowInactive,
cdoShowRevert, cdoShowToday, cdoShowNavBtns,
cdoHideActive);
TOvcCalDisplayOptions = set of TOvcCalDisplayOption;
type
TOvcCalColorArray = array[0..5] of TColor;
TOvcCalColorScheme = (cscalCustom, cscalClassic, cscalWindows,
cscalGold, cscalOcean, cscalRose);
TOvcCalSchemeArray = array[TOvcCalColorScheme] of TOvcCalColorArray;
const
{ActiveDay, DayNames, Days, InactiveDays, MonthAndYear, Weekend}
CalScheme : TOvcCalSchemeArray =
((0, 0, 0, 0, 0, 0),
(clHighlight, clWindow, clWindow, clWindow, clWindow, clWindow),
(clRed, clMaroon, clBlack, clGray, clBlue, clRed),
(clBlack, clBlack, clYellow, clGray, clBlack, clTeal),
(clBlack, clBlack, clAqua, clGray, clBlack, clNavy),
(clRed, clRed, clFuchsia, clGray, clBlue, clTeal)
);
type
TOvcCalColors = class(TPersistent)
{.Z+}
protected {private}
{property variables}
FUpdating : Boolean;
FOnChange : TNotifyEvent;
{internal variables}
SettingScheme : Boolean;
{property methods}
function GetColor(Index : Integer) : TColor;
procedure SetColor(Index : Integer; Value : TColor);
procedure SetColorScheme(Value : TOvcCalColorScheme);
{internal methods}
procedure DoOnChange;
public
{public property variables}
FCalColors : TOvcCalColorArray;
FColorScheme : TOvcCalColorScheme;
procedure Assign(Source : TPersistent);
override;
procedure BeginUpdate;
procedure EndUpdate;
property OnChange : TNotifyEvent
read FOnChange write FOnChange;
{.Z-}
published
property ActiveDay : TColor index 0
read GetColor write SetColor;
property ColorScheme : TOvcCalColorScheme
read FColorScheme write SetColorScheme;
property DayNames : TColor index 1
read GetColor write SetColor;
property Days : TColor index 2
read GetColor write SetColor;
property InactiveDays : TColor index 3
read GetColor write SetColor;
property MonthAndYear : TColor index 4
read GetColor write SetColor;
property Weekend : TColor index 5
read GetColor write SetColor;
end;
type
TDateChangeEvent = procedure(Sender : TObject; Date : TDateTime)
of object;
TCalendarDateEvent =
procedure(Sender : TObject; ADate : TDateTime; const Rect : TRect)
of object;
TGetHighlightEvent =
procedure(Sender : TObject; ADate : TDateTime; var Color : TColor)
of object;
TGetDateEnabledEvent =
procedure(Sender : TObject; ADate : TDateTime; var Enabled : Boolean)
of object;
TOvcCustomCalendar = class(TOvcCustomControl)
{.Z+}
protected {private}
{property variables}
FBorderStyle : TBorderStyle;
FBrowsing : Boolean;
FColors : TOvcCalColors;
FOptions : TOvcCalDisplayOptions;
FDate : TDateTime;
FDay : Integer; {calendar day}
FDateFormat : TOvcDateFormat;
FDayNameWidth : TOvcDayNameWidth;
FDrawHeader : Boolean; {true to draw day name header}
FIntlSup : TOvcIntlSup; {international date/time support}
FMonth : Integer; {calendar month}
FReadOnly : Boolean; {true if in read only mode}
FWantDblClicks : Boolean; {true to include cs_dblclks style}
FWeekStarts : TOvcDayType; {the day that begins the week}
FYear : Integer; {calendar year}
{$IFDEF LCL}
FCtl3D : Boolean;
{$ENDIF}
{event variables}
FOnChange : TDateChangeEvent;
FOnDrawDate : TCalendarDateEvent;
FOnDrawItem : TCalendarDateEvent;
FOnGetDateEnabled: TGetDateEnabledEvent;
FOnGetHighlight : TGetHighlightEvent;
{internal variables}
clBtnLeft : TSpeedButton;
clBtnRevert : TSpeedButton;
clBtnRight : TSpeedButton;
clBtnToday : TSpeedButton;
clInPopup : Boolean;
clBtnNextYear : TSpeedButton;
clBtnPrevYear : TSpeedButton;
clCalendar : array[1..49] of Byte; {current month grid}
clDay : Word;
clFirst : Byte; {index for first day in current month}
clLast : Byte; {index for last day in current month}
clMonth : Word;
clRowCol : array[0..8, 0..6] of TRect; {cell TRect info}
cSettingScheme : Boolean;
clYear : Word;
clWidth : Integer; {client width - margins}
clMask : array[0..MaxDateLen] of AnsiChar; {default date mask}
clPopup : Boolean; {true if being created as a popup}
clRevertDate : TDateTime; {date on entry}
clRowCount : Integer; {7 if no header, otherwise 8}
clStartRow : Integer; {first row number}
{property methods}
function GetAsDateTime : TDateTime;
function GetAsStDate : TStDate;
function GetCalendarDate : TDateTime;
function GetDay : Integer;
function GetMonth : Integer;
function GetYear : Integer;
procedure SetAsDateTime(Value : TDateTime);
procedure SetAsStDate(Value : TStDate);
procedure SetBorderStyle(Value : TBorderStyle);
procedure SetDate(Value : TDateTime);
procedure SetDateFormat(Value : TOvcDateFormat);
procedure SetDayNameWidth(Value : TOvcDayNameWidth);
procedure SetDisplayOptions(Value : TOvcCalDisplayOptions);
procedure SetDrawHeader(Value : Boolean);
procedure SetIntlSupport(Value : TOvcIntlSup);
procedure SetWantDblClicks(Value : Boolean);
procedure SetWeekStarts(Value : TOvcDayType);
{internal methods}
procedure calChangeMonth(Sender : TObject);
procedure calColorChange(Sender : TObject);
function calGetCurrentRectangle : TRect;
{-get bounding rectangle for the current calendar day}
function calGetValidDate(ADate : TDateTime; Delta : Integer) : TDateTime;
procedure calRebuildCalArray;
{-recalculate the contents of the calendar array}
procedure calRecalcSize;
{-calcualte new sizes for rows and columns}
{VCL control methods}
{$IFNDEF LCL}
procedure CMCtl3DChanged(var Msg : TMessage);
message CM_CTL3DCHANGED;
{$ENDIF}
procedure CMEnter(var Msg : TMessage);
message {$IFNDEF LCL} CM_ENTER; {$ELSE} LM_SETFOCUS; {$ENDIF}
//CM_ messages not supported in LCL, so use something similar
// so clRevertDate is initialized properly.
procedure CMExit(var Msg : TMessage);
message CM_EXIT;
procedure CMFontChanged(var Msg : TMessage);
message CM_FONTCHANGED;
{windows message methods}
procedure WMEraseBkgnd(var Msg : TWMEraseBkgnd);
message WM_ERASEBKGND;
procedure WMGetDlgCode(var Msg : TWMGetDlgCode);
message WM_GETDLGCODE;
procedure WMKillFocus(var Msg : TWMKillFocus);
message WM_KILLFOCUS;
protected
procedure calBtnClick(Sender : TObject);
procedure CreateParams(var Params : TCreateParams);
override;
procedure CreateWnd;
override;
procedure DoOnChange(Value : TDateTime);
dynamic;
function DoOnGetDateEnabled(ADate : TDateTime) : Boolean;
dynamic;
procedure DoOnMouseWheel(Shift : TShiftState; Delta, XPos, YPos : SmallInt);
override;
function IsReadOnly : Boolean;
dynamic;
{-return true if the calendar is in read-only mode}
procedure KeyDown(var Key : Word; Shift : TShiftState);
override;
procedure KeyPress(var Key : Char);
override;
procedure MouseDown(Button : TMouseButton; Shift : TShiftState; X, Y : Integer);
override;
procedure MouseUp(Button : TMouseButton; Shift : TShiftState; X, Y : Integer);
override;
procedure Paint;
override;
{virtual property methods}
procedure SetCalendarDate(Value : TDateTime);
virtual;
public
constructor Create(AOwner : TComponent);
override;
constructor CreateEx(AOwner : TComponent; AsPopup : Boolean);
virtual;
destructor Destroy;
override;
procedure SetBounds(ALeft, ATop, AWidth, AHeight : Integer);
override;
{.Z-}
function DateString(const Mask : string): string;
function DayString : string;
procedure IncDay(Delta : Integer);
procedure IncMonth(Delta : Integer);
procedure IncYear(Delta : Integer);
function MonthString : string;
procedure SetToday;
property AsDateTime : TDateTime
read GetAsDateTime write SetAsDateTime;
property AsStDate : TStDate
read GetAsStDate write SetAsStDate;
property Browsing : Boolean
read FBrowsing;
property Canvas;
property Day : Integer
read GetDay;
property Month : Integer
read GetMonth;
property Year : Integer
read GetYear;
{$IFDEF LCL}
property Ctl3D : Boolean read FCtl3D write FCtl3D default True;
{$ENDIF}
{properties}
property BorderStyle : TBorderStyle
read FBorderStyle write SetBorderStyle;
property CalendarDate : TDateTime
read GetCalendarDate write SetCalendarDate;
property Colors : TOvcCalColors
read FColors write FColors;
property Date : TDateTime
read FDate write SetDate;
property DateFormat : TOvcDateFormat
read FDateFormat write SetDateFormat;
property DayNameWidth : TOvcDayNameWidth
read FDayNameWidth write SetDayNameWidth;
property DrawHeader : Boolean
read FDrawHeader write SetDrawHeader;
property IntlSupport : TOvcIntlSup
read FIntlSup write SetIntlSupport;
property Options : TOvcCalDisplayOptions
read FOptions write SetDisplayOptions;
property ReadOnly : Boolean
read FReadOnly write FReadOnly;
property WantDblClicks : Boolean
read FWantDblClicks write SetWantDblClicks;
property WeekStarts : TOvcDayType
read FWeekStarts write SetWeekStarts;
{events}
property OnChange : TDateChangeEvent
read FOnChange write FOnChange;
property OnDrawDate : TCalendarDateEvent
read FOnDrawDate write FOnDrawDate;
property OnDrawItem : TCalendarDateEvent
read FOnDrawItem write FOnDrawItem;
property OnGetDateEnabled : TGetDateEnabledEvent
read FOnGetDateEnabled write FOnGetDateEnabled;
property OnGetHighlight : TGetHighlightEvent
read FOnGetHighlight write FOnGetHighlight;
end;
TOvcCalendar = class(TOvcCustomCalendar)
published
{properties}
{$IFDEF VERSION4}
property Anchors;
property Constraints;
property DragKind;
{$ENDIF}
property About;
property Align;
property BorderStyle;
property Colors;
property Color;
property Ctl3D;
property Cursor;
property DateFormat default dfLong;
property DayNameWidth;
property DragCursor;
property DragMode;
property Enabled;
property Font;
property LabelInfo;
property Options;
{$IFNDEF LCL}
property ParentCtl3D;
{$ENDIF}
property ParentFont;
property ParentShowHint;
property PopupMenu;
property ReadOnly default False;
property ShowHint;
property TabOrder;
property TabStop default True;
property Visible;
property WantDblClicks default True;
property WeekStarts default dtSunday;
{events}
property AfterEnter;
property AfterExit;
property OnChange;
property OnClick;
property OnDblClick;
property OnDragDrop;
property OnDragOver;
property OnDrawDate;
property OnDrawItem;
property OnEndDrag;
property OnEnter;
property OnExit;
property OnGetDateEnabled;
property OnGetHighlight;
property OnKeyDown;
property OnKeyPress;
property OnKeyUp;
property OnMouseDown;
property OnMouseMove;
property OnMouseUp;
property OnStartDrag;
end;
implementation
{$IFDEF LCL}
uses
LazUTF8;
{$ENDIF}
const
calMargin = 4; {left, right, and top margin}
{*** TOvcCalColors ***}
procedure TOvcCalColors.Assign(Source : TPersistent);
begin
if Source is TOvcCalColors then begin
FCalColors := TOvcCalColors(Source).FCalColors;
FColorScheme := TOvcCalColors(Source).FColorScheme;
FOnChange := TOvcCalColors(Source).FOnChange;
end else
inherited Assign(Source);
end;
procedure TOvcCalColors.BeginUpdate;
begin
FUpdating := True;
end;
procedure TOvcCalColors.EndUpdate;
begin
FUpdating := False;
DoOnChange;
end;
procedure TOvcCalColors.DoOnChange;
begin
if not FUpdating and Assigned(FOnChange) then
FOnChange(Self);
if not SettingScheme then
FColorScheme := cscalCustom;
end;
function TOvcCalColors.GetColor(Index : Integer) : TColor;
begin
Result := FCalColors[Index];
end;
procedure TOvcCalColors.SetColor(Index : Integer; Value : TColor);
begin
if Value <> FCalColors[Index] then begin
FCalColors[Index] := Value;
DoOnChange;
end;
end;
procedure TOvcCalColors.SetColorScheme(Value : TOvcCalColorScheme);
begin
if Value <> FColorScheme then begin
SettingScheme := True;
try
FColorScheme := Value;
if Value <> cscalCustom then begin
FCalColors := CalScheme[Value];
DoOnChange;
end;
finally
SettingScheme := False;
end;
end;
end;
{*** TOvcCustomCalendar ***}
procedure TOvcCustomCalendar.calBtnClick(Sender : TObject);
var
Key : Word;
begin
SetFocus;
Key := 0;
if Sender = clBtnLeft then begin
Key := VK_PRIOR;
KeyDown(Key, []);
end else if Sender = clBtnRevert then begin
Key := VK_ESCAPE;
KeyDown(Key, []);
end else if Sender = clBtnRight then begin
Key := VK_NEXT;
KeyDown(Key, []);
end else if Sender = clBtnToday then begin
Key := VK_BACK;
KeyDown(Key, [ssAlt]);
end else if Sender = clBtnNextYear then begin
Key := VK_NEXT;
KeyDown(Key, [ssCtrl]);
end else if Sender = clBtnPrevYear then begin
Key := VK_PRIOR;
KeyDown(Key, [ssCtrl]);
end;
end;
procedure TOvcCustomCalendar.calChangeMonth(Sender : TObject);
var
Y, M, D : Word;
MO : Integer;
MI : TMenuItem;
begin
MI := (Sender as TMenuItem);
DecodeDate(FDate, Y, M, D);
MO := MI.Tag;
{set month and year}
if (MO > M) and (MI.HelpContext < 3) then
Dec(Y)
else if (MO < M) and (MI.HelpContext > 3) then
Inc(Y);
M := M + MO;
{set day}
if D > DaysInMonth(MO, Y, 0) then
D := DaysInMonth(MO, Y, 0);
SetDate(calGetValidDate(EncodeDate(Y, MO, D)-1, +1));
if (Assigned(FOnChange)) then
FOnChange(Self, FDate);
end;
procedure TOvcCustomCalendar.calColorChange(Sender : TObject);
begin
Invalidate;
end;
function TOvcCustomCalendar.calGetCurrentRectangle : TRect;
{-get bounding rectangle for the current date}
var
Idx : Integer;
R, C : Integer;
begin
{index into the month grid}
Idx := clFirst + Pred(clDay) + 13;
R := (Idx div 7);
C := (Idx mod 7);
Result := clRowCol[R,C];
end;
{added}
{Modified July 9, 2001}
function TOvcCustomCalendar.calGetValidDate(ADate : TDateTime;
Delta : Integer) : TDateTime;
var
I, X : Integer;
Valid: Boolean;
Fwd: Boolean;
begin
Valid := false;
Fwd := false;
X := Delta;
I := 1;
while not Valid and (I < 1000) do begin
{If the date is valid then yay!}
if (DoOnGetDateEnabled(ADate + (X * I))) then begin
Valid := true;
Fwd := True;
end
{otherwise check the other direction}
else if (DoOnGetDateEnabled(ADate - (X * I))) then begin
valid := true;
end
else Inc(I);
end;
if Valid then
if Fwd then Result := ADate + (X * I)
else Result := ADate - (X * I)
else
raise(Exception.Create(GetOrphStr(SCInvalidDate)));
end;
procedure TOvcCustomCalendar.calRebuildCalArray;
var
Day1 : TOvcDayType;
I, J : Integer;
begin
HandleNeeded;
DecodeDate(FDate, clYear, clMonth, clDay);
{get the first day of the current month and year}
Day1 := TOvcDayType(SysUtils.DayOfWeek(EncodeDate(clYear, clMonth, 1)) -1);
{find its index}
I := Byte(Day1) - Byte(WeekStarts) + 1;
if I < 1 then
Inc(I, 7);
clFirst := I;
{find the index of the last day in the month}
clLast := clFirst + DaysInMonth(clMonth, clYear, 0) - 1;
{initialize the first part of the calendar}
if clMonth = 1 then
J := DaysInMonth(12, clYear-1, 0)
else
J := DaysInMonth(clMonth-1, clYear, 0);
for I := clFirst-1 downto 1 do begin
clCalendar[I] := J;
Dec(J);
end;
{initialize the rest of the calendar}
J := 1;
for I := clFirst to 49 do begin
clCalendar[I] := J;
if I = clLast then
J := 1
else
Inc(J);
end;
end;
procedure TOvcCustomCalendar.calRecalcSize;
{-calcualte new sizes for rows and columns}
var
R : Integer;
C : Integer;
D1 : Integer;
D2 : Integer;
CH : Integer;
RH : Integer;
Row : array[0..8] of Integer;
Col : array[0..6] of Integer;
function SumOf(const A : array of Integer; First, Last : Integer) : Integer;
var
I : Integer;
begin
Result := 0;
for I := First to Last do
Result := Result + A[I];
end;
begin
if not HandleAllocated then
Exit;
{clear row/col position structure}
FillChar(clRowCol, SizeOf(clRowCol), #0);
{set the way the buttons should look}
clBtnLeft.Flat := not Ctl3D and not clPopup;
clBtnRevert.Flat := not Ctl3D and not clPopup;
clBtnRight.Flat := not Ctl3D and not clPopup;
clBtnToday.Flat := not Ctl3D and not clPopup;
clBtnNextYear.Flat := not Ctl3D and not clPopup;
clBtnPrevYear.Flat := not Ctl3D and not clPopup;
clBtnRevert.Visible := cdoShowRevert in FOptions;
clBtnToday.Visible := cdoShowToday in FOptions;
clBtnLeft.Visible := (cdoShowNavBtns in FOptions);
clBtnRight.Visible := (cdoShowNavBtns in FOptions);
clBtnNextYear.Visible := (cdoShowNavBtns in FOptions);
clBtnPrevYear.Visible := (cdoShowNavBtns in FOptions);
clWidth := ClientWidth - 2*calMargin;
{store row and column sizes}
for C := 0 to 6 do
Col[C] := clWidth div 7;
Canvas.Font := Font;
(*
Row[0] := Round(1.3 * Canvas.TextHeight('Yy')); {button and date row}
Row[1] := Round(1.5 * Canvas.TextHeight('Yy'));; {day name row}
*)
if (DrawHeader) then begin
{button and date row}
Row[0] := Round(1.4 * Canvas.TextHeight('Yy'));
{day name row}
Row[1] := Round(1.5 * Canvas.TextHeight('Yy'))
end else begin
{button and date row}
Row[0] := Round(1.3 * Canvas.TextHeight('Yy'));
{day name row}
Row[1] := 0;
end;
CH := ClientHeight - 2*calMargin - Row[0] - Row[1];
RH := CH div 7;
for R := 2 to 8 do
Row[R] := RH;
{distribute any odd horizontal space equally among the columns}
for C := 0 to clWidth mod 7 do
Inc(Col[C]);
{distribute odd vertical space to top 2 rows}
D1 := 0;
for R := 0 to 8 do
D1 := D1 + Row[R];
D1 := ClientHeight - D1 - 2*calMargin;
D2 := D1 div 2;
D1 := D1 - D2;
Row[0] := Row[0] + D1;
if (DrawHeader) then
Row[1] := Row[1] + D2;
{initialize each cells TRect structure using}
{the row heights from the Row[] array and the}
{column widths from the Col[] array}
for R := clStartRow to 7 do begin
for C := 0 to 6 do begin
clRowCol[R, C].Left := SumOf(Col, 0, C-1) + calMargin;
clRowCol[R, C].Right := SumOf(Col, 0, C) + calMargin;
clRowCol[R, C].Top := SumOf(Row, 0, R-1) + calMargin;
clRowCol[R, C].Bottom := SumOf(Row, 0, R) + calMargin;
end;
end;
{position and size the left and right month buttons}
{position and size the next and prev year buttons}
clBtnNextYear.Height := Row[0] - calMargin;
clBtnNextYear.Width := Col[1] - calMargin;
if clBtnNextYear.Width < clBtnNextYear.Glyph.Width + 3 then
clBtnNextYear.Width := clBtnNextYear.Glyph.Width + 3;
clBtnNextYear.Top := calMargin;
clBtnNextYear.Left := ClientWidth - calMargin - clBtnNextYear.Width;
clBtnPrevYear.Height := Row[0] - calMargin;
clBtnPrevYear.Width := Col[5] - calMargin;
if clBtnPrevYear.Width < clBtnPrevYear.Glyph.Width + 3 then
clBtnPrevYear.Width := clBtnPrevYear.Glyph.Width + 3;
clBtnPrevYear.Top := calMargin;
clBtnPrevYear.Left := calMargin;
clBtnLeft.Height := Row[0] - calMargin;
clBtnLeft.Width := Col[0] - calMargin;
if clBtnLeft.Width < clBtnLeft.Glyph.Width + 3 then
clBtnLeft.Width := clBtnLeft.Glyph.Width + 3;
clBtnLeft.Top := calMargin;
clBtnLeft.Left := clBtnPrevYear.Left + clBtnPrevYear.Width;
clBtnRight.Height := Row[0] - calMargin;
clBtnRight.Width := Col[6] - calMargin;
if clBtnRight.Width < clBtnRight.Glyph.Width + 3 then
clBtnRight.Width := clBtnRight.Glyph.Width + 3;
clBtnRight.Top := calMargin;
clBtnRight.Left := clBtnNextYear.Left - clBtnRight.Width;
{position and size "today" button}
clBtnToday.Height := Row[8];
clBtnToday.Width := Col[5] + Col[6] - calMargin;
clBtnToday.Top := ClientHeight - calMargin - clBtnToday.Height + 1;
clBtnToday.Left := ClientWidth - calMargin - clBtnToday.Width;
{position and size "revert" button}
clBtnRevert.Height := Row[8];
clBtnRevert.Width := Col[5] + Col[6] - calMargin;
clBtnRevert.Top := ClientHeight - calMargin - clBtnRevert.Height + 1;
clBtnRevert.Left := clBtnToday.Left - clBtnRevert.Width - calMargin;
end;
{$IFNDEF LCL}
procedure TOvcCustomCalendar.CMCtl3DChanged(var Msg : TMessage);
begin
inherited;
if (csLoading in ComponentState) or not HandleAllocated then
Exit;
if NewStyleControls and (FBorderStyle = bsSingle) then
{$IFNDEF LCL}
RecreateWnd;
{$ELSE}
MyMisc.RecreateWnd(Self);
{$ENDIF}
calReCalcSize;
Invalidate;
end;
{$ENDIF}
procedure TOvcCustomCalendar.CMEnter(var Msg : TMessage);
var
R : TRect;
begin
inherited;
clRevertDate := FDate;
{invalidate the active date to ensure that the focus rect is painted}
R := calGetCurrentRectangle;
InvalidateRect(Handle, @R, False);
end;
procedure TOvcCustomCalendar.CMExit(var Msg : TMessage);
var
R : TRect;
begin
inherited;
{invalidate the active date to ensure that the focus rect is painted}
R := calGetCurrentRectangle;
InvalidateRect(Handle, @R, False);
end;
procedure TOvcCustomCalendar.CMFontChanged(var Msg : TMessage);
begin
inherited;
if csLoading in ComponentState then
Exit;
calRecalcSize;
Invalidate;
end;
constructor TOvcCustomCalendar.Create(AOwner : TComponent);
begin
inherited Create(AOwner);
ControlStyle := ControlStyle + [csClickEvents, csFramed] - [csCaptureMouse];
Height := 140;
TabStop := True;
Width := 200;
{$IFDEF MSWINDOWS}
Font.Name := 'MS Sans Serif';
Font.Size := 8;
{$ELSE} //Size 8 doesn't convert in LCL to Height -11 as it should.
Font.Name := 'Arial';
Font.Height := -11;
{$ENDIF}
{$IFDEF LCL}
FCtl3D := True;
{$ENDIF}
FBorderStyle := bsNone;
FDayNameWidth := 3;
FDateFormat := dfLong;
FOptions := [cdoShortNames, cdoShowYear, cdoShowInactive,
cdoShowRevert, cdoShowToday, cdoShowNavBtns];
FWantDblClicks := True;
FWeekStarts := dtSunday;
{create navigation buttons}
clBtnLeft := TSpeedButton.Create(Self);
clBtnLeft.Parent := Self;
{$IFNDEF LCL}
clBtnLeft.Glyph.Handle := LoadBaseBitmap('ORLEFTARROW');
{$ELSE}
clBtnLeft.Glyph.LoadFromLazarusResource('ORLEFTARROW');
{$ENDIF}
clBtnLeft.OnClick := calBtnClick;
clBtnRight := TSpeedButton.Create(Self);
clBtnRight.Parent := Self;
{$IFNDEF LCL}
clBtnRight.Glyph.Handle := LoadBaseBitmap('ORRIGHTARROW');
{$ELSE}
clBtnRight.Glyph.LoadFromLazarusResource('ORRIGHTARROW');
{$ENDIF}
clBtnRight.OnClick := calBtnClick;
clBtnNextYear := TSpeedButton.Create(Self);
clBtnNextYear.Parent := Self;
{$IFNDEF LCL}
clBtnNextYear.Glyph.Handle := LoadBaseBitmap('ORRIGHTARROWS');
{$ELSE}
clBtnNextYear.Glyph.LoadFromLazarusResource('ORRIGHTARROWS');
{$ENDIF}
clBtnNextYear.OnClick := calBtnClick;
clBtnPrevYear := TSpeedButton.Create(Self);
clBtnPrevYear.Parent := Self;
{$IFNDEF LCL}
clBtnPrevYear.Glyph.Handle := LoadBaseBitmap('ORLEFTARROWS');
{$ELSE}
clBtnPrevYear.Glyph.LoadFromLazarusResource('ORLEFTARROWS');
{$ENDIF}
clBtnPrevYear.OnClick := calBtnClick;
{create "revert" button}
clBtnRevert := TSpeedButton.Create(Self);
clBtnRevert.Parent := Self;
{$IFNDEF LCL}
clBtnRevert.Glyph.Handle := LoadBaseBitmap('ORREVERT');
{$ELSE}
clBtnRevert.Glyph.LoadFromLazarusResource('ORREVERT');
{$ENDIF}
clBtnRevert.OnClick := calBtnClick;
{create "today" button}
clBtnToday := TSpeedButton.Create(Self);
clBtnToday.Parent := Self;
{$IFNDEF LCL}
clBtnToday.Glyph.Handle := LoadBaseBitmap('ORTODAY');
{$ELSE}
clBtnToday.Glyph.LoadFromLazarusResource('ORTODAY');
{$ENDIF}
clBtnToday.OnClick := calBtnClick;
{assign default color scheme}
FColors := TOvcCalColors.Create;
FColors.OnChange := calColorChange;
FColors.FCalColors := CalScheme[cscalWindows];
{assign default international support object}
FIntlSup := OvcIntlSup;
FDrawHeader:= True;
clRowCount := 8;
clStartRow := 0;
end;
constructor TOvcCustomCalendar.CreateEx(AOwner : TComponent; AsPopup : Boolean);
begin
clPopup := AsPopup;
Create(AOwner);
end;
procedure TOvcCustomCalendar.CreateParams(var Params : TCreateParams);
const
BorderStyles : array[TBorderStyle] of LongInt = (0, WS_BORDER);
begin
inherited CreateParams(Params);
with Params do begin
Style := LongInt(Style) or BorderStyles[FBorderStyle];
if clPopup then begin
Style := WS_POPUP or WS_BORDER;
WindowClass.Style := WindowClass.Style or CS_SAVEBITS;
end;
end;
if NewStyleControls and (Ctl3D or clPopup) and (FBorderStyle = bsSingle) then begin
if not clPopup then
Params.Style := Params.Style and not WS_BORDER;
Params.ExStyle := Params.ExStyle or WS_EX_CLIENTEDGE;
end;
{$IFDEF LCL}
inherited SetBorderStyle(FBorderStyle);
{$ENDIF}
{set style to reflect desire for double clicks}
if FWantDblClicks then
ControlStyle := ControlStyle + [csDoubleClicks]
else
ControlStyle := ControlStyle - [csDoubleClicks];
{get windows date mask}
FIntlSup.InternationalLongDatePChar(clMask, cdoShortNames in FOptions, False);
end;
procedure TOvcCustomCalendar.CreateWnd;
begin
inherited CreateWnd;
calRecalcSize;
{if not set, get current date}
if FDate = 0 then
SetDate(calGetValidDate(SysUtils.Date-1, +1));
end;
destructor TOvcCustomCalendar.Destroy;
begin
FColors.Free;
FColors := nil;
inherited Destroy;
end;
function TOvcCustomCalendar.DateString(const Mask : string): string;
var
M : string;
begin
M := Mask;
if Length(M) = 0 then
M := StrPas(clMask);
{convert calendar month and year to a string}
Result := FIntlSup.DateToDateString(M, DateTimeToStDate(FDate), True);
end;
function TOvcCustomCalendar.DayString: string;
begin
Result := IntlSupport.DayOfWeekToString(DayOfWeek(DateTimeToStDate(FDate)));
end;
procedure TOvcCustomCalendar.DoOnChange(Value : TDateTime);
begin
if Assigned(FOnChange) then
FOnChange(Self, Value);
end;
function TOvcCustomCalendar.DoOnGetDateEnabled(ADate : TDateTime) : Boolean;
begin
Result := True;
if Assigned(FOnGetDateEnabled) then
FOnGetDateEnabled(Self, ADate, Result);
end;
procedure TOvcCustomCalendar.DoOnMouseWheel(Shift : TShiftState; Delta, XPos, YPos : SmallInt);
var
Key : Word;
begin
inherited DoOnMouseWheel(Shift, Delta, XPos, YPos);
if Abs(Delta) = WHEEL_DELTA then begin
{inc/dec month}
if Delta < 0 then
Key := VK_NEXT
else
Key := VK_PRIOR;
KeyDown(Key, []);
end else if Abs(Delta) > WHEEL_DELTA then begin
{inc/dec year}
if Delta < 0 then
Key := VK_NEXT
else
Key := VK_PRIOR;
KeyDown(Key, [ssCtrl]);
end else if Abs(Delta) < WHEEL_DELTA then begin
{inc/dec Week}
if Delta < 0 then
Key := VK_DOWN
else
Key := VK_UP;
KeyDown(Key, []);
end;
end;
function TOvcCustomCalendar.GetAsDateTime : TDateTime;
begin
Result := FDate;
end;
function TOvcCustomCalendar.GetAsStDate : TStDate;
begin
Result := DateTimeToStDate(FDate)
end;
function TOvcCustomCalendar.IsReadOnly : Boolean;
begin
Result := ReadOnly;
end;
{revised}
procedure TOvcCustomCalendar.KeyDown(var Key : Word; Shift : TShiftState);
var
Y : Word;
M : Word;
D : Word;
HD : TDateTime;
begin
inherited KeyDown(Key, Shift);
if IsReadOnly then
Exit;
HD := FDate;
case Key of
VK_LEFT : if Shift = [] then
SetDate(calGetValidDate(FDate, -1));
VK_RIGHT : if Shift = [] then
SetDate(calGetValidDate(FDate, +1));
VK_UP : if Shift = [] then
SetDate(calGetValidDate(FDate, -7));
VK_DOWN : if Shift = [] then
SetDate(calGetValidDate(FDate, +7));
VK_HOME :
begin
if ssCtrl in Shift then begin
DecodeDate(FDate, Y, M, D);
SetDate(calGetValidDate(EncodeDate(Y, 1, 1)-1, +1));
end else if Shift = [] then begin
DecodeDate(FDate, Y, M, D);
SetDate(calGetValidDate(EncodeDate(Y, M, 1)-1, +1));
end;
end;
VK_END :
begin
if ssCtrl in Shift then begin
DecodeDate(FDate, Y, M, D);
SetDate(calGetValidDate(EncodeDate(Y, 12, DaysInMonth(12, Y, 0))+1, -1));
end else if Shift = [] then begin
DecodeDate(FDate, Y, M, D);
SetDate(calGetValidDate(EncodeDate(Y, M, DaysInMonth(M, Y, 0))+1, -1));
end;
end;
VK_PRIOR :
begin
if ssCtrl in Shift then begin
IncYear(-1);
end else if Shift = [] then begin
IncMonth(-1);
end;
end;
VK_NEXT :
begin
if ssCtrl in Shift then begin
IncYear(1);
end else if Shift = [] then begin
IncMonth(1);
end;
end;
VK_BACK :
begin
if ssAlt in Shift then
SetDate(calGetValidDate(SysUtils.Date-1, +1));
end;
VK_ESCAPE:
begin
if Shift = [] then
SetDate(calGetValidDate(clRevertDate-1, +1));
end;
end;
if HD <> FDate then begin
FBrowsing := True;
try
DoOnChange(FDate);
finally
FBrowsing := False;
end;
end;
end;
procedure TOvcCustomCalendar.KeyPress(var Key : Char);
begin
inherited KeyPress(Key);
if IsReadOnly then
Exit;
case Key of
'+' : SetDate(calGetValidDate(FDate, +1));
'-' : SetDate(calGetValidDate(FDate, -1));
#13 : DoOnChange(FDate); {date selected}
#32 : DoOnChange(FDate); {date selected}
^Z : SetDate(calGetValidDate(SysUtils.Date-1, +1));
end;
end;
procedure TOvcCustomCalendar.MouseDown(Button : TMouseButton; Shift : TShiftState; X, Y : Integer);
var
Yr : Word;
M : Word;
D : Word;
Yr2 : Word;
M2 : Word;
D2 : Word;
R, C : Integer;
OldIdx : Integer;
NewIdx : Integer;
Re : TRect;
Ignore : Boolean;
begin
{exit if this click happens when the popup menu is active}
if clInPopup then
Exit;
SetFocus;
inherited MouseDown(Button, Shift, X, Y);
if IsReadOnly then
Exit;
{if we have the mouse captured, see if a button was clicked}
if GetCapture = Handle then begin
if (cdoShowNavBtns in Options) then begin
Re := clBtnLeft.ClientRect;
Re.TopLeft := ScreenToClient(clBtnLeft.ClientToScreen(Re.TopLeft));
Re.BottomRight := ScreenToClient(clBtnLeft.ClientToScreen(Re.BottomRight));
if PtInRect(Re, Point(X, Y)) then begin
clBtnLeft.Click;
Exit;
end;
Re := clBtnRight.ClientRect;
Re.TopLeft := ScreenToClient(clBtnRight.ClientToScreen(Re.TopLeft));
Re.BottomRight := ScreenToClient(clBtnRight.ClientToScreen(Re.BottomRight));
if PtInRect(Re, Point(X, Y)) then begin
clBtnRight.Click;
Exit;
end;
Re := clBtnNextYear.ClientRect;
Re.TopLeft := ScreenToClient(clBtnNextYear.ClientToScreen(Re.TopLeft));
Re.BottomRight := ScreenToClient(clBtnNextYear.ClientToScreen(Re.BottomRight));
if PtInRect(Re, Point(X, Y)) then begin
clBtnNextYear.Click;
Exit;
end;
Re := clBtnPrevYear.ClientRect;
Re.TopLeft := ScreenToClient(clBtnPrevYear.ClientToScreen(Re.TopLeft));
Re.BottomRight := ScreenToClient(clBtnPrevYear.ClientToScreen(Re.BottomRight));
if PtInRect(Re, Point(X, Y)) then begin
clBtnPrevYear.Click;
Exit;
end;
end;
if (cdoShowRevert in Options) then begin
Re := clBtnRevert.ClientRect;
Re.TopLeft := ScreenToClient(clBtnRevert.ClientToScreen(Re.TopLeft));
Re.BottomRight := ScreenToClient(clBtnRevert.ClientToScreen(Re.BottomRight));
if PtInRect(Re, Point(X, Y)) then begin
clBtnRevert.Click;
Exit;
end;
end;
if (cdoShowToday in Options) then begin
Re := clBtnToday.ClientRect;
Re.TopLeft := ScreenToClient(clBtnToday.ClientToScreen(Re.TopLeft));
Re.BottomRight := ScreenToClient(clBtnToday.ClientToScreen(Re.BottomRight));
if PtInRect(Re, Point(X, Y)) then begin
clBtnToday.Click;
Exit;
end;
end;
end;
{save current date}
DecodeDate(FDate, Yr, M, D);
M2 := M;
{calculate the row and column clicked on}
for R := 2 to 8 do begin
for C := 0 to 6 do begin
if PtInRect(clRowCol[R,C], Point(X, Y)) then begin
{convert to an index}
NewIdx := ((R-2) * 7) + Succ(C);
OldIdx := clFirst + Pred(clDay);
Ignore := False;
if NewIdx <> OldIdx then begin
{see if this date is disabled - selection not allowed}
if not DoOnGetDateEnabled(FDate+(NewIdx-OldIdx)) then
Break;
DecodeDate(FDate+(NewIdx-OldIdx), Yr2, M2, D2);
if not (cdoShowInactive in FOptions) then begin
{will this change the month?}
if M2 <> M then
Ignore := True;
end;
{convert to a date and redraw}
if not Ignore then
SetDate(FDate+(NewIdx-OldIdx));
end;
if (not Ignore) and (Button = mbLeft) then begin
if M2 <> M then begin
FBrowsing := True;
try
DoOnChange(FDate);
finally
FBrowsing := False;
end;
end else
DoOnChange(FDate);
end;
Break;
end;
end;
end;
end;
procedure TOvcCustomCalendar.MouseUp(Button : TMouseButton; Shift : TShiftState; X, Y : Integer);
var
P : TPoint;
M : TPopUpMenu;
MI : TMenuItem;
I : Integer;
J : Integer;
K : Integer;
MO : Integer;
YR : Word;
MM : Word;
DA : Word;
HC : Boolean;
begin
inherited MouseUp(Button, Shift, X, Y);
if (PopUpMenu = nil) and (Button = mbRight) and
(Y < clRowCol[1,0].Top) {above day names} and
(X > clBtnPrevYear.Left + clBtnNextYear.Width) and
(X < clBtnNextYear.Left) then begin
if not Focused and CanFocus then
SetFocus;
M := TPopupMenu.Create(Self);
try
DecodeDate(FDate, YR, MM, DA);
MO := MM; {convert to integer to avoid wrap-around errors with words}
{determine the starting month}
I := MO - 3;
if I < 1 then
I := MO - 3 + 12;
{determine the ending month + 1}
J := MO + 4;
if J > 12 then
J := MO + 4 - 12;
K := 0;
{create the menu items}
repeat
MI := TMenuItem.Create(M);
MI.Caption := LongMonthNames[I];
MI.Enabled := Enabled;
MI.OnClick := calChangeMonth;
MI.Tag := I;
MI.HelpContext := K;
M.Items.Add(MI);
Inc(I);
Inc(K);
if I > 12 then
I := 1;
until I = J;
HC := GetCapture = Handle;
P.X := X-20;
P.Y := Y - ((GetSystemMetrics(SM_CYMENU)*7) div 2);
P := ClientToScreen(P);
{move the mouse to cause the menu item to highlight}
PostMessage(Handle, WM_MOUSEMOVE, 0, MAKELONG(P.X,P.Y+1));
clInPopup := True;
try
M.PopUp(P.X, P.Y);
Application.ProcessMessages;
{capture the mouse again}
if clPopup and HC then
SetCapture(Handle);
finally
clInPopup := false;
end;
finally
M.Free;
end;
end;
end;
procedure TOvcCustomCalendar.IncDay(Delta : Integer);
{-change the day by Delta (signed) days}
begin
if Delta > 0 then
SetDate(calGetValidDate(FDate+Delta-1, +1))
else
SetDate(calGetValidDate(FDate+Delta+1, -1))
end;
procedure TOvcCustomCalendar.IncMonth(Delta : Integer);
{-change the month by Delta (signed) months}
var
Y, M, D : Word;
iY, iM, iD : Integer;
begin
DecodeDate(FDate, Y, M, D);
iY := Y; iM := M; iD := D;
Inc(iM, Delta);
if iM > 12 then begin
iM := iM - 12;
Inc(iY);
end else if iM < 1 then begin
iM := iM + 12;
Dec(iY);
end;
if iD > DaysInMonth(iM, iY, 0) then
iD := DaysInMonth(iM, iY, 0);
SetDate(calGetValidDate(EncodeDate(iY, iM, iD)-1, +1));
end;
procedure TOvcCustomCalendar.IncYear(Delta : Integer);
var
Y, M, D : Word;
iY, iM, iD : Integer;
begin
DecodeDate(FDate, Y, M, D);
iY := Y; iM := M; iD := D;
Inc(iY, Delta);
if iD > DaysInMonth(iM, iY, 0) then
iD := DaysInMonth(iM, iY, 0);
SetDate(calGetValidDate(EncodeDate(iY, iM, iD)-1, +1));
end;
function TOvcCustomCalendar.MonthString: string;
var
M, D, Y : Word;
begin
DecodeDate(FDate, Y, M, D);
Result := IntlSupport.MonthToString(M);
end;
procedure TOvcCustomCalendar.Paint;
var
R, C : Integer;
I : Integer;
{CurIndex : Integer;}
SatCol : Integer;
SunCol : Integer;
DOW : TOvcDayType;
procedure DrawDate;
var
R : TRect;
S : string;
begin
if FDateFormat = dfLong then
if cdoShowYear in FOptions then
S := FormatDateTime('mmmm yyyy', FDate)
else
S := FormatDateTime('mmmm', FDate)
else
if cdoShowYear in FOptions then
S := FormatDateTime('mmm yyyy', FDate)
else
S := FormatDateTime('mmm', FDate);
R := clRowCol[0,1];
R.Right := clRowCol[0,6].Left;
{switch to short date format if string won't fit}
if FDateFormat = dfLong then
if Canvas.TextWidth(S) > R.Right-R.Left then
S := FormatDateTime('mmm yyyy', FDate);
{$IFDEF LCL}
S := SysToUTF8(S); //DrawText expects UTF8 with most widgetsets
{$ENDIF}
Canvas.Font.Color := FColors.MonthAndYear;
if Assigned(FOnDrawDate) then
FOnDrawDate(Self, FDate, R)
else
DrawText(Canvas.Handle, @S[1], Length(S), R, DT_SINGLELINE or DT_CENTER or DT_VCENTER);
end;
procedure DrawDayNames;
var
I : Integer;
S : string[3];
begin
{draw the day name column labels}
Canvas.Font.Color := FColors.DayNames;
I := 0;
DOW := FWeekStarts;
repeat
{record columns for weekends}
if DOW = dtSaturday then
SatCol := I;
if DOW = dtSunday then
SunCol := I;
{get the day name}
S := Copy(ShortDayNames[Ord(DOW)+1], 1, FDayNameWidth);
{$IFDEF LCL}
S := SysToUTF8(S);
{$ENDIF}
{draw the day name above each column}
DrawText(Canvas.Handle, @S[1], Length(S), clRowCol[1,I],
DT_SINGLELINE or DT_CENTER or DT_VCENTER);
Inc(I);
if DOW < High(DOW) then
Inc(DOW)
else
DOW := Low(DOW);
until DOW = WeekStarts;
end;
procedure DrawLine;
begin
if Ctl3D then begin
Canvas.Pen.Color := clBtnHighlight;
Canvas.MoveTo(0, clRowCol[1,0].Bottom-3);
Canvas.LineTo(ClientWidth, clRowCol[1,0].Bottom-3);
Canvas.Pen.Color := clBtnShadow;
Canvas.MoveTo(0, clRowCol[1,0].Bottom-2);
Canvas.LineTo(ClientWidth, clRowCol[1,0].Bottom-2);
end else if BorderStyle = bsSingle then begin
Canvas.Pen.Color := Font.Color;
Canvas.MoveTo(0, clRowCol[1,0].Bottom-3);
Canvas.LineTo(ClientWidth, clRowCol[1,0].Bottom-3);
end;
end;
procedure DrawDay(R, C, I : Integer; Grayed{, Current} : Boolean);
var
Cl : TColor;
OldIdx : Integer;
NewIdx : Integer;
S : string[10];
begin
{avoid painting day number under buttons}
if cdoShowRevert in FOptions then
if (R = 8) {bottom} and (C >= 3) then
Exit;
if cdoShowToday in FOptions then
if (R = 8) {bottom} and (C >= 5) then
Exit;
{convert to a string and draw it centered in its rectangle}
S := IntToStr(clCalendar[I]);
if Grayed then
Canvas.Font.Color := FColors.InactiveDays;
if not Grayed or (cdoShowInactive in FOptions) then begin
NewIdx := ((R-2) * 7) + Succ(C);
OldIdx := clFirst + Pred(clDay);
if Assigned(FOnGetHighlight) then begin
Cl := Canvas.Font.Color;
FOnGetHighlight(Self, FDate+(NewIdx-OldIdx), Cl);
Canvas.Font.Color := Cl;
end;
{$IFDEF LCL}
S := SysToUTF8(S);
{$ENDIF}
if Assigned(FOnDrawItem) then
FOnDrawItem(Self, FDate+(NewIdx-OldIdx), clRowCol[R,C])
else
DrawText(Canvas.Handle, @S[1], Length(S), clRowCol[R,C], DT_SINGLELINE or DT_CENTER or DT_VCENTER);
end;
end;
procedure DrawFocusBox;
var
R : TRect;
S : string[10];
BS : TButtonStyle;
begin
S := IntToStr(clDay);
if Ctl3D then
BS := bsNew
else
BS := bsWin31;
if Focused then
R := DrawButtonFace(Canvas, calGetCurrentRectangle, 1, BS, True, True, False)
else
R := DrawButtonFace(Canvas, calGetCurrentRectangle, 1, BS, True, False, False);
{$IFDEF LCL}
S := SysToUTF8(S);
{$ENDIF}
DrawText(Canvas.Handle, @S[1], Length(S), R, DT_CENTER or DT_VCENTER or DT_SINGLELINE);
end;
begin
Canvas.Font := Font;
Canvas.Brush.Color := Color;{clBtnFace;}
Canvas.FillRect(ClientRect);
{draw the month and year at the top of the calendar}
DrawDate;
{draw the days of the week}
DrawDayNames;
{draw line under day names}
DrawLine;
{draw each day}
{CurIndex := clFirst + Pred(clDay);}
I := 1;
for R := 2 to 8 do
for C := 0 to 6 do begin
if (C = SatCol) or (C = SunCol) then
Canvas.Font.Color := FColors.WeekEnd
else
Canvas.Font.Color := FColors.Days;
DrawDay(R, C, I, (I < clFirst) or (I > clLast){, I = CurIndex});
Inc(I);
end;
Canvas.Font.Color := FColors.ActiveDay;
if not Assigned(FOnDrawItem) then
if not (cdoHideActive in FOptions) then
DrawFocusBox;
end;
function TOvcCustomCalendar.GetCalendarDate : TDateTime;
begin
Result := FDate;
end;
function TOvcCustomCalendar.GetDay : Integer;
begin
Result := clDay;
end;
function TOvcCustomCalendar.GetMonth : Integer;
begin
Result := clMonth;
end;
function TOvcCustomCalendar.GetYear : Integer;
begin
Result := clYear;
end;
procedure TOvcCustomCalendar.SetAsDateTime(Value : TDateTime);
begin
SetDate(calGetValidDate(Value-1, +1));
end;
procedure TOvcCustomCalendar.SetAsStDate(Value : TStDate);
begin
SetDate(calGetValidDate(StDateToDateTime(Value)-1, +1));
end;
procedure TOvcCustomCalendar.SetBorderStyle(Value : TBorderStyle);
begin
if Value <> FBorderStyle then begin
FBorderStyle := Value;
{$IFNDEF LCL}
RecreateWnd;
{$ELSE}
MyMisc.RecreateWnd(Self);
{$ENDIF}
end;
end;
procedure TOvcCustomCalendar.SetBounds(ALeft, ATop, AWidth, AHeight : Integer);
begin
inherited Setbounds(ALeft, ATop, AWidth, AHeight);
if csLoading in ComponentState then
Exit;
calRecalcSize;
end;
procedure TOvcCustomCalendar.SetCalendarDate(Value : TDateTime);
var
NewDate : TStDate;
begin
NewDate := DateTimeToStDate(Value);
if (NewDate = BadDate) or (NewDate = CalendarDate) or
(IncDateTrunc(NewDate, 1, 0) = BadDate) or
(IncDateTrunc(NewDate, -1, 0) = BadDate) then begin
Exit;
end;
SetDate(calGetValidDate(Value-1, +1));
end;
procedure TOvcCustomCalendar.SetDate(Value : TDateTime);
var
R : TRect;
Y : Word;
M : Word;
D : Word;
begin
if Value <> FDate then begin
{determine if the new date is in the same month}
DecodeDate(Value, Y, M, D);
if (clYear = Y) and (clMonth = M) then begin
{invalidate the old date}
R := calGetCurrentRectangle;
InvalidateRect(Handle, @R, False);
end else
Invalidate;
DecodeDate(Value, clYear, clMonth, clDay);
FDate := Value;
calRebuildCalArray;
{invalidate the new date}
R := calGetCurrentRectangle;
InvalidateRect(Handle, @R, False);
end;
end;
procedure TOvcCustomCalendar.SetDateFormat(Value : TOvcDateFormat);
begin
if Value <> FDateFormat then begin
FDateFormat := Value;
Invalidate;
end;
end;
procedure TOvcCustomCalendar.SetDayNameWidth(Value : TOvcDayNameWidth);
begin
if Value <> FDayNameWidth then begin
FDayNameWidth := Value;
Invalidate;
end;
end;
procedure TOvcCustomCalendar.SetDisplayOptions(Value : TOvcCalDisplayOptions);
begin
if Value <> FOptions then begin
FOptions := Value;
calRecalcSize;
Invalidate;
end;
end;
procedure TOvcCustomCalendar.SetDrawHeader(Value : Boolean);
{-set the DrawHeader property value}
begin
if Value <> FDrawHeader then begin
FDrawHeader := Value;
if FDrawHeader then begin
clStartRow := 0;
clRowCount := 8;
end else begin
clStartRow := 2;
clRowCount := 7;
end;
calRecalcSize;
Refresh;
end;
end;
procedure TOvcCustomCalendar.SetIntlSupport(Value : TOvcIntlSup);
{-set the international support object this field will use}
begin
if Assigned(Value) then
FIntlSup := Value
else
FIntlSup := OvcIntlSup;
end;
procedure TOvcCustomCalendar.SetToday;
{-set the calendar to todays date}
begin
SetDate(calGetValidDate(SysUtils.Date-1, +1));
end;
procedure TOvcCustomCalendar.SetWantDblClicks(Value : Boolean);
begin
if Value <> FWantDblClicks then begin
FWantDblClicks := Value;
{$IFNDEF LCL}
RecreateWnd;
{$ELSE}
MyMisc.RecreateWnd(Self);
{$ENDIF}
end;
end;
procedure TOvcCustomCalendar.SetWeekStarts(Value : TOvcDayType);
begin
if Value <> FWeekStarts then begin
FWeekStarts := Value;
if csLoading in ComponentState then
Exit;
calRebuildCalArray;
Invalidate;
end;
end;
procedure TOvcCustomCalendar.WMEraseBkgnd(var Msg : TWMEraseBkgnd);
begin
Msg.Result := 1; {don't erase background, just say we did}
end;
procedure TOvcCustomCalendar.WMGetDlgCode(var Msg : TWMGetDlgCode);
begin
Msg.Result := DLGC_WANTARROWS;
end;
procedure TOvcCustomCalendar.WMKillFocus(var Msg : TWMKillFocus);
begin
inherited;
Invalidate;
end;
end.