{----------------------------------------------------------------------------- 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,v 1.24 2005/10/28 07:37:55 obones Exp $ unit JvCheckListBox; {$I jvcl.inc} interface uses {$IFDEF UNITVERSIONING} JclUnitVersioning, {$ENDIF UNITVERSIONING} {$IFDEF CLR} System.Runtime.InteropServices, {$ENDIF CLR} Windows, Messages, SysUtils, Classes, Controls, Graphics, JvExCheckLst; type 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); protected procedure DrawItem(Index: Integer; Rect: TRect; State: TOwnerDrawState); override; 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; destructor Destroy; override; {$ENDIF COMPILER5} public constructor Create(AOwner: TComponent); 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 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} end; {$IFDEF UNITVERSIONING} const UnitVersioning: TUnitVersionInfo = ( RCSfile: '$RCSfile: JvCheckListBox.pas,v $'; Revision: '$Revision: 1.24 $'; Date: '$Date: 2005/10/28 07:37:55 $'; LogPath: 'JVCL\run' ); {$ENDIF UNITVERSIONING} implementation uses JvItemsSearchs, JvJCLUtils; type // Used for the load/save methods TCheckListRecord = record Checked: Boolean; StringSize: Integer; end; //=== { TJvCheckListBox } ==================================================== constructor TJvCheckListBox.Create(AOwner: TComponent); begin inherited Create(AOwner); {$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; {$IFDEF COMPILER5} destructor TJvCheckListBox.Destroy; begin FHeaders.Free; inherited Destroy; end; 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; procedure TJvCheckListBox.DrawItem(Index: Integer; Rect: TRect; State: TOwnerDrawState); begin if Header[Index] then begin Canvas.Font.Color := HeaderColor; Canvas.Brush.Color := HeaderBackgroundColor; end; inherited DrawItem(Index, Rect, State); end; {$ENDIF COMPILER5} 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] + ' '); 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} if FMaxWidth < ItemWidth then FMaxWidth := ItemWidth; SetHScroll(FScroll); end; LB_DELETESTRING: begin ItemWidth := Canvas.TextWidth(Items[Msg.wParam] + ' '); if ItemWidth = FMaxWidth then begin inherited WndProc(Msg); RefreshH; Exit; end; end; LB_RESETCONTENT: begin FMaxWidth := 0; SetHScroll(FScroll); 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.