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

1461 lines
42 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: JvStringGrid.PAS, released on 2001-02-28.
The Initial Developer of the Original Code is S?stien Buysse [sbuysse att buypin dott com]
Portions created by S?stien Buysse are Copyright (C) 2001 S?stien Buysse.
All Rights Reserved.
Contributor(s): Michael Beck [mbeck att bigfoot 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: JvStringGrid.pas,v 1.51 2006/02/06 14:50:43 obones Exp $
unit JvStringGrid;
{$I jvcl.inc}
//---------------------------------------------------------------
// The inplace-edit-list feature is enabled dynamically when
// compiling JVCL, if the underlying JVCL and VCL base classes
// support it. Look for COMPILER6_UP below for places where this
// requires us to conditionally define this feature in or out.
//---------------------------------------------------------------
interface
uses
{$IFDEF UNITVERSIONING}
JclUnitVersioning,
{$ENDIF UNITVERSIONING}
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Grids, StdCtrls,
JvTypes, JvJCLUtils, JvExGrids;
const
GM_ACTIVATECELL = WM_USER + 123;
type
// (rom) renamed elements made packed
TGMActivateCell = packed record
Msg: Cardinal;
Column: Integer;
Row: Integer;
Result: Integer;
end;
TJvStringGrid = class;
TExitCellEvent = procedure(Sender: TJvStringGrid; AColumn, ARow: Integer;
const EditText: string) of object;
TGetCellAlignmentEvent = procedure(Sender: TJvStringGrid; AColumn, ARow: Integer;
State: TGridDrawState; var CellAlignment: TAlignment) of object;
TCaptionClickEvent = procedure(Sender: TJvStringGrid; AColumn, ARow: Integer) of object;
TEditShowEvent = procedure(Sender: TJvStringGrid; ACol, ARow: Longint;
var AllowEdit: Boolean) of object;
TJvSortType = (stNone, stAutomatic, stClassic, stCaseSensitive, stNumeric, stDate, stCurrency);
TProgress = procedure(Sender: TObject; Progression, Total: Integer) of object;
{$IFDEF COMPILER6_UP}
TJvOnGetEditStyleEvent = procedure(Sender: TJvStringGrid; AColumn, ARow: Integer; PickListStrings: TStrings; var EditStyle: TEditStyle) of object;
{$ENDIF COMPILER6_UP}
TJvStringGrid = class(TJvExStringGrid)
private
FAlignment: TAlignment;
FSetCanvasProperties: TDrawCellEvent;
FGetCellAlignment: TGetCellAlignmentEvent;
FCaptionClick: TCaptionClickEvent;
FCellOnMouseDown: TGridCoord;
FOnExitCell: TExitCellEvent;
FOnLoadProgress: TProgress;
FOnSaveProgress: TProgress;
FOnHorizontalScroll: TNotifyEvent;
FOnVerticalScroll: TNotifyEvent;
FOnShowEditor: TEditShowEvent;
{$IFDEF COMPILER6_UP}
FCustomInplaceEditStyle: TEditStyle; // NEW
FOnGetEditStyle: TJvOnGetEditStyleEvent;
FPickListStrings: TStringList;
FOnEditButtonClick: TNotifyEvent;
FOnListBoxCloseUp: TNotifyEvent;
{$ENDIF COMPILER6_UP}
FFixedFont: TFont;
procedure SetAlignment(const Value: TAlignment);
procedure GMActivateCell(var Msg: TGMActivateCell); message GM_ACTIVATECELL;
{$IFDEF VCL}
procedure WMCommand(var Msg: TWMCommand); message WM_COMMAND;
{$ENDIF VCL}
procedure SetFixedFont(const Value: TFont);
procedure DoFixedFontChange(Sender: TObject);
{$IFDEF COMPILER6_UP}
procedure EditButtonClick(Sender: TObject); dynamic; // NEW-WP
procedure ListBoxCloseUp(Sender: TObject); dynamic;
{$ENDIF COMPILER6_UP}
protected
function CreateEditor: TInplaceEdit; override;
function CanEditShow: Boolean; override;
procedure MouseDown(Button: TMouseButton; Shift: TShiftState;
X, Y: Integer); override;
procedure MouseUp(Button: TMouseButton; Shift: TShiftState;
X, Y: Integer); override;
procedure ExitCell(const EditText: string; AColumn, ARow: Integer); virtual;
procedure SetCanvasProperties(AColumn, ARow: Longint;
Rect: TRect; State: TGridDrawState); virtual;
procedure DrawCell(AColumn, ARow: Longint;
Rect: TRect; State: TGridDrawState); override;
{$IFDEF COMPILER6_UP}
// NEW: Override to provide dropdown list editing as an
// event-handler in TJvStringGrid.
function GetEditStyle(ACol, ARow: Longint): TEditStyle; override;
{$ENDIF COMPILER6_UP}
procedure CaptionClick(AColumn, ARow: Longint); dynamic;
{$IFDEF VCL}
procedure WMHScroll(var Msg: TWMHScroll); message WM_HSCROLL;
procedure WMVScroll(var Msg: TWMVScroll); message WM_VSCROLL;
{$ENDIF VCL}
{$IFDEF VisualCLX}
procedure ModifyScrollBar(ScrollBar: TScrollBarKind; ScrollCode: TScrollCode;
Pos: Cardinal; UseRightToLeft: Boolean); override;
function SelectCell(ACol, ARow: Longint): Boolean; override;
{$ENDIF VisualCLX}
procedure DoLoadProgress(Position, Count: Integer);
procedure DoSaveProgress(Position, Count: Integer);
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
function GetCellAlignment(AColumn, ARow: Longint;
State: TGridDrawState): TAlignment; virtual;
procedure DefaultDrawCell(AColumn, ARow: Longint;
Rect: TRect; State: TGridDrawState); virtual;
procedure ActivateCell(AColumn, ARow: Integer);
procedure InvalidateCell(AColumn, ARow: Integer);
procedure InvalidateCol(AColumn: Integer);
procedure InvalidateRow(ARow: Integer);
procedure MoveColumn(FromIndex, ToIndex: Integer);
procedure MoveRow(FromIndex, ToIndex: Longint);
property InplaceEditor;
// Calculates and sets the width of a specific column or all columns if Index < 0
// based on the text in the affected Cells.
// MinWidth is the minimum width of the column(s). If MinWidth is < 0,
// DefaultColWidth is used instead
procedure AutoSizeCol(Index, MinWidth: Integer);
// Inserts a new row at the specified Index and moves all existing rows >= Index down one step
// Returns the inserted row as an empty TStrings
function InsertRow(Index: Integer): TStrings;
// Inserts a new column at the specified Index and moves all existing columns >= Index to the right
// Returns the inserted column as an empty TStrings
function InsertCol(Index: Integer): TStrings;
// Removes the row at Index and moves all rows > Index up one step
procedure RemoveRow(Index: Integer);
// Removes the column at Index and moves all cols > Index to the left
procedure RemoveCol(Index: Integer);
// Hides the row at Index by setting it's height = -1
// Calling this method repeatedly does nothing (the row retains it's Index even if it's hidden)
procedure HideRow(Index: Integer);
// Shows the row at Index by setting it's height to AHeight
// if AHeight <= 0, DefaultRowHeight is used instead
procedure ShowRow(Index, AHeight: Integer);
// Hides the column at Index by setting it's ColWidth = -1
// Calling this method repeatedly does nothing (the column retains it's Index even if it's hidden)
procedure HideCol(Index: Integer);
// Returns True if the Cell at ACol/ARow is hidden, i.e if it's RowHeight or ColWidth < 0
function IsHidden(ACol, ARow: Integer): Boolean;
// Shows the column at Index by setting it's width to AWidth
// If AWidth <= 0, DefaultColWidth is used instead
procedure ShowCol(Index, AWidth: Integer);
// HideCell hides a cell by hiding the row and column that it belongs to.
// This means that both a row and a column is hidden
procedure HideCell(ACol, ARow: Integer);
// ShowCell shows a previously hidden cell by showing it's corresponding row and column and
// using AWidth/AHeight to set it's size. If AWidth < 0, DefaultColWidth is used instead.
// If AHeight < 0, DefaultRowHeight is used instead. If one dimension of the Cell wasn't
// hidden, nothing happens to that dimension (i.e if ColWidth < 0 but RowHeight := 24, only ColWidth is
// changed to AWidth
procedure ShowCell(ACol, ARow, AWidth, AHeight: Integer);
// Hides all rows and columns
procedure HideAll;
// Shows all hidden rows and columns, setting their width/height to AWidth/AHeight as necessary
// If AWidth < 0, DefaultColWidth is used. If AHeight < 0, DefaultRowHeight is used
procedure ShowAll(AWidth, AHeight: Integer);
// Removes the content in the Cells but does not remove any rows or columns
procedure Clear;
// Clears selection rectangle!
procedure ClearSelection;
procedure SortGrid(Column: Integer; Ascending: Boolean = True; Fixed: Boolean = False;
SortType: TJvSortType = stClassic; BlankTop: Boolean = True);
// Sort grid using the column indices in ColOrder. For example if ColOrder contains
// [1, 3, 0, 2], column 3 is used when the items in column 1 are identical
procedure SortGridByCols(ColOrder: array of Integer; Fixed: Boolean = False);
procedure SaveToFile(FileName: string);
procedure LoadFromFile(FileName: string);
procedure LoadFromCSV(FileName: string; Separator: Char = ';'; QuoteChar: Char = '"'; StripQuotes: Boolean = True);
procedure SaveToCSV(FileName: string; Separator: Char = ';'; QuoteChar: Char = '"');
procedure LoadFromStream(Stream: TStream);
procedure SaveToStream(Stream: TStream);
published
property HintColor;
property Alignment: TAlignment read FAlignment write SetAlignment;
property FixedFont: TFont read FFixedFont write SetFixedFont;
property OnExitCell: TExitCellEvent read FOnExitCell write FOnExitCell;
property OnSetCanvasProperties: TDrawCellEvent read FSetCanvasProperties write FSetCanvasProperties;
property OnGetCellAlignment: TGetCellAlignmentEvent read FGetCellAlignment write FGetCellAlignment;
property OnCaptionClick: TCaptionClickEvent read FCaptionClick write FCaptionClick;
property OnMouseEnter;
property OnMouseLeave;
property OnParentColorChange;
property OnLoadProgress: TProgress read FOnLoadProgress write FOnLoadProgress;
property OnSaveProgress: TProgress read FOnSaveProgress write FOnSaveProgress;
property OnVerticalScroll: TNotifyEvent read FOnVerticalScroll write FOnVerticalScroll;
property OnHorizontalScroll: TNotifyEvent read FOnHorizontalScroll write FOnHorizontalScroll;
property OnShowEditor: TEditShowEvent read FOnShowEditor write FOnShowEditor;
{$IFDEF COMPILER6_UP}
property OnGetEditStyle: TJvOnGetEditStyleEvent read FOnGetEditStyle write FOnGetEditStyle; // NEW -WP (D6 UP)
property OnEditButtonClick: TNotifyEvent read FOnEditButtonClick write FOnEditButtonClick; // NEW-WP - User clicks on Ellipsis button, get event fired!
property OnListBoxCloseUp: TNotifyEvent read FOnListBoxCloseUp write FOnListBoxCloseUp; // Listbox close up.
{$ENDIF COMPILER6_UP}
end;
{$IFDEF UNITVERSIONING}
const
UnitVersioning: TUnitVersionInfo = (
RCSfile: '$RCSfile: JvStringGrid.pas,v $';
Revision: '$Revision: 1.51 $';
Date: '$Date: 2006/02/06 14:50:43 $';
LogPath: 'JVCL\run'
);
{$ENDIF UNITVERSIONING}
implementation
uses
Math,
JvVCL5Utils, JvJVCLUtils;
const
BufSize = 1024;
//=== { TExInplaceEditList } =================================================
// If the feature exists in the VCL base classes, we can enable the
// feature here.
type
{$IFDEF COMPILER6_UP}
TExInplaceEditList = class(TJvExPubInplaceEditList) // was inheriting from TJvExInplaceEdit.-WAP
{$ELSE}
TExInplaceEditList = class(TJvExInplaceEdit)
{$ENDIF COMPILER6_UP}
private
// Important: Style of this inplace editor is set in TInplaceEditList
// FEditStyle - inherited - See VCL Source: ($delphi)\Source\vcl\Grids.pas
//FActiveList : TWinControl; // WP-New: Listbox control stuff
//FPickListLoaded: Boolean; // WP-New
//FPickList : TCustomListbox; // WP-New
//FListVisible : Boolean; // WP-New
FLastCol: Integer;
FLastRow: Integer;
protected
{$IFDEF COMPILER6_UP}
procedure CloseUp(Accept: Boolean); override; // fire event on close up!
procedure DoEditButtonClick; override;
procedure UpdateContents; override; //WP-New! - Put items into listbox!
{$ENDIF COMPILER6_UP}
procedure FocusKilled(NextWnd: THandle); override;
procedure FocusSet(PrevWnd: THandle); override;
public
constructor Create(Owner: TComponent); override;//WP-New!
//property ActiveList: TWinControl read FActiveList write FActiveList;//WP-New!
{$IFDEF VCL}
procedure CreateParams(var Params: TCreateParams); override;
{$ENDIF VCL}
end;
constructor TExInplaceEditList.Create(Owner: TComponent);
begin
inherited Create(Owner);
// todo: tweakage!
end;
//NEW!
{$IFDEF COMPILER6_UP}
procedure TExInplaceEditList.UpdateContents;
var
OwnerGrid: TJvStringGrid;
begin
inherited UpdateContents;
if EditStyle = esPickList then
begin
ActiveList := PickList;
// Populate the listbox:
Assert(Assigned(Grid));
OwnerGrid := (Grid as TJvStringGrid);
PickList.Items.Assign(OwnerGrid.FPickListStrings);
end;
end;
{$ENDIF COMPILER6_UP}
{$IFDEF VCL}
procedure TExInplaceEditList.CreateParams(var Params: TCreateParams);
const
Flags: array [TAlignment] of DWORD = (ES_LEFT, ES_RIGHT, ES_CENTER);
begin
inherited CreateParams(Params);
Params.Style := Params.Style or Flags[TJvStringGrid(Grid).Alignment];
end;
{$ENDIF VCL}
procedure TExInplaceEditList.FocusKilled(NextWnd: THandle);
begin
TJvStringGrid(Grid).ExitCell(Text, FLastCol, FLastRow);
inherited FocusKilled(NextWnd);
end;
procedure TExInplaceEditList.FocusSet(PrevWnd: THandle);
begin
FLastCol := TJvStringGrid(Grid).Col;
FLastRow := TJvStringGrid(Grid).Row;
inherited FocusSet(PrevWnd);
end;
{$IFDEF COMPILER6_UP}
procedure TExInplaceEditList.CloseUp(Accept: Boolean); //override; // fire event on close up!
begin
inherited CloseUp(Accept);
if Assigned(Grid) then
TJvStringGrid(Grid).ListBoxCloseUp(Self);
end;
procedure TExInplaceEditList.DoEditButtonClick;
begin
if Assigned(Grid) then
TJvStringGrid(Grid).EditButtonClick(Self);
end;
{$ENDIF COMPILER6_UP}
// ahuser: TExPopupListBox is not used anywhere
(*
//=== { TExPopupListBox } ====================================================
type
// TExPopupListBox - a popup listbox has no parent.
TExPopupListBox = class(TCustomListBox)
private
FSearchText: string;
FSearchTickCount: Longint;
protected
procedure CreateParams(var Params: TCreateParams); override;
procedure CreateWnd; override;
procedure KeyPress(var Key: Char); override;
{$IFDEF COMPILER6_UP}
procedure MouseUp(Button: TMouseButton; Shift: TShiftState; X, Y: Integer); override;
{$ENDIF COMPILER6_UP}
end;
procedure TExPopupListBox.CreateParams(var Params: TCreateParams);
begin
inherited CreateParams(Params);
with Params do
begin
Style := Style or WS_BORDER;
ExStyle := WS_EX_TOOLWINDOW or WS_EX_TOPMOST;
AddBiDiModeExStyle(ExStyle);
WindowClass.Style := CS_SAVEBITS;
end;
end;
procedure TExPopupListBox.CreateWnd;
begin
inherited CreateWnd;
Windows.SetParent(Handle, 0);
CallWindowProc(DefWndProc, Handle, wm_SetFocus, 0, 0);
end;
procedure TExPopupListBox.Keypress(var Key: Char);
var
TickCount: Integer;
begin
case Key of
#8, #27:
FSearchText := '';
#32..#255:
begin
TickCount := GetTickCount;
if TickCount - FSearchTickCount > 2000 then
FSearchText := '';
FSearchTickCount := TickCount;
if Length(FSearchText) < 32 then
FSearchText := FSearchText + Key;
SendMessage(Handle, LB_SelectString, WPARAM(-1), LPARAM(PChar(FSearchText)));
Key := #0;
end;
end;
inherited Keypress(Key);
end;
{$IFDEF COMPILER6_UP}
procedure TExPopupListBox.MouseUp(Button: TMouseButton; Shift: TShiftState;
X, Y: Integer);
var
Accept: Boolean;
begin
inherited MouseUp(Button, Shift, X, Y);
Accept := (X >= 0) and (Y >= 0) and (X < Width) and (Y < Height);
TExInplaceEditList(Owner).CloseUp(Accept);
end;
{$ENDIF COMPILER6_UP}
*)
//=== { TJvStringGrid } ======================================================
constructor TJvStringGrid.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
FFixedFont := TFont.Create;
FFixedFont.Assign(Font);
FFixedFont.OnChange := DoFixedFontChange;
// ControlStyle := ControlStyle + [csAcceptsControls];
{$IFDEF COMPILER6_UP}
FPickListStrings := TStringList.Create; //NEW -WP
{$ENDIF COMPILER6_UP}
end;
destructor TJvStringGrid.Destroy;
begin
FreeAndNil(FFixedFont);
{$IFDEF COMPILER6_UP}
FreeAndNil(FPickListStrings); //NEW-WP
{$ENDIF COMPILER6_UP}
inherited Destroy;
end;
function TJvStringGrid.CanEditShow: Boolean;
begin
Result := inherited CanEditShow;
if Result and Assigned(FOnShowEditor) then
begin
FOnShowEditor(Self, Col, Row, Result);
if not Result then
EditorMode := False;
end;
end;
procedure TJvStringGrid.SortGrid(Column: Integer;
Ascending, Fixed: Boolean; SortType: TJvSortType; BlankTop: Boolean);
const
cFloatDelta = 0.01;
var
St: string;
TmpC: Currency;
TmpF: Extended;
TmpD: TDateTime;
LStart: Integer;
LEnd: Integer;
procedure ExchangeGridRows(I, J: Integer);
var
K: Integer;
begin
if Fixed then
for K := 0 to ColCount - 1 do
Cols[K].Exchange(I, J)
else
for K := FixedCols to ColCount - 1 do
Cols[K].Exchange(I, J);
end;
function IsSmaller(First, Second: string): Boolean;
function DetectType(const S1, S2: string): TJvSortType;
var
ExtValue: Extended;
CurrValue: Currency;
DateValue: TDateTime;
begin
if TextToFloat(PChar(S1), ExtValue, fvExtended) and TextToFloat(PChar(S2), ExtValue, fvExtended) then
Result := stNumeric
else
if TextToFloat(PChar(S1), CurrValue, fvCurrency) and TextToFloat(PChar(S2), CurrValue, fvCurrency) then
Result := stCurrency
else
if TryStrToDateTime(S1, DateValue) and TryStrToDateTime(S2, DateValue) then
Result := stDate
else
Result := stClassic;
end;
begin
case DetectType(First, Second) of
stNumeric:
Result := StrToFloat(First) < StrToFloat(Second);
stCurrency:
Result := StrToCurr(First) < StrToCurr(Second);
stDate:
Result := StrToDateTime(First) < StrToDateTime(Second);
stClassic:
Result := AnsiCompareText(First, Second) < 0;
else
Result := First > Second;
end;
end;
function IsBigger(First, Second: string): Boolean;
begin
Result := IsSmaller(Second, First);
end;
// (rom) A HeapSort has no worst case for O(X)
// (rom) I donated one a long time ago to JCL
// (p3) maybe implemented a secondary sort index when items are equal?
// (p3) ...or use another stable sort method, like heapsort
procedure QuickSort(L, R: Integer);
var
I, J, m: Integer;
begin
repeat
I := L;
J := R;
m := (L + R) div 2;
St := Cells[Column, m];
repeat
case SortType of
stClassic:
begin
while AnsiCompareText(Cells[Column, I], St) < 0 do
Inc(I);
while AnsiCompareText(Cells[Column, J], St) > 0 do
Dec(J);
end;
stCaseSensitive:
begin
while AnsiCompareStr(Cells[Column, I], St) < 0 do
Inc(I);
while AnsiCompareStr(Cells[Column, J], St) > 0 do
Dec(J);
end;
stNumeric:
begin
TmpF := StrToFloatDef(St, 0);
while StrToFloatDef(Cells[Column, I], 0) < TmpF do
Inc(I);
while StrToFloatDef(Cells[Column, J], 0) > TmpF do
Dec(J);
end;
stDate:
begin
TmpD := StrToDateTimeDef(St, 0);
while StrToDateTimeDef(Cells[Column, I], 0) < TmpD do
Inc(I);
while StrToDateTimeDef(Cells[Column, J], 0) > TmpD do
Dec(J);
end;
stCurrency:
begin
TmpC := StrToCurrDef(St, 0);
while StrToCurrDef(Cells[Column, I], 0) < TmpC do
Inc(I);
while StrToCurrDef(Cells[Column, J], 0) > TmpC do
Dec(J);
end;
stAutomatic:
begin
while IsSmaller(Cells[Column, I], St) do
Inc(I);
while IsBigger(Cells[Column, J], St) do
Dec(J);
end;
end;
if I <= J then
begin
if I <> J then
ExchangeGridRows(I, J);
Inc(I);
Dec(J);
end;
until (I > J);
if L < J then
QuickSort(L, J);
L := I;
until I >= R;
end;
procedure InvertGrid;
var
I, J: Integer;
begin
I := FixedRows;
J := RowCount - 1;
while I < J do
begin
ExchangeGridRows(I, J);
Inc(I);
Dec(J);
end;
end;
function MoveBlankTop: Integer;
var
I, J: Integer;
begin
I := FixedRows;
Result := I;
J := RowCount - 1;
while I <= J do
begin
if Trim(Cells[Column, I]) = '' then
begin
ExchangeGridRows(Result, I);
Inc(Result);
end;
Inc(I);
end;
end;
procedure MoveBlankBottom;
var
I, J: Integer;
DoSort: Boolean;
begin
I := FixedRows;
DoSort := False;
// avoid empty columns
for J := I to RowCount - 1 do
if Cells[Column, J] <> '' then
begin
DoSort := True;
Break;
end;
if not DoSort then
Exit;
// this is already sorted, so blank items should be at top
while Trim(Cells[Column, I]) = '' do
begin
InsertRow(RowCount).Assign(Rows[I]);
DeleteRow(I);
Inc(J);
if J >= RowCount then
Exit;
end;
end;
begin
// (p3) NB!! sorting might trigger the OnExitCell, OnGetEditText and OnSetEditText events!
// make sure you don't do anything in these events
if (Column >= 0) and (Column < ColCount) and (SortType <> stNone) then
begin
LStart := FixedRows;
LEnd := RowCount - 1;
if BlankTop then
LStart := MoveBlankTop;
if LStart < LEnd then
begin
QuickSort(LStart, LEnd);
if not BlankTop then
MoveBlankBottom;
if not Ascending then
InvertGrid;
end;
end;
end;
procedure TJvStringGrid.LoadFromFile(FileName: string);
var
Stream: TFileStream;
begin
Stream := TFileStream.Create(FileName, fmOpenRead or fmShareDenyWrite);
// (rom) secured
try
LoadFromStream(Stream);
finally
Stream.Free;
end;
end;
procedure TJvStringGrid.LoadFromCSV(FileName: string; Separator: Char = ';'; QuoteChar: Char = '"'; StripQuotes: Boolean = True);
var
I: Longint;
Lines, Fields: TStringList;
procedure SplitLine(const Line: string; Result: TStrings; Delimiter, QuoteChar: Char; StripQuotes: Boolean);
var
I, SLen, QuoteCount: Integer;
S: string;
IgnoreDelim: Boolean;
QuotedStr: PChar;
begin
S := '';
SLen := Length(Line);
IgnoreDelim := False;
QuoteCount := 0;
Result.Clear;
for I := 1 to SLen do
begin
if Line[I] = QuoteChar then
begin
Inc(QuoteCount);
{* A Delimiter surrounded by a pair of QuoteChar has to be ignored.
See example above: "FirstName, LastName"
therefor: *}
IgnoreDelim := QuoteCount mod 2 <> 0;
end;
if IgnoreDelim then
S := S + Line[I]
else
if Line[I] <> Delimiter then
S := S + Line[I]
else
begin
if S <> '' then
begin
if StripQuotes and (S[1] = QuoteChar) then
begin
QuotedStr := PChar(S);
Result.Add(AnsiExtractQuotedStr(QuotedStr, QuoteChar));
end
else
Result.Add(S);
end
else
Result.Add(S);
S := '';
end;
end;
if S <> '' then
begin
if StripQuotes and (S[1] = QuoteChar) then
begin
QuotedStr := PChar(S);
Result.Add(AnsiExtractQuotedStr(QuotedStr, QuoteChar));
end
else
Result.Add(S);
end
else
Result.Add(S);
end;
begin
Lines := TStringList.Create;
Fields := TStringList.Create;
try
Lines.LoadFromFile(FileName);
DoLoadProgress(0, Lines.Count);
RowCount := Lines.Count;
ColCount := FixedCols + 1;
for I := 0 to Lines.Count - 1 do
begin
{* added John *}
SplitLine(Lines[I], Fields, Separator, QuoteChar, StripQuotes);
DoLoadProgress(I, Lines.Count);
if Fields.Count > ColCount then
ColCount := Fields.Count;
Rows[I].Assign(Fields);
end;
DoLoadProgress(Lines.Count, Lines.Count);
finally
Fields.Free;
Lines.Free;
end;
end;
procedure TJvStringGrid.LoadFromStream(Stream: TStream);
var
Col, Row, I, Count: Integer;
Buffer: array [0..BufSize - 1] of Byte;
St: string;
begin
Col := 0;
Row := 1;
DoLoadProgress(0, Stream.Size);
while Stream.Position < Stream.Size do
begin
Count := Stream.Read(Buffer, 1024);
DoLoadProgress(Stream.Position, Stream.Size);
for I := 0 to Count - 1 do
case Buffer[I] of
0:
begin
Inc(Col);
if Row > RowCount then
RowCount := Row;
if Col > ColCount then
ColCount := Col;
Cells[Col - 1, Row - 1] := St;
St := '';
end;
1:
begin
Inc(Col);
if Col > ColCount then
ColCount := Col;
Cells[Col - 1, Row - 1] := St;
Inc(Row);
if Row > RowCount then
RowCount := Row;
Col := 0;
St := '';
end;
else
St := St + Char(Buffer[I]);
end;
end;
RowCount := RowCount - 1;
DoLoadProgress(Stream.Size, Stream.Size);
end;
{$IFDEF VCL}
procedure TJvStringGrid.WMHScroll(var Msg: TWMHScroll);
begin
inherited;
if Assigned(FOnHorizontalScroll) then
FOnHorizontalScroll(Self);
end;
procedure TJvStringGrid.WMVScroll(var Msg: TWMVScroll);
begin
inherited;
if Assigned(FOnVerticalScroll) then
FOnVerticalScroll(Self);
end;
{$ENDIF VCL}
{$IFDEF VisualCLX}
procedure TJvStringGrid.ModifyScrollBar(ScrollBar: TScrollBarKind; ScrollCode: TScrollCode;
Pos: Cardinal; UseRightToLeft: Boolean);
begin
case ScrollBar of
sbHorizontal:
if Assigned(FOnHorizontalScroll) then
FOnHorizontalScroll(Self);
sbVertical:
if Assigned(FOnVerticalScroll) then
FOnVerticalScroll(Self);
end;
end;
{$ENDIF VisualCLX}
procedure TJvStringGrid.SaveToFile(FileName: string);
var
Stream: TFileStream;
begin
Stream := TFileStream.Create(FileName, fmCreate or fmShareDenyWrite);
try
SaveToStream(Stream);
finally
Stream.Free;
end;
end;
procedure TJvStringGrid.SaveToCSV(FileName: string; Separator: Char = ';'; QuoteChar: Char = '"');
var
I, J: Longint;
BufStr, Value: string;
Lines: TStringList;
begin
Lines := TStringList.Create;
DoSaveProgress(0, RowCount);
try
Lines.Clear;
for I := 0 to RowCount - 1 do
begin
BufStr := '';
DoSaveProgress(I, RowCount);
for J := 0 to ColCount - 1 do
begin
{* added John *}
Value := Cells[J, I];
if Pos(Separator, Value) > 0 then
Value := AnsiQuotedStr(Value, QuoteChar);
{* end added John *}
BufStr := BufStr + Value;
if J <> (ColCount - 1) then
BufStr := BufStr + Separator;
end;
Lines.Add(BufStr);
end;
DoSaveProgress(RowCount, RowCount);
Lines.SaveToFile(FileName);
finally
Lines.Free;
end;
end;
procedure TJvStringGrid.SaveToStream(Stream: TStream);
var
I, J, K, ATotal: Integer;
St: array [0..BufSize - 1] of Char;
Stt: string;
A, B: Byte;
begin
A := 0;
B := 1; // A for end of string, B for end of line
ATotal := RowCount * ColCount;
DoSaveProgress(0, ATotal);
for I := 0 to RowCount - 1 do
begin
for J := 0 to ColCount - 1 do
begin
DoSaveProgress(I * ColCount + J, ATotal);
Stt := Cells[J, I];
for K := 1 to Length(Stt) do
St[K - 1] := Stt[K];
Stream.Write(St, Length(Cells[J, I]));
if J <> ColCount - 1 then
Stream.Write(A, 1);
end;
Stream.Write(B, 1);
end;
DoSaveProgress(ATotal, ATotal);
end;
procedure TJvStringGrid.ActivateCell(AColumn, ARow: Integer);
begin
PostMessage(Handle, GM_ACTIVATECELL, AColumn, ARow);
end;
procedure TJvStringGrid.CaptionClick(AColumn, ARow: Integer);
begin
if Assigned(FCaptionClick) then
FCaptionClick(Self, AColumn, ARow);
end;
function TJvStringGrid.CreateEditor: TInplaceEdit;
begin
Result := TExInplaceEditList.Create(Self);
end;
procedure TJvStringGrid.DefaultDrawCell(AColumn, ARow: Integer; Rect: TRect;
State: TGridDrawState);
const
Flags: array [TAlignment] of DWORD = (DT_LEFT, DT_RIGHT, DT_CENTER);
var
S: string;
begin
if RowHeights[ARow] < CanvasMaxTextHeight(Canvas) then
Exit;
Canvas.FillRect(Rect);
S := Cells[AColumn, ARow];
if Length(S) > 0 then
begin
InflateRect(Rect, -2, -2);
DrawText(Canvas, S, Length(S), Rect,
DT_SINGLELINE or DT_NOPREFIX or DT_VCENTER or
Flags[GetCellAlignment(AColumn, ARow, State)]);
end;
end;
{$IFDEF COMPILER6_UP}
function TJvStringGrid.GetEditStyle(ACol, ARow: Longint): TEditStyle; //override;
begin
FCustomInplaceEditStyle := esSimple;
if Assigned(FOnGetEditStyle) then
begin
FPickListStrings.Clear;
FOnGetEditStyle(Self, ACol, ARow, FPickListStrings, FCustomInplaceEditStyle);
end;
Result := FCustomInplaceEditStyle;
end;
{$ENDIF COMPILER6_UP}
procedure TJvStringGrid.DrawCell(AColumn, ARow: Integer; Rect: TRect; State: TGridDrawState);
begin
if (AColumn < FixedCols) or (ARow < FixedRows) then
Canvas.Font := FixedFont;
if Assigned(OnDrawCell) then
inherited DrawCell(AColumn, ARow, Rect, State)
else
begin
SetCanvasProperties(AColumn, ARow, Rect, State);
DefaultDrawCell(AColumn, ARow, Rect, State);
Canvas.Font := Font;
Canvas.Brush := Brush;
end;
end;
procedure TJvStringGrid.ExitCell(const EditText: string;
AColumn, ARow: Integer);
begin
if Assigned(FOnExitCell) then
FOnExitCell(Self, AColumn, ARow, EditText);
end;
function TJvStringGrid.GetCellAlignment(AColumn, ARow: Integer;
State: TGridDrawState): TAlignment;
begin
Result := FAlignment;
if Assigned(FGetCellAlignment) then
FGetCellAlignment(Self, AColumn, ARow, State, Result);
end;
{$IFDEF VisualCLX}
function TJvStringGrid.SelectCell(ACol, ARow: Longint): Boolean;
begin
Result := inherited SelectCell(ACol, ARow);
if Result then
begin
Col := ACol;
Row := ARow;
EditorMode := True;
InplaceEditor.SelectAll;
end;
end;
{$ENDIF VisualCLX}
procedure TJvStringGrid.GMActivateCell(var Msg: TGMActivateCell);
begin
Col := Msg.Column;
Row := Msg.Row;
EditorMode := True;
InplaceEditor.SelectAll;
end;
procedure TJvStringGrid.InvalidateCell(AColumn, ARow: Integer);
begin
inherited InvalidateCell(AColumn, ARow);
end;
procedure TJvStringGrid.InvalidateCol(AColumn: Integer);
begin
inherited InvalidateCol(AColumn);
end;
procedure TJvStringGrid.InvalidateRow(ARow: Integer);
begin
inherited InvalidateRow(ARow);
end;
procedure TJvStringGrid.MouseDown(Button: TMouseButton; Shift: TShiftState;
X, Y: Integer);
begin
inherited MouseDown(Button, Shift, X, Y);
if Button = mbLeft then
MouseToCell(X, Y, FCellOnMouseDown.X, FCellOnMouseDown.Y)
else
FCellOnMouseDown := TGridCoord(Point(-1, -1));
end;
procedure TJvStringGrid.MouseUp(Button: TMouseButton; Shift: TShiftState;
X, Y: Integer);
var
Cell: TGridCoord;
begin
if Button = mbLeft then
MouseToCell(X, Y, Cell.X, Cell.Y);
if CompareMem(@Cell, @FCellOnMouseDown, SizeOf(Cell)) and
((Cell.X < FixedCols) or (Cell.Y < FixedRows)) then
CaptionClick(Cell.X, Cell.Y);
FCellOnMouseDown := TGridCoord(Point(-1, -1));
inherited MouseUp(Button, Shift, X, Y);
end;
procedure TJvStringGrid.SetAlignment(const Value: TAlignment);
begin
if FAlignment <> Value then
begin
FAlignment := Value;
Invalidate;
if Assigned(InplaceEditor) then
{$IFDEF VCL}
TExInplaceEditList(InplaceEditor).RecreateWnd;
{$ENDIF VCL}
{$IFDEF VisualCLX}
TExInplaceEditList(InplaceEditor).RecreateWidget;
{$ENDIF VisualCLX}
end;
end;
procedure TJvStringGrid.SetCanvasProperties(AColumn, ARow: Integer;
Rect: TRect; State: TGridDrawState);
begin
if Assigned(FSetCanvasProperties) then
FSetCanvasProperties(Self, AColumn, ARow, Rect, State);
end;
{$IFDEF VCL}
procedure TJvStringGrid.WMCommand(var Msg: TWMCommand);
begin
if EditorMode and (Msg.Ctl = InplaceEditor.Handle) then
inherited
else
if Msg.Ctl <> 0 then
Msg.Result := SendMessage(Msg.Ctl, CN_COMMAND, TMessage(Msg).wParam, TMessage(Msg).lParam);
end;
{$ENDIF VCL}
function TJvStringGrid.InsertCol(Index: Integer): TStrings;
var
I: Integer;
AStr: TStrings;
begin
ColCount := ColCount + 1;
if Index < 0 then
Index := 0;
if Index >= ColCount then
Index := ColCount - 1;
Result := Cols[Index];
if ColCount = 1 then
Exit;
for I := ColCount - 2 downto Index do
begin
AStr := Cols[I];
Cols[I + 1] := AStr;
end;
Result := Cols[Index];
Result.Clear;
end;
function TJvStringGrid.InsertRow(Index: Integer): TStrings;
var
I: Integer;
AStr: TStrings;
begin
RowCount := RowCount + 1;
if Index < 0 then
Index := 0;
if Index >= RowCount then
Index := RowCount - 1;
Result := Rows[Index];
if RowCount = 1 then
Exit;
for I := RowCount - 2 downto Index do
begin
AStr := Rows[I];
Rows[I + 1] := AStr;
end;
Result.Clear;
end;
procedure TJvStringGrid.RemoveCol(Index: Integer);
var
I: Integer;
AStr: TStrings;
begin
if Index < 0 then
Index := 0;
if Index >= ColCount then
Index := ColCount - 1;
for I := Index + 1 to ColCount - 1 do
begin
AStr := Cols[I];
Cols[I - 1] := AStr;
end;
if ColCount > 1 then
ColCount := ColCount - 1;
end;
procedure TJvStringGrid.RemoveRow(Index: Integer);
var
I: Integer;
AStr: TStrings;
begin
if Index < 0 then
Index := 0;
if Index >= RowCount then
Index := RowCount - 1;
for I := Index + 1 to RowCount - 1 do
begin
AStr := Rows[I];
Rows[I - 1] := AStr;
end;
if RowCount > 1 then
RowCount := RowCount - 1;
end;
procedure TJvStringGrid.Clear;
var
I: Integer;
begin
for I := 0 to ColCount - 1 do
Cols[I].Clear;
end;
procedure TJvStringGrid.HideCol(Index: Integer);
begin
ColWidths[Index] := -1;
end;
procedure TJvStringGrid.HideRow(Index: Integer);
begin
RowHeights[Index] := -1;
end;
procedure TJvStringGrid.ShowCol(Index, AWidth: Integer);
begin
if AWidth <= 0 then
AWidth := DefaultColWidth;
ColWidths[Index] := AWidth;
end;
procedure TJvStringGrid.ShowRow(Index, AHeight: Integer);
begin
if AHeight <= 0 then
AHeight := DefaultRowHeight;
RowHeights[Index] := AHeight;
end;
procedure TJvStringGrid.SetFixedFont(const Value: TFont);
begin
FFixedFont.Assign(Value);
end;
procedure TJvStringGrid.DoFixedFontChange(Sender: TObject);
begin
Invalidate;
end;
{$IFDEF COMPILER6_UP}
// NEW-WP - invoked dynamically from the TInplaceEditList
procedure TJvStringGrid.ListBoxCloseUp(Sender: TObject);
begin
if Assigned(FOnListBoxCloseUp) then
FOnListBoxCloseUp(Self)
end;
procedure TJvStringGrid.EditButtonClick(Sender: TObject); //dynamic;
begin
if Assigned(FOnEditButtonClick) then
FOnEditButtonClick(Self)
end;
{$ENDIF COMPILER6_UP}
procedure TJvStringGrid.AutoSizeCol(Index, MinWidth: Integer);
var
I, J, AColWidth: Integer;
ASize: TSize;
begin
if (Index >= 0) and (Index < ColCount) then
begin
if MinWidth < 0 then
AColWidth := DefaultColWidth
else
AColWidth := MinWidth;
for J := 0 to RowCount - 1 do
begin
{$IFDEF VCL}
if GetTextExtentPoint32(Canvas.Handle, PChar(Cells[Index, J]), Length(Cells[Index, J]), ASize) then
{$ENDIF VCL}
{$IFDEF VisualCLX}
if GetTextExtentPoint32W(Canvas.Handle, PWideChar(Cells[Index, J]), Length(Cells[Index, J]), ASize) then
{$ENDIF VisualCLX}
AColWidth := Max(AColWidth, ASize.cx + 8);
end;
ColWidths[Index] := AColWidth;
end
else
begin
for I := 0 to ColCount - 1 do
begin
if MinWidth < 0 then
AColWidth := DefaultColWidth
else
AColWidth := MinWidth;
for J := 0 to RowCount - 1 do
begin
{$IFDEF VCL}
if GetTextExtentPoint32(Canvas.Handle, PChar(Cells[I, J]), Length(Cells[I, J]), ASize) then
{$ENDIF VCL}
{$IFDEF VisualCLX}
if GetTextExtentPoint32W(Canvas.Handle, PWideChar(Cells[I, J]), Length(Cells[I, J]), ASize) then
{$ENDIF VisualCLX}
AColWidth := Max(AColWidth, ASize.cx + 8);
end;
ColWidths[I] := AColWidth;
end;
end;
end;
procedure TJvStringGrid.HideAll;
var
I: Integer;
begin
if ColCount < RowCount then
for I := 0 to ColCount - 1 do
ColWidths[I] := -1
else
for I := 0 to RowCount - 1 do
RowHeights[I] := -1;
end;
procedure TJvStringGrid.ClearSelection; // Clears selection rectangle!
var
S: TGridRect;
begin
S.Left := -1;
S.Top := -1;
S.Right := -1;
S.Bottom := -1;
Self.Selection := S;
Refresh;
end;
procedure TJvStringGrid.ShowAll(AWidth, AHeight: Integer);
var
I: Integer;
begin
if AWidth < 0 then
AWidth := DefaultColWidth;
if AHeight < 0 then
AHeight := DefaultRowHeight;
for I := 0 to ColCount - 1 do
if ColWidths[I] < 0 then
ColWidths[I] := AWidth;
for I := 0 to RowCount - 1 do
if RowHeights[I] < 0 then
RowHeights[I] := AHeight;
end;
function TJvStringGrid.IsHidden(ACol, ARow: Integer): Boolean;
begin
Result := (ColWidths[ACol] < 0) or (RowHeights[ARow] < 0);
end;
procedure TJvStringGrid.HideCell(ACol, ARow: Integer);
begin
ColWidths[ACol] := -1;
RowHeights[ARow] := -1;
end;
procedure TJvStringGrid.ShowCell(ACol, ARow, AWidth, AHeight: Integer);
begin
if AWidth < 0 then
AWidth := DefaultColWidth;
if AHeight < 0 then
AWidth := DefaultRowHeight;
if ColWidths[ACol] < 0 then
ColWidths[ACol] := AWidth;
if RowHeights[ARow] < 0 then
RowHeights[ARow] := AHeight;
end;
procedure TJvStringGrid.DoLoadProgress(Position, Count: Integer);
begin
if Assigned(FOnLoadProgress) then
FOnLoadProgress(Self, Position, Count);
end;
procedure TJvStringGrid.DoSaveProgress(Position, Count: Integer);
begin
if Assigned(FOnSaveProgress) then
FOnSaveProgress(Self, Position, Count);
end;
procedure TJvStringGrid.SortGridByCols(ColOrder: array of Integer; Fixed: Boolean);
var
I, J, FirstRow: Integer;
Sorted: Boolean;
function Sort(Row1, Row2: Integer): Integer;
var
C: Integer;
begin
C := 0;
Result := AnsiCompareStr(Cols[ColOrder[C]][Row1], Cols[ColOrder[C]][Row2]);
if Result = 0 then
begin
Inc(C);
while (C <= High(ColOrder)) and (Result = 0) do
begin
Result := AnsiCompareStr(Cols[ColOrder[C]][Row1], Cols[ColOrder[C]][Row2]);
Inc(C);
end;
end;
end;
begin
// (p3) is this really necessary? Doesn't seem so to me...
// No, it isn't... Removed by Dom
//if SizeOf(ColOrder) div SizeOf(I) <> ColCount then
// Exit;
for I := 0 to High(ColOrder) do
if (ColOrder[I] < 0) or (ColOrder[I] >= ColCount) then
Exit;
if Fixed then
FirstRow := 0
else
FirstRow := FixedRows;
J := FirstRow;
Sorted := False;
repeat
Inc(J);
for I := FirstRow to RowCount - 2 do
if Sort(I, I + 1) > 0 then
begin
MoveRow(I + 1, I);
Sorted := False;
end;
until Sorted or (J = 1000);
Repaint;
end;
procedure TJvStringGrid.MoveColumn(FromIndex, ToIndex: Integer);
begin
inherited MoveColumn(FromIndex, ToIndex);
end;
procedure TJvStringGrid.MoveRow(FromIndex, ToIndex: Integer);
begin
inherited MoveRow(FromIndex, ToIndex);
end;
{$IFDEF UNITVERSIONING}
initialization
RegisterUnitVersion(HInstance, UnitVersioning);
finalization
UnregisterUnitVersion(HInstance);
{$ENDIF UNITVERSIONING}
end.