Componentes.Terceros.jvcl/official/3.00/run/JvCheckListBox.pas

594 lines
15 KiB
ObjectPascal

{-----------------------------------------------------------------------------
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.