[fpc-pascal] Where is the 'write' function defined and how is it different from 'writeln'?
Sven Barth
pascaldragon at googlemail.com
Sat Mar 21 12:35:51 CET 2015
On 21.03.2015 11:13, vfclists . wrote:
>
>
> On 20 March 2015 at 20:54, Sven Barth <pascaldragon at googlemail.com
> <mailto:pascaldragon at googlemail.com>> wrote:
>
> On 20.03.2015 21:18, vfclists . wrote:
>
>
>
> On 20 March 2015 at 19:34, Sven Barth
> <pascaldragon at googlemail.com <mailto:pascaldragon at googlemail.com>
> <mailto:pascaldragon at __googlemail.com
> <mailto:pascaldragon at googlemail.com>>> wrote:
>
> Am 20.03.2015 19:19 schrieb "vfclists ."
> <vfclists at gmail.com <mailto:vfclists at gmail.com>
> <mailto:vfclists at gmail.com <mailto:vfclists at gmail.com>>>:
>
>
> snip
>
> How do you ensure own implementation overrides the system's
> implementation, does the compiler take care of that
> automatically, or
> will you have to name your function differently?
>
>
> There is no need to ensure that. Here is an example:
>
> === code begin ===
>
> var
> f, oldout: TextFile;
> begin
> Writeln('Hello Output as StdOut');
>
> oldout := Output;
>
> Assign(Output, 'test.txt');
> Rewrite(Output);
>
> Writeln('Hello Output as file');
>
> Close(f);
>
> Output := oldout;
>
> Writeln('Hello Output as StdOut again');
> end.
>
> === code end ===
>
> To see how such a TextFile is implemented you can take a look at
> unit StreamIO which is part of FPC's units.
>
> (Though I wonder why "Assign(f, 'test.txt'); Output := f;
> Writeln('Hello Output as file');" does not work :/ )
>
> Regards,
> Sven
>
> _________________________________________________
> fpc-pascal maillist - fpc-pascal at lists.freepascal.__org
> <mailto:fpc-pascal at lists.freepascal.org>
> http://lists.freepascal.org/__cgi-bin/mailman/listinfo/fpc-__pascal
> <http://lists.freepascal.org/cgi-bin/mailman/listinfo/fpc-pascal>
>
>
> I need to get the output of a program which uses a lot of Write and
> Writeln commands into the GUI in realtime, by that I not having to
> output it to a text file and reading it afterwards, but by capturing the
> output of each Write command into a variable and displaying it in the
> GUI immediately.
>
> If each Write or Writeln could trigger an event, I could use the event
> to capture the output. My other option is to replace the calls to write
> with my own function, but Write has different number of call parameters
> and it may require as many variants of the function as are used in the
> program, assuming that the call syntax is regular, not something like
> this one - write(JSValToDouble(cx,pom^)):1:scale).
The usage of a text file was merely an example. As I said there already
is the possibility to use a TStream provided by FPC. But since I'm nice
here you also have an example for a TMemo, I'm sure you can adjust that
for your needs:
=== code begin ===
resourcestring
SErrNilMemo = 'Memo is nil';
type
PMemo = ^TMemo;
function GetMemo(var F: TTextRec): TMemo;
begin
Result:=PMemo(@F.Userdata)^;
end;
function MemoWrite(var F: TTextRec): LongInt;
var
s: String;
begin
Result := 0;
with F do
if BufPos > 0 then
try
SetLength(s, BufPos);
Move(BufPtr^, s[1], BufPos);
GetMemo(F).SelText := s;
BufPos:=0;
except
Result:=101;
end;
end;
function MemoClose(var F: TTextRec): LongInt;
begin
Result := 0;
end;
function MemoOpen(var F: TTextRec): LongInt;
begin
Result := 0;
with F do begin
BufPos:=0;
Bufend:=0;
case Mode of
fmInput: begin
Result := 104;
end;
fmOutput, fmAppend: begin
InOutFunc := @MemoWrite;
FlushFunc := @MemoWrite;
if Mode = fmAppend then
Try
with GetMemo(F) do begin
SelStart := Length(Text);
end;
except
Result := 156;
end;
end else begin
GetMemo(F).Clear;
end;
end;
end;
end;
procedure AssignMemo(var F: Text; aMemo: TMemo);
var
e: EInoutError;
begin
if not Assigned(aMemo) then begin
E:=EInOutError.Create(SErrNilMemo);
E.ErrorCode:=6;
Raise E;
end;
with TTextRec(F) do begin
OpenFunc := @MemoOpen;
CloseFunc := @MemoClose;
case DefaultTextLineBreakStyle Of
tlbsLF:
TextRec(f).LineEnd := #10;
tlbsCRLF:
TextRec(f).LineEnd := #13#10;
tlbsCR:
TextRec(f).LineEnd := #13;
end;
PMemo(@UserData)^ := aMemo;
Mode := fmClosed;
BufSize := SizeOf(Buffer);
BufPtr := @Buffer;
Name[0] := #0;
end;
end;
=== code end ===
To use it you should use the following code (for example in FormCreate):
=== code begin ===
fOldOutput := Output; // store the old Output somewhere
AssignMemo(Output, Memo1);
Rewrite(Output);
=== code end ===
Then in FormDestroy you cleanup:
=== code begin ===
CloseFile(Output);
Output := fOldOutput; // restore old Output
=== code end ===
And to illustrate that it works I've used a TTimer and added the
following to its OnTimer event:
=== code begin ===
Writeln('Hello World ', fIndex);
Inc(fIndex); // fIndex is a LongInt field of the form
=== code end ===
And with that TMemo gets spammed with 'Hello World N' messages ;)
You could of course also implement it in a way that you assign a event
handler to a text file instead of a memo. As I said the Pascal I/O
mechanism is very flexible here.
And just in case: by using this mechanism you don't need to adjust any
of the Write(Ln)s, the full spectrum of variants of how Write(Ln) can be
called is supported.
Regards,
Sven
More information about the fpc-pascal
mailing list