Componentes.Terceros.FastRe.../internal/4.2/1/Source/ExportPack/frxExportText.pas
2007-11-18 19:40:07 +00:00

529 lines
14 KiB
ObjectPascal

{******************************************}
{ }
{ FastReport v4.0 }
{ Simple text export }
{ }
{ Copyright (c) 1998-2007 }
{ by Alexander Fediachov, }
{ Fast Reports Inc. }
{ }
{******************************************}
unit frxExportText;
interface
{$I frx.inc}
uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
StdCtrls, extctrls, frxClass, frxExportMatrix, ShellAPI
{$IFDEF Delphi6}, Variants {$ENDIF};
type
TfrxSimpleTextExportDialog = class(TForm)
OkB: TButton;
CancelB: TButton;
SaveDialog1: TSaveDialog;
GroupPageRange: TGroupBox;
DescrL: TLabel;
AllRB: TRadioButton;
CurPageRB: TRadioButton;
PageNumbersRB: TRadioButton;
PageNumbersE: TEdit;
GroupQuality: TGroupBox;
PageBreaksCB: TCheckBox;
OpenCB: TCheckBox;
FramesCB: TCheckBox;
EmptyLinesCB: TCheckBox;
OEMCB: TCheckBox;
procedure FormCreate(Sender: TObject);
procedure PageNumbersEChange(Sender: TObject);
procedure PageNumbersEKeyPress(Sender: TObject; var Key: Char);
procedure FormKeyDown(Sender: TObject; var Key: Word;
Shift: TShiftState);
end;
TfrxSimpleTextExport = class(TfrxCustomExportFilter)
private
FPageBreaks: Boolean;
FMatrix: TfrxIEMatrix;
FOpenAfterExport: Boolean;
Exp: TStream;
FPage: TfrxReportPage;
FFrames: Boolean;
pX: Extended;
pY: Extended;
pT: Extended;
FEmptyLines: Boolean;
FOEM: Boolean;
procedure ExportPage(Stream: TStream);
public
constructor Create(AOwner: TComponent); override;
class function GetDescription: String; override;
function ShowModal: TModalResult; override;
function Start: Boolean; override;
procedure Finish; override;
procedure FinishPage(Page: TfrxReportPage; Index: Integer); override;
procedure StartPage(Page: TfrxReportPage; Index: Integer); override;
procedure ExportObject(Obj: TfrxComponent); override;
published
property PageBreaks: Boolean read FPageBreaks write FPageBreaks default True;
property Frames: Boolean read FFrames write FFrames;
property EmptyLines: Boolean read FEmptyLines write FEmptyLines;
property OEMCodepage: Boolean read FOEM write FOEM;
property OpenAfterExport: Boolean read FOpenAfterExport write FOpenAfterExport default False;
end;
implementation
uses frxUtils, frxFileUtils, frxUnicodeUtils, frxRes, frxrcExports;
{$R *.dfm}
{ TfrxSimpleTextExport }
constructor TfrxSimpleTextExport.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
FPageBreaks := True;
FFrames := False;
FEmptyLines := False;
FOEM := False;
FilterDesc := frxGet(8801);
DefaultExt := frxGet(8802);
end;
class function TfrxSimpleTextExport.GetDescription: String;
begin
Result := frxResources.Get('SimpleTextExport');
end;
procedure TfrxSimpleTextExport.ExportPage(Stream: TStream);
var
x, y, i: Integer;
FScreen: array of Char;
FScreenWidth: Integer;
FScreenHeight: Integer;
Obj: TfrxIEMObject;
c: Char;
s: String;
function MakeStr(C: Char; N: Integer): String;
begin
if N < 1 then
Result := ''
else
begin
SetLength(Result, N);
FillChar(Result[1], Length(Result), C);
end;
end;
function AddChar(C: Char; const S: String; N: Integer): String;
begin
if Length(S) < N then
Result := MakeStr(C, N - Length(S)) + S else
Result := S;
end;
function AddCharR(C: Char; const S: String; N: Integer): String;
begin
if Length(S) < N then
Result := S + MakeStr(C, N - Length(S)) else
Result := S;
end;
function LeftStr(const S: String; N: Integer): String;
begin
Result := AddCharR(' ', S, N);
end;
function RightStr(const S: String; N: Integer): String;
begin
Result := AddChar(' ', S, N);
end;
function CenterStr(const S: String; Len: Integer): String;
begin
if Length(S) < Len then
begin
Result := MakeStr(' ', (Len div 2) - (Length(S) div 2)) + S;
Result := Result + MakeStr(' ', Len - Length(Result));
end
else
Result := S;
end;
function AlignStr(const Buf: String; const style: TfrxIEMStyle; const Width: Integer): String;
begin
if (style.HAlign = haRight) then
Result := RightStr(buf, Width - 1)
else if (style.HAlign = haCenter) then
Result := CenterStr(buf, Width - 1)
else
Result := LeftStr(buf, Width - 1);
end;
procedure ScreenCreate;
begin
Initialize(FScreen);
SetLength(FScreen, FScreenWidth * FScreenHeight);
FillChar(FScreen[0], Length(FScreen), #32);
end;
procedure ScreenType(const x,y: Integer; const c: Char);
begin
FScreen[FScreenWidth * y + x] := c;
end;
procedure ScreenString(const x, y: Integer; const s: String);
var
i: Integer;
begin
for i := 0 to Length(s) - 1 do
ScreenType(x + i, y, s[i + 1]);
end;
procedure ScreenMemo(const Obj: TfrxIEMObject);
var
i: Integer;
curx, cury: Integer;
s: String;
function StrToOem(const AnsiStr: String): String;
begin
SetLength(Result, Length(AnsiStr));
if Length(Result) > 0 then
CharToOemBuff(PChar(AnsiStr), PChar(Result), Length(Result));
end;
begin
curx := Round(Obj.Left / pX);
cury := Round(Obj.Top / pY);
for i := 0 to Obj.Memo.Count - 1 do
begin
if FOEM then
s := StrToOem(Obj.Memo[i])
else
s := Obj.Memo[i];
ScreenString(curx, cury + i, AlignStr(s, Obj.Style, Round(Obj.Width / pX) - 1));
end;
if FFrames then
begin
if (ftLeft in Obj.Style.FrameTyp) then
for i := 0 to Round(Obj.Height / pY) - 1 do
ScreenType(curx - 1, cury + i, '|');
if (ftRight in Obj.Style.FrameTyp) then
for i := 0 to Round(Obj.Height / pY) - 1 do
ScreenType(curx + Round(Obj.Width / pX) - 2, cury + i, '|');
if (ftTop in Obj.Style.FrameTyp) then
for i := 0 to Round(Obj.Width / pX) - 1 do
ScreenType(curx - 1 + i, cury - 1, '-');
if (ftBottom in Obj.Style.FrameTyp) then
for i := 0 to Round(Obj.Width / pX) - 1 do
ScreenType(curx - 1 + i, cury + Round(Obj.Height / pY) - 1, '-');
end;
end;
function ScreenGet(const x, y: Integer): Char;
begin
if (x < FScreenWidth) and (y < FScreenHeight) and
(x >= 0) and (y >= 0) then
Result := FScreen[FScreenWidth * y + x] else
Result := ' ';
end;
function GetMaxPX(const Obj: TfrxIEMObject): Extended;
var
i : Integer;
begin
Result := 0;
for i := 0 to Obj.Memo.Count - 1 do
if Length(Obj.Memo[i]) > Result then
Result := Length(Obj.Memo[i]);
if (Result > 0) then //and (Obj.Width )
Result := 6 * Obj.Width / (Result * Obj.Style.Font.Size);
if Result < 1 then
Result := 1;
end;
function GetMaxPY(const Obj: TfrxIEMObject): Extended;
begin
if Obj.Memo.Count > 0 then
Result := 5 * Obj.Height / (Obj.Memo.Count * Obj.Style.Font.Size)
else
Result := 0;
end;
function ColumnEmpty(const x: Integer): Boolean;
var
y: Integer;
begin
Result := True;
for y := 0 to FScreenHeight - 1 do
if FScreen[FScreenWidth * y + x] <> #32 then
begin
Result := False;
break;
end;
end;
procedure DeleteColumn(const x: Integer);
var
i, j: Integer;
begin
for i := 0 to FScreenHeight - 1 do
begin
for j := x to FScreenWidth - 2 do
FScreen[FScreenWidth * i + j] := FScreen[FScreenWidth * i + j + 1];
FScreen[FScreenWidth * i + FScreenWidth - 1] := #32;
end;
end;
begin
FMatrix.Prepare;
for i := 0 to FMatrix.ObjectsCount - 1 do
begin
pT := GetMaxPX(FMatrix.GetObjectById(i));
if (pT < pX) and (pT <> 0) then
pX := pT;
pT := GetMaxPY(FMatrix.GetObjectById(i));
if (pT < pY) and (pT <> 0) then
pY := pT;
end;
FScreenWidth := Round(FPage.Width / pX);
FScreenHeight := Round(FPage.Height / pY);
ScreenCreate;
for y := 0 to FMatrix.Height - 2 do
begin
for x := 0 to FMatrix.Width - 1 do
begin
i := FMatrix.GetCell(x, y);
if (i <> -1) then
begin
Obj := FMatrix.GetObjectById(i);
ScreenMemo(Obj);
end;
end;
end;
x := 0;
i := 2;
y := FScreenWidth;
while x < y - 1 do
if ColumnEmpty(x) then
begin
if i = 0 then
begin
DeleteColumn(x);
Dec(y);
end
else begin
Dec(i);
Inc(x);
end;
end
else begin
Inc(x);
i := 2;
end;
for y := 0 to FScreenHeight - 1 do
begin
s := '';
for x := 0 to FScreenWidth - 1 do
begin
c := ScreenGet(x, y);
s := s + c;
end;
s := TrimRight(s);
if (Length(Trim(s)) > 0) or FEmptyLines then
begin
s := s + #13#10;
Stream.Write(s[1], Length(s));
end;
end;
if FPageBreaks then
Stream.Write(String(#12), 1);
end;
function TfrxSimpleTextExport.ShowModal: TModalResult;
begin
if not Assigned(Stream) then
begin
with TfrxSimpleTextExportDialog.Create(nil) do
begin
OpenCB.Visible := not SlaveExport;
if SlaveExport then
FOpenAfterExport := False;
if (FileName = '') and (not SlaveExport) then
SaveDialog1.FileName := ChangeFileExt(ExtractFileName(frxUnixPath2WinPath(Report.FileName)), SaveDialog1.DefaultExt)
else
SaveDialog1.FileName := FileName;
PageBreaksCB.Checked := FPageBreaks;
OpenCB.Checked := FOpenAfterExport;
FramesCB.Checked := FFrames;
EmptyLinesCB.Checked := FEmptyLines;
OEMCB.Checked := FOEM;
if PageNumbers <> '' then
begin
PageNumbersE.Text := PageNumbers;
PageNumbersRB.Checked := True;
end;
Result := ShowModal;
if Result = mrOk then
begin
PageNumbers := '';
CurPage := False;
if CurPageRB.Checked then
CurPage := True
else if PageNumbersRB.Checked then
PageNumbers := PageNumbersE.Text;
FPageBreaks := PageBreaksCB.Checked;
FOpenAfterExport := OpenCB.Checked;
FFrames := FramesCB.Checked;
FEmptyLines := EmptyLinesCB.Checked;
FOEM := OEMCB.Checked;
if not SlaveExport then
begin
if DefaultPath <> '' then
SaveDialog1.InitialDir := DefaultPath;
if SaveDialog1.Execute then
FileName := SaveDialog1.FileName
else
Result := mrCancel;
end
else
FileName := ChangeFileExt(GetTempFile, SaveDialog1.DefaultExt);
end;
Free;
end;
end else
Result := mrOk;
end;
function TfrxSimpleTextExport.Start: Boolean;
begin
if (FileName <> '') or Assigned(Stream) then
begin
if (ExtractFilePath(FileName) = '') and (DefaultPath <> '') then
FileName := DefaultPath + '\' + FileName;
FMatrix := TfrxIEMatrix.Create(UseFileCache, Report.EngineOptions.TempDir);
FMatrix.Background := False;
FMatrix.BackgroundImage := False;
FMatrix.Printable := ExportNotPrintable;
FMatrix.RichText := True;
FMatrix.PlainRich := True;
FMatrix.AreaFill := False;
FMatrix.CropAreaFill := True;
FMatrix.Inaccuracy := 0.5;
FMatrix.DeleteHTMLTags := True;
FMatrix.Images := False;
FMatrix.WrapText := True;
FMatrix.ShowProgress := False;
FMatrix.FramesOptimization := True;
try
if Assigned(Stream) then
Exp := Stream
else
Exp := TFileStream.Create(FileName, fmCreate);
Result := True;
except
Result := False;
end;
end
else
Result := False;
end;
procedure TfrxSimpleTextExport.StartPage(Page: TfrxReportPage; Index: Integer);
begin
FMatrix.Clear;
pX := MAXWORD;
pY := MAXWORD;
end;
procedure TfrxSimpleTextExport.ExportObject(Obj: TfrxComponent);
begin
if Obj is TfrxView then
FMatrix.AddObject(TfrxView(Obj));
end;
procedure TfrxSimpleTextExport.FinishPage(Page: TfrxReportPage; Index: Integer);
begin
FPage := Page;
ExportPage(Exp);
end;
procedure TfrxSimpleTextExport.Finish;
begin
FMatrix.Free;
if not Assigned(Stream) then
Exp.Free;
if FOpenAfterExport and (not Assigned(Stream)) then
ShellExecute(GetDesktopWindow, 'open', PChar(FileName), nil, nil, SW_SHOW);
end;
{ TfrxSimpleTextExportDialog }
procedure TfrxSimpleTextExportDialog.FormCreate(Sender: TObject);
begin
Caption := frxGet(8800);
OkB.Caption := frxGet(1);
CancelB.Caption := frxGet(2);
GroupPageRange.Caption := frxGet(7);
AllRB.Caption := frxGet(3);
CurPageRB.Caption := frxGet(4);
PageNumbersRB.Caption := frxGet(5);
DescrL.Caption := frxGet(9);
GroupQuality.Caption := frxGet(8302);
PageBreaksCB.Caption := frxGet(6);
FramesCB.Caption := frxGet(8312);
EmptyLinesCB.Caption := frxGet(8305);
OEMCB.Caption := frxGet(8304);
OpenCB.Caption := frxGet(8706);
SaveDialog1.Filter := frxGet(8801);
SaveDialog1.DefaultExt := frxGet(8802);
if UseRightToLeftAlignment then
FlipChildren(True);
end;
procedure TfrxSimpleTextExportDialog.PageNumbersEChange(Sender: TObject);
begin
PageNumbersRB.Checked := True;
end;
procedure TfrxSimpleTextExportDialog.PageNumbersEKeyPress(Sender: TObject;
var Key: Char);
begin
case key of
'0'..'9':;
#8, '-', ',':;
else
key := #0;
end;
end;
procedure TfrxSimpleTextExportDialog.FormKeyDown(Sender: TObject;
var Key: Word; Shift: TShiftState);
begin
if Key = VK_F1 then
frxResources.Help(Self);
end;
end.