{----------------------------------------------------------------------------- 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/MPL-1.1.html Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either expressed or implied. See the License for the specific language governing rights and limitations under the License. The Original Code is: JvCheckListBox.PAS, released on 2001-02-28. The Initial Developer of the Original Code is Sébastien Buysse [sbuysse att buypin dott com] Portions created by Sébastien Buysse are Copyright (C) 2001 Sébastien Buysse. All Rights Reserved. This is a merging of the code in the original JvCheckListBox.pas and JvFixedCheckListBox.pas Merging done 2002-06-05 by Peter Thornqvist [peter3 at sourceforge dot net] Contributor(s): Michael Beck [mbeck att bigfoot dott com]. Peter Below <100113 dott 1101 att compuserve dott com> You may retrieve the latest version of this file at the Project JEDI's JVCL home page, located at http://jvcl.sourceforge.net Known Issues: -----------------------------------------------------------------------------} // $Id: JvCheckListBox.pas 10909 2006-08-23 07:34:47Z obones $ unit JvCheckListBox; {$I jvcl.inc} interface uses {$IFDEF UNITVERSIONING} JclUnitVersioning, {$ENDIF UNITVERSIONING} {$IFDEF CLR} System.Runtime.InteropServices, {$ENDIF CLR} Windows, Messages, SysUtils, Classes, Contnrs, Controls, Graphics, StdCtrls, JvExCheckLst, JvVCL5Utils, JvDataSourceIntf; const DefaultValueChecked = '1'; DefaultValueUnchecked = '0'; type TJvCheckListBox = class; TJvCheckListBoxDataConnector = class(TJvLookupDataConnector) private FCheckListBox: TJvCheckListBox; FValueChecked: string; FValueUnchecked: string; FMap: TList; FRecNumMap: TBucketList; procedure SetValueChecked(const Value: string); procedure SetValueUnchecked(const Value: string); function IsValueCheckedStored: Boolean; function IsValueUncheckedStored: Boolean; protected procedure Popuplate; virtual; procedure ActiveChanged; override; procedure UpdateData; override; procedure RecordChanged; override; procedure GetKeyNames(out KeyName, ListKeyName: TDataFieldString); procedure GotoCurrent; public constructor Create(ACheckListBox: TJvCheckListBox); destructor Destroy; override; procedure Assign(Source: TPersistent); override; function IsValid: Boolean; virtual; published property ValueChecked: string read FValueChecked write SetValueChecked stored IsValueCheckedStored; property ValueUnchecked: string read FValueUnchecked write SetValueUnchecked stored IsValueUncheckedStored; end; TJvCheckListBox = class(TJvExCheckListBox) {$IFDEF VCL} private FHotTrack: Boolean; FOnSelectCancel: TNotifyEvent; FMaxWidth: Integer; FScroll: Boolean; FOnHScroll: TNotifyEvent; FOnVScroll: TNotifyEvent; procedure SetHScroll(const Value: Boolean); procedure WMHScroll(var Msg: TWMHScroll); message WM_HSCROLL; procedure WMVScroll(var Msg: TWMVScroll); message WM_VSCROLL; procedure CNDrawItem(var Msg: TWMDrawItem); message CN_DRAWITEM; procedure LBNSelCancel(var Msg: TMessage); message LBN_SELCANCEL; procedure RefreshH; procedure SetHotTrack(const Value: Boolean); protected procedure CreateParams(var Params: TCreateParams); override; procedure WndProc(var Msg: TMessage); override; procedure MouseEnter(Control: TControl); override; procedure MouseLeave(Control: TControl); override; {$ENDIF VCL} {$IFDEF COMPILER5} private FHeaders: TList; FHeaderColor: TColor; FHeaderBackgroundColor: TColor; function GetHeader(Index: Integer): Boolean; procedure SetHeader(Index: Integer; const Value: Boolean); procedure SetHeaderColor(Value: TColor); procedure SetHeaderBackgroundColor(Value: TColor); public property Header[Index: Integer]: Boolean read GetHeader write SetHeader; published property HeaderColor: TColor read FHeaderColor write SetHeaderColor default clInfoText; property HeaderBackgroundColor: TColor read FHeaderBackgroundColor write SetHeaderBackgroundColor default clInfoBk; {$ENDIF COMPILER5} private FDataConnector: TJvCheckListBoxDataConnector; FOnItemDrawing: TDrawItemEvent; procedure SetDataConnector(const Value: TJvCheckListBoxDataConnector); procedure CMChanged(var Msg: TMessage); message CM_CHANGED; protected function CreateDataConnector: TJvCheckListBoxDataConnector; virtual; procedure ClickCheck; override; procedure DrawItem(Index: Integer; Rect: TRect; State: TOwnerDrawState); override; procedure DoItemDrawing(Index: Integer; Rect: TRect; State: TOwnerDrawState); public constructor Create(AOwner: TComponent); override; destructor Destroy; override; function SearchExactString(Value: string; CaseSensitive: Boolean = True): Integer; function SearchPrefix(Value: string; CaseSensitive: Boolean = True): Integer; function SearchSubString(Value: string; CaseSensitive: Boolean = True): Integer; function DeleteExactString(Value: string; All: Boolean; CaseSensitive: Boolean = True): Integer; procedure SelectAll; {$IFDEF VCL}{$IFDEF COMPILER6_UP} override; {$ENDIF}{$ENDIF} procedure UnselectAll; procedure InvertSelection; procedure CheckAll; procedure UnCheckAll; procedure InvertCheck; function GetChecked: TStringList; function GetUnChecked: TStringList; procedure DeleteSelected; {$IFDEF VCL}{$IFDEF COMPILER6_UP} override; {$ENDIF}{$ENDIF} procedure SaveToFile(FileName: TFileName); procedure LoadFromFile(FileName: TFileName); procedure LoadFromStream(Stream: TStream); procedure SaveToStream(Stream: TStream); published property OnItemDrawing: TDrawItemEvent read FOnItemDrawing write FOnItemDrawing; property MultiSelect; property HintColor; property OnMouseEnter; property OnMouseLeave; property OnParentColorChange; {$IFDEF VCL} property HotTrack: Boolean read FHotTrack write SetHotTrack default False; property HorScrollbar: Boolean read FScroll write SetHScroll default True; property OnSelectCancel: TNotifyEvent read FOnSelectCancel write FOnSelectCancel; property OnVerticalScroll: TNotifyEvent read FOnVScroll write FOnVScroll; property OnHorizontalScroll: TNotifyEvent read FOnHScroll write FOnHScroll; {$ENDIF VCL} property DataConnector: TJvCheckListBoxDataConnector read FDataConnector write SetDataConnector; end; {$IFDEF UNITVERSIONING} const UnitVersioning: TUnitVersionInfo = ( RCSfile: '$URL: https://jvcl.svn.sourceforge.net/svnroot/jvcl/tags/JVCL3_32/run/JvCheckListBox.pas $'; Revision: '$Revision: 10909 $'; Date: '$Date: 2006-08-23 09:34:47 +0200 (mer., 23 août 2006) $'; LogPath: 'JVCL\run' ); {$ENDIF UNITVERSIONING} implementation uses JvItemsSearchs, JvJCLUtils; //=== { TJvCheckListBoxDataConnector } ======================================= constructor TJvCheckListBoxDataConnector.Create(ACheckListBox: TJvCheckListBox); begin inherited Create; FCheckListBox := ACheckListBox; FValueChecked := '1'; FValueUnchecked := '0'; FMap := TList.Create; FRecNumMap := TBucketList.Create(bl256); end; destructor TJvCheckListBoxDataConnector.Destroy; begin FMap.Free; FRecNumMap.Free; inherited Destroy; end; procedure TJvCheckListBoxDataConnector.GetKeyNames(out KeyName, ListKeyName: TDataFieldString); begin if List.Key.IsValid then ListKeyName := List.KeyField; if Key.IsValid then KeyName := KeyField; if ListKeyName = '' then ListKeyName := KeyName; if KeyName = '' then KeyName := ListKeyName; end; procedure TJvCheckListBoxDataConnector.GotoCurrent; var RecNo: Integer; begin if IsValid then begin RecNo := Integer(FMap[FCheckListBox.ItemIndex]); if ListSource.RecNo <> RecNo then begin Active := False; try ListSource.RecNo := RecNo; finally Active := True; end; end; end; end; function TJvCheckListBoxDataConnector.IsValid: Boolean; begin Result := List.DataSetConnected and List.Field.IsValid and DataSetConnected and Field.IsValid and (Key.IsValid or List.Key.IsValid or (ListSource.DataSet = DataSource.DataSet)); end; function TJvCheckListBoxDataConnector.IsValueCheckedStored: Boolean; begin Result := FValueChecked <> DefaultValueChecked; end; function TJvCheckListBoxDataConnector.IsValueUncheckedStored: Boolean; begin Result := FValueUnchecked <> DefaultValueUnchecked; end; procedure TJvCheckListBoxDataConnector.Popuplate; var IsChecked: TList; ListKeyName, KeyName: TDataFieldString; I, Index: Integer; begin FMap.Clear; FRecNumMap.Clear; FCheckListBox.Items.BeginUpdate; try FCheckListBox.Items.Clear; Index := -1; if IsValid then begin ListSource.BeginUpdate; try if DataSource.DataSet <> ListSource.DataSet then DataSource.BeginUpdate; try IsChecked := TList.Create; try ListSource.First; while not ListSource.Eof do begin Index := FCheckListBox.Items.Add(List.Field.AsString); FMap.Add(TObject(ListSource.RecNo)); FRecNumMap.Add(TObject(ListSource.RecNo), TObject(Index)); if ListSource.DataSet = DataSource.DataSet then IsChecked.Add(TObject(AnsiCompareText(Field.AsString, ValueUnchecked) <> 0)) else begin GetKeyNames(KeyName, ListKeyName); if DataSource.Locate(KeyName, ListSource.FieldValue[ListSource.FieldByName(ListKeyName)], []) then IsChecked.Add(TObject(AnsiCompareText(Field.AsString, ValueUnchecked) <> 0)) else IsChecked.Add(TObject(False)); end; ListSource.Next; end; for I := 0 to IsChecked.Count - 1 do FCheckListBox.Checked[I] := Boolean(IsChecked[I]); finally IsChecked.Free; end; finally if DataSource.DataSet <> ListSource.DataSet then DataSource.EndUpdate; end; finally ListSource.EndUpdate; end; if not FRecNumMap.Find(TObject(ListSource.RecNo), Pointer(Index)) then Index := -1; end; FCheckListBox.ItemIndex := Index; finally FCheckListBox.Items.EndUpdate; end end; procedure TJvCheckListBoxDataConnector.Assign(Source: TPersistent); begin inherited Assign(Source); if Source is TJvCheckListBoxDataConnector then begin FValueChecked := TJvCheckListBoxDataConnector(Source).ValueChecked; FValueUnchecked := TJvCheckListBoxDataConnector(Source).ValueUnchecked; Reset; end; end; procedure TJvCheckListBoxDataConnector.ActiveChanged; begin Popuplate; end; procedure TJvCheckListBoxDataConnector.RecordChanged; var Index: Integer; begin if IsValid then begin if ListSource.RecordCount <> FCheckListBox.Items.Count then Popuplate else if ListSource.RecNo <> -1 then begin if FRecNumMap.Find(TObject(ListSource.RecNo), Pointer(Index)) then begin FCheckListBox.Items[Index] := List.Field.AsString; FCheckListBox.Checked[Index] := AnsiCompareText(Field.AsString, ValueUnchecked) <> 0; if Index <> FCheckListBox.ItemIndex then FCheckListBox.ItemIndex := Index; end; end; end; end; procedure TJvCheckListBoxDataConnector.UpdateData; var KeyName, ListKeyName: TDataFieldString; Value: string; begin if Field.CanModify and IsValid and (ValueChecked <> '') and (ValueUnchecked <> '') and (FCheckListBox.ItemIndex <> -1) then begin if FCheckListBox.Checked[FCheckListBox.ItemIndex] then Value := ValueChecked else Value := ValueUnchecked; GotoCurrent; DataSource.Edit; if ListSource.DataSet = DataSource.DataSet then Field.AsString := Value else begin GetKeyNames(KeyName, ListKeyName); DataSource.BeginUpdate; try if DataSource.Locate(KeyName, ListSource.FieldValue[ListSource.FieldByName(ListKeyName)], []) then Field.AsString := Value; finally DataSource.EndUpdate; end; end; end; end; procedure TJvCheckListBoxDataConnector.SetValueChecked(const Value: string); begin if Value <> FValueChecked then begin FValueChecked := Value; Reset; end; end; procedure TJvCheckListBoxDataConnector.SetValueUnchecked(const Value: string); begin if Value <> FValueUnchecked then begin FValueUnchecked := Value; Reset; end; end; //=== { TJvCheckListBox } ==================================================== type // Used for the load/save methods TCheckListRecord = record Checked: Boolean; StringSize: Integer; end; constructor TJvCheckListBox.Create(AOwner: TComponent); begin inherited Create(AOwner); FDataConnector := CreateDataConnector; {$IFDEF COMPILER5} FHeaders := TList.Create; FHeaderColor := clInfoText; FHeaderBackgroundColor := clInfoBk; {$ENDIF COMPILER5} {$IFDEF VCL} FHotTrack := False; FMaxWidth := 0; FScroll := True; {$ENDIF VCL} // ControlStyle := ControlStyle + [csAcceptsControls]; end; destructor TJvCheckListBox.Destroy; begin {$IFDEF COMPILER5} FHeaders.Free; {$ENDIF COMPILER5} FDataConnector.Free; inherited Destroy; end; {$IFDEF COMPILER5} procedure TJvCheckListBox.SetHeaderColor(Value: TColor); begin if Value <> FHeaderColor then begin FHeaderColor := Value; Invalidate; end; end; procedure TJvCheckListBox.SetHeaderBackgroundColor(Value: TColor); begin if Value <> FHeaderBackgroundColor then begin FHeaderBackgroundColor := Value; Invalidate; end; end; procedure TJvCheckListBox.SetHeader(Index: Integer; const Value: Boolean); var Idx: Integer; begin Idx := FHeaders.IndexOf(Pointer(Index)); if Idx < 0 then begin if Value then FHeaders.Add(Pointer(Index)); end else if not Value then FHeaders.Delete(Idx); end; function TJvCheckListBox.GetHeader(Index: Integer): Boolean; begin Result := FHeaders.IndexOf(Pointer(Index)) >= 0; end; {$ENDIF COMPILER5} function TJvCheckListBox.CreateDataConnector: TJvCheckListBoxDataConnector; begin Result := TJvCheckListBoxDataConnector.Create(Self); end; procedure TJvCheckListBox.ClickCheck; begin inherited; if not (csLoading in ComponentState) then begin DataConnector.Modify; DataConnector.UpdateRecord; end; end; procedure TJvCheckListBox.SetDataConnector(const Value: TJvCheckListBoxDataConnector); begin if Value <> FDataConnector then FDataConnector.Assign(Value); end; procedure TJvCheckListBox.CMChanged(var Msg: TMessage); begin inherited; DataConnector.GotoCurrent; end; procedure TJvCheckListBox.DoItemDrawing(Index: Integer; Rect: TRect; State: TOwnerDrawState); begin if Assigned(OnItemDrawing) then OnItemDrawing(Self, Index, Rect, State); end; procedure TJvCheckListBox.DrawItem(Index: Integer; Rect: TRect; State: TOwnerDrawState); begin DoItemDrawing(Index, Rect, State); {$IFDEF COMPILER5} if Header[Index] then begin Canvas.Font.Color := HeaderColor; Canvas.Brush.Color := HeaderBackgroundColor; end; {$ENDIF COMPILER5} inherited DrawItem(Index, Rect, State); end; procedure TJvCheckListBox.CheckAll; var I: Integer; begin for I := 0 to Items.Count - 1 do Checked[I] := True; end; {$IFDEF VCL} procedure TJvCheckListBox.CreateParams(var Params: TCreateParams); begin inherited CreateParams(Params); with Params do if FScroll then Style := Style or WS_HSCROLL else Style := Style xor WS_HSCROLL; end; {$ENDIF VCL} function TJvCheckListBox.DeleteExactString(Value: string; All: Boolean; CaseSensitive: Boolean): Integer; begin Result := TJvItemsSearchs.DeleteExactString(Items, Value, CaseSensitive); end; procedure TJvCheckListBox.DeleteSelected; var I: Integer; begin if MultiSelect then begin for I := Items.Count - 1 downto 0 do if Selected[I] then Items.Delete(I); end else if ItemIndex <> -1 then begin I := ItemIndex; Items.Delete(I); if I > 0 then Dec(I); if Items.Count > 0 then ItemIndex := I; end; end; function TJvCheckListBox.GetChecked: TStringList; var I: Integer; begin Result := TStringList.Create; for I := 0 to Items.Count - 1 do if Checked[I] then Result.AddObject(Items[I], Items.Objects[I]); end; function TJvCheckListBox.GetUnChecked: TStringList; var I: Integer; begin Result := TStringList.Create; for I := 0 to Items.Count - 1 do if not Checked[I] then Result.AddObject(Items[I], Items.Objects[I]); end; procedure TJvCheckListBox.InvertCheck; var I: Integer; begin for I := 0 to Items.Count - 1 do Checked[I] := not Checked[I]; end; procedure TJvCheckListBox.InvertSelection; var I: Integer; begin if MultiSelect then for I := 0 to Items.Count - 1 do Selected[I] := not Selected[I]; end; {$IFDEF VCL} procedure TJvCheckListBox.LBNSelCancel(var Msg: TMessage); begin if Assigned(FOnSelectCancel) then FOnSelectCancel(Self); end; procedure TJvCheckListBox.CNDrawItem(var Msg: TWMDrawItem); begin if (Items.Count = 0) or (Msg.DrawItemStruct.itemID >= UINT(Items.Count)) then Exit; {$IFDEF COMPILER5} with Msg.DrawItemStruct {$IFNDEF CLR}^{$ENDIF} do if Header[itemID] then if not UseRightToLeftAlignment then rcItem.Left := rcItem.Left - GetCheckWidth else rcItem.Right := rcItem.Right + GetCheckWidth; {$ENDIF COMPILER5} inherited; end; {$ENDIF VCL} procedure TJvCheckListBox.LoadFromFile(FileName: TFileName); var Stream: TFileStream; begin Stream := TFileStream.Create(FileName, fmOpenRead or fmShareDenyWrite); try LoadFromStream(Stream); finally Stream.Free; end; end; procedure TJvCheckListBox.LoadFromStream(Stream: TStream); var CheckLst: TCheckListRecord; Buf: array [0..1023] of Char; begin Items.Clear; while Stream.Position + SizeOf(TCheckListRecord) <= Stream.Size do begin {$IFDEF CLR} Stream.Read(CheckLst.Checked); Stream.Read(CheckLst.StringSize); {$ELSE} Stream.Read(CheckLst, SizeOf(TCheckListRecord)); {$ENDIF CLR} if (Stream.Position + CheckLst.StringSize <= Stream.Size) and (CheckLst.StringSize < High(Buf)) then begin {$IFDEF CLR} ReadCharsFromStream(Stream, Buf, CheckLst.StringSize); {$ELSE} Stream.Read(Buf, CheckLst.StringSize); {$ENDIF CLR} Buf[CheckLst.StringSize] := #0; Checked[Items.Add(Buf)] := CheckLst.Checked; end; end; end; {$IFDEF VCL} procedure TJvCheckListBox.MouseEnter(Control: TControl); begin if csDesigning in ComponentState then Exit; if not MouseOver then begin if HotTrack then Ctl3D := True; inherited MouseEnter(Control); end; end; procedure TJvCheckListBox.MouseLeave(Control: TControl); begin if MouseOver then begin if HotTrack then Ctl3D := False; inherited MouseLeave(Control); end; end; procedure TJvCheckListBox.RefreshH; var I: Integer; ItemWidth: Word; begin FMaxWidth := 0; for I := 0 to Items.Count - 1 do begin ItemWidth := Canvas.TextWidth(Items[I] + ' '); Inc(ItemWidth, GetCheckWidth); if FMaxWidth < ItemWidth then FMaxWidth := ItemWidth; end; SetHScroll(FScroll); end; {$ENDIF VCL} procedure TJvCheckListBox.SaveToFile(FileName: TFileName); var Stream: TFileStream; begin Stream := TFileStream.Create(FileName, fmCreate or fmShareExclusive); try SaveToStream(Stream); finally Stream.Free; end; end; procedure TJvCheckListBox.SaveToStream(Stream: TStream); var I, J: Integer; CheckLst: TCheckListRecord; Buf: array [1..1023] of Char; begin for I := 0 to Items.Count - 1 do begin CheckLst.Checked := Checked[I]; CheckLst.StringSize := Length(Items[I]); {$IFDEF CLR} Stream.Write(CheckLst.Checked); Stream.Write(CheckLst.StringSize); {$ELSE} Stream.Write(CheckLst, SizeOf(TCheckListRecord)); {$ENDIF CLR} for J := 1 to Length(Items[I]) do Buf[J] := Items[I][J]; {$IFDEF CLR} WriteStringToStream(Stream, Buf, CheckLst.StringSize) {$ELSE} Stream.Write(Buf, CheckLst.StringSize); {$ENDIF CLR} end; end; function TJvCheckListBox.SearchExactString(Value: string; CaseSensitive: Boolean): Integer; begin Result := TJvItemsSearchs.SearchExactString(Items, Value, CaseSensitive); end; function TJvCheckListBox.SearchPrefix(Value: string; CaseSensitive: Boolean): Integer; begin Result := TJvItemsSearchs.SearchPrefix(Items, Value, CaseSensitive); end; function TJvCheckListBox.SearchSubString(Value: string; CaseSensitive: Boolean): Integer; begin Result := TJvItemsSearchs.SearchSubString(Items, Value, CaseSensitive); end; procedure TJvCheckListBox.SelectAll; var I: Integer; begin if MultiSelect then for I := 0 to Items.Count - 1 do Selected[I] := True; end; {$IFDEF VCL} procedure TJvCheckListBox.SetHotTrack(const Value: Boolean); begin FHotTrack := Value; if FHotTrack then Ctl3D := False; end; procedure TJvCheckListBox.SetHScroll(const Value: Boolean); begin FScroll := Value; if FScroll then SendMessage(Handle, LB_SETHORIZONTALEXTENT, FMaxWidth, 0); end; {$ENDIF VCL} procedure TJvCheckListBox.UnCheckAll; var I: Integer; begin for I := 0 to Items.Count - 1 do Checked[I] := False; end; procedure TJvCheckListBox.UnselectAll; var I: Integer; begin if MultiSelect then for I := 0 to Items.Count - 1 do Selected[I] := False; end; {$IFDEF VCL} procedure TJvCheckListBox.WMHScroll(var Msg: TWMHScroll); var ScrollPos: Integer; R: TRect; begin inherited; // (p3) what does this code do, really? if Msg.ScrollCode <> SB_ENDSCROLL then begin ScrollPos := GetScrollPos(Handle, SB_HORZ); if ScrollPos < 20 then begin R := ClientRect; R.Right := R.Left + 20; InvalidateRect(Handle, {$IFNDEF CLR}@{$ENDIF}R, False); end; end; if Assigned(FOnHScroll) then FOnHScroll(Self); end; procedure TJvCheckListBox.WMVScroll(var Msg: TWMVScroll); begin inherited; if Assigned(FOnVScroll) then FOnVScroll(Self); end; procedure TJvCheckListBox.WndProc(var Msg: TMessage); var ItemWidth: Word; begin case Msg.Msg of LB_ADDSTRING, LB_INSERTSTRING: begin {$IFDEF CLR} if Msg.LParam <> 0 then ItemWidth := Canvas.TextWidth(Marshal.PtrToStringAuto(IntPtr(Msg.lParam)) + ' ') else ItemWidth := Canvas.TextWidth(' '); {$ELSE} ItemWidth := Canvas.TextWidth(StrPas(PChar(Msg.lParam)) + ' '); {$ENDIF CLR} Inc(ItemWidth, GetCheckWidth); if FMaxWidth < ItemWidth then FMaxWidth := ItemWidth; SetHScroll(FScroll); end; LB_DELETESTRING: begin ItemWidth := Canvas.TextWidth(Items[Msg.wParam] + ' '); Inc(ItemWidth, GetCheckWidth); if ItemWidth = FMaxWidth then begin inherited WndProc(Msg); RefreshH; Exit; end; end; LB_RESETCONTENT: begin inherited WndProc(Msg); FMaxWidth := 0; SetHScroll(FScroll); Exit; end; WM_SETFONT: begin inherited WndProc(Msg); Canvas.Font.Assign(Font); RefreshH; Exit; end; end; inherited WndProc(Msg); end; {$ENDIF VCL} {$IFDEF UNITVERSIONING} initialization RegisterUnitVersion(HInstance, UnitVersioning); finalization UnregisterUnitVersion(HInstance); {$ENDIF UNITVERSIONING} end.