| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522 | {    This file is part of the Free Pascal Integrated Development Environment    Copyright (c) 2000 by Berczi Gabor    Help support for Norton Guide (.NG) files    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. **********************************************************************}{$R-}unit WNGHelp;{$ifdef cpullvm}{$modeswitch nestedprocvars}{$endif}interfaceuses Objects,     WUtils,WHelp;const      NGFileSignature = 'NG';      NGXORByte       = $1a;      NGMinRecordSize = $1a;      ng_rtContainer    = Byte ($0);      ng_rtTopic        = Byte ($1);type      TNGFileHeader = packed record        Signature     : array[1..2] of char;        Unknown       : word;        Version       : word;        MenuCount     : word;        GuideName     : array[8..47] of char;        Credits       : array[48..377] of char;      end;      TNGRecordHeader = packed record        RecType      : word;        RecLength    : word;      end;      TNGContainerItem = packed record        EntryNameOfs  : word; { relative in record }        SubItemsOfs   : longint; { file offset to a new record header }      end;      PNGContainerRecord = ^TNGContainerRecord;      TNGContainerRecord = packed record        ItemCount     : word;        Unknown       : word;        IndexInParent : integer;        ParentOfs     : longint;        MenuNo        : integer;{ belongs to menu # }        MenuItemNo    : integer;{ belongs to menu item # }        Unknown2      : array[18..25] of byte;        Items         : array[0..0] of TNGContainerItem;      end;      TNGTopicRecord = packed record        NumberOfLines : word;        SeeAlsoOfs    : word;        IndexInParent : integer;        ParentOfs     : longint;        MenuNo        : integer;{ belongs to menu # }        MenuItemNo    : integer;{ belongs to menu item # }        PrevTopicOfs  : longint;        NextTopicOfs  : longint;        TopicLines    : record end;        { array of TNGSeeAlsoRec }      end;      PNGSeeAlsoRec = ^TNGSeeAlsoRec;      TNGSeeAlsoRec = packed record        EntryCount    : word;        Entries       : record end;        { array of LinkedRecOfs  : longint; }        { array of LinkNames : ASCIIZ; }      end;      PContainerItemRec = ^TContainerItemRec;      TContainerItemRec = record        Name     : string;        FilePos  : longint;        Container: PNGContainerRecord;      end;      PLinkRec = ^TLinkRec;      TLinkRec = record        Name     : string;        FilePos  : longint;      end;      PNGHelpFile = ^TNGHelpFile;      TNGHelpFile = object(THelpFile)        constructor Init(AFileName: string; AID: word);        destructor  Done; virtual;      public        function    LoadIndex: boolean; virtual;        function    ReadTopic(T: PTopic): boolean; virtual;      private        F: PStream;        Header: TNGFileHeader;        FirstRecordPos: longint;        IndexLoaded: boolean;{        NextHelpCtx: longint;}        function ReadHeader: boolean;        function ReadContainer(EnumProc: TCallbackProcParam): boolean;        function ReadTopicRec(LineEnumProc: TCallbackProcParam; LinkEnumProc: TCallbackProcParam): boolean;        function ReadRecord(var R: TRecord; ReadData: boolean): boolean;      end;    TNGGetAttrColorProc = function(Attr: char; var Color: byte): boolean;function DefNGGetAttrColor(Attr: char; var Color: byte): boolean;const NGGetAttrColor : TNGGetAttrColorProc = {$ifdef fpc}@{$endif}DefNGGetAttrColor;procedure RegisterHelpType;implementationfunction DefNGGetAttrColor(Attr: char; var Color: byte): boolean;begin  DefNGGetAttrColor:=false;end;function NGDecompressStr(const S: string): string;var NS: string;    I: sw_integer;begin  NS:='';  I:=1;  while (I<=length(S)) do  begin    if S[I]=#255 then      begin        NS:=NS+CharStr(' ',ord(S[I+1]));        Inc(I);      end    else      NS:=NS+S[I];    Inc(I);  end;  NGDecompressStr:=NS;end;function TranslateStr(const S: string): string;var NS: string;    I: sw_integer;    InHiLite: boolean;    Color: byte;begin  NS:=''; InHiLite:=false;  I:=1;  while (I<=length(S)) do  begin    case S[I] of      '^'  : begin               Inc(I);               case S[I] of                 '^' : NS:=NS+'^';                 'A'..'Z',                 'a'..'z' :                       begin                         if InHiLite then                           NS:=NS+hscNormText                         else                           if NGGetAttrColor(S[I],Color) then                             NS:=NS+hscTextColor+chr(Color);                         InHiLite:=not InHiLite;                       end;               else                 NS:=NS;               end;             end;    else NS:=NS+S[I];    end;    Inc(I);  end;  if InHiLite then NS:=NS+hscNormText;  TranslateStr:=NS;end;procedure TranslateLines(P: PUnsortedStringCollection);var S: string;    I: sw_integer;begin  for I:=0 to P^.Count-1 do  begin    S:=GetStr(P^.At(I));    P^.AtFree(I);    P^.AtInsert(I,NewStr(TranslateStr(S)));  end;end;constructor TNGHelpFile.Init(AFileName: string; AID: word);var OK: boolean;begin  if inherited Init(AID)=false then Fail;  F:=New(PFastBufStream, Init(AFileName, stOpenRead, HelpStreamBufSize));  OK:=F<>nil;  if OK then OK:=(F^.Status=stOK);  if OK then    begin      OK:=ReadHeader;      if OK then        FirstRecordPos:=F^.GetPos;    end;  if OK=false then  begin    Done;    Fail;  end;end;function TNGHelpFile.ReadHeader: boolean;var OK: boolean;begin  F^.Read(Header,sizeof(Header));  OK:=(F^.Status=stOK);  OK:=OK and (Header.Signature=NGFileSignature);  ReadHeader:=OK;end;function TNGHelpFile.ReadContainer(EnumProc: TCallbackProcParam): boolean;var OK: boolean;    R: TRecord;    I: longint;    P: pointer;    CIR: TContainerItemRec;begin  OK:=ReadRecord(R, true);  if OK then  with TNGContainerRecord(R.Data^) do  begin    I:=0;    while (I<ItemCount) do    with Items[I] do    begin      P:=@(PByteArray(R.Data)^[NGMinRecordSize-sizeof(TNGRecordHeader)+EntryNameOfs]);      FillChar(CIR,sizeof(CIR),0);      with CIR do      begin        Container:=R.Data;        Name:=NGDecompressStr(StrPas(P));        FilePos:=SubItemsOfs;      end;      CallPointerLocal(EnumProc,get_caller_frame(get_frame,get_pc_addr),@CIR);      Inc(I);    end;  end;  DisposeRecord(R);  ReadContainer:=OK;end;function TNGHelpFile.ReadTopicRec(LineEnumProc, LinkEnumProc: TCallbackProcParam): boolean;var OK: boolean;    R: TRecord;    I: sw_integer;    LineP: pointer;    S: string;    ParamS: string;    NextLinkOfsPtr,NextLinkNamePtr: pointer;    SeeAlso: PNGSeeAlsoRec;    LR: TLinkRec;begin  OK:=ReadRecord(R, true);  if OK then  with TNGTopicRecord(R.Data^) do  begin    LineP:=@TopicLines;    if Assigned(LineEnumProc) then    for I:=1 to NumberOfLines do    begin      S:=StrPas(LineP);      ParamS:=NGDecompressStr(S);      CallPointerLocal(LineEnumProc,get_caller_frame(get_frame,get_pc_addr),@ParamS);      Inc(Ptrint(LineP),length(S)+1);    end;    if Assigned(LinkEnumProc) and (SeeAlsoOfs>0) then    begin      SeeAlso:=@PByteArray(R.Data)^[NGMinRecordSize-sizeof(TNGRecordHeader)+SeeAlsoOfs];      NextLinkOfsPtr:=@SeeAlso^.Entries;      NextLinkNamePtr:=@PByteArray(NextLinkOfsPtr)^[SeeAlso^.EntryCount*4];      for I:=1 to SeeAlso^.EntryCount do      begin        FillChar(LR,sizeof(LR),0);        S:=StrPas(NextLinkNamePtr);        LR.Name:=S;        Move(NextLinkOfsPtr^,LR.FilePos,4);        CallPointerLocal(LinkEnumProc,get_caller_frame(get_frame,get_pc_addr),@LR);        Inc(Ptrint(NextLinkNamePtr),length(S)+1);        Inc(Ptrint(NextLinkOfsPtr),4);      end;    end;  end;  DisposeRecord(R);  ReadTopicRec:=OK;end;function TNGHelpFile.ReadRecord(var R: TRecord; ReadData: boolean): boolean;var OK: boolean;    H: TNGRecordHeader;    I: sw_integer;begin  FillChar(R, SizeOf(R), 0);  F^.Read(H,SizeOf(H));  OK:=F^.Status=stOK;  if OK then    for I:=0 to SizeOf(H)-1 do      PByteArray(@H)^[I]:=PByteArray(@H)^[I] xor NGXORByte;  if OK then  begin    R.SClass:=H.RecType; R.Size:=H.RecLength+(NGMinRecordSize-sizeof(TNGRecordHeader));    if (R.Size>0) and ReadData then    begin      GetMem(R.Data,R.Size);      F^.Read(R.Data^,R.Size);      if R.Size>0 then      for I:=0 to R.Size-1 do        PByteArray(R.Data)^[I]:=PByteArray(R.Data)^[I] xor NGXORByte;      OK:=F^.Status=stOK;    end;    if OK=false then DisposeRecord(R);  end;  ReadRecord:=OK;end;function TNGHelpFile.LoadIndex: boolean;{function FormatAlias(Alias: string): string;var StartP,EndP: sw_integer;begin  repeat    StartP:=Pos('  ',Alias);    if StartP>0 then    begin      EndP:=StartP;      while (EndP+1<=length(Alias)) and (Alias[EndP+1]=' ') do Inc(EndP);      Alias:=copy(Alias,1,StartP-1)+' | '+copy(Alias,EndP+1,High(Alias));    end;  until StartP=0;  if Assigned(HelpFacility) then    if length(Alias)>HelpFacility^.IndexTabSize-4 then      Alias:=Trim(copy(Alias,1,HelpFacility^.IndexTabSize-4-2))+'..';  FormatAlias:=Alias;end;}procedure AddToIndex(P: PContainerItemRec);var S: string;begin  S:=Trim(P^.Name);  S:=TranslateStr(S);  S:=Trim({FormatAlias}(S));  if (S<>'') and (P^.FilePos<>-1) then    begin{      Inc(NextHelpCtx);}      AddIndexEntry(S,P^.FilePos);      AddTopic(P^.FilePos,P^.FilePos,'',nil,0);    end;end;var OK: boolean;    FS: longint;    R: TRecord;    L: longint;begin  if IndexLoaded then OK:=true else  begin    FS:=F^.GetSize;    OK:=FirstRecordPos<>0;    while OK do    begin      L:=F^.GetPos;      if (L>=FS) then Break;      OK:=ReadRecord(R,false);      if (OK=false) then Break;      case R.SClass of        ng_rtContainer : begin F^.Seek(L); OK:=ReadContainer(TCallbackProcParam(@AddToIndex)); end;        ng_rtTopic     : ;      else       begin       {$ifdef DEBUGMSG}         ClearFormatParams;         AddFormatParamInt(R.SClass);         AddFormatParamInt(L);         AddFormatParamInt(R.Size);         ErrorBox('Uknown help record tag %x encountered, '+                  'offset %x, size %d',@FormatParams);       {$else}         {Skip};       {$endif}       end;      end;      if OK then         begin           Inc(L, sizeof(TNGRecordHeader)+R.Size); F^.Seek(L);           OK:=(F^.Status=stOK);         end;    end;    IndexLoaded:=OK;  end;  LoadIndex:=OK;end;function TNGHelpFile.ReadTopic(T: PTopic): boolean;function ExtractStr(var Buf; BufSize: word): string;var S: string;    I,Size: integer;    StartP,EndP: sw_integer;begin  S:='';  Size:=BufSize;  while (PByteArray(@Buf)^[Size-1]=0) and (Size>0) do    Dec(Size);  S:=MemToStr(Buf,Size);  S:=NGDecompressStr(S);  for I:=1 to length(S) do    if S[I]=#0 then S[I]:=#32;  { kill long spaces }  repeat    StartP:=Pos('  ',S);    if StartP>0 then    begin      EndP:=StartP;      while (EndP+1<=length(S)) and (S[EndP+1]=' ') do Inc(EndP);      S:=copy(S,1,StartP-1)+hscLineBreak+copy(S,EndP+1,High(S));    end;  until StartP=0;  S:=Trim(S);  ExtractStr:=S;end;var Lines: PUnsortedStringCollection;procedure AddLine(const S: string);begin  Lines^.InsertStr(S);end;procedure AddToTopic(P: PContainerItemRec);begin  AddLine(hscLink+Trim(P^.Name)+hscLink);  AddLinkToTopic(T,ID,P^.FilePos);end;procedure AddTopicLine(P: PString);begin  AddLine(' '+GetStr(P));end;var LinkCount: sw_integer;procedure AddLink(P: PLinkRec);begin  Inc(LinkCount);  if LinkCount=1 then  begin    AddLine('');    AddLine(' See also :');  end;  AddLine('  '+hscLink+Trim(P^.Name)+hscLink);  AddLinkToTopic(T,ID,P^.FilePos);end;var OK: boolean;    R: TRecord;begin  LinkCount:=0;  New(Lines, Init(100,100));  F^.Seek(T^.FileOfs); OK:=F^.Status=stOK;  if OK then OK:=ReadRecord(R,false);  case R.SClass of      ng_rtContainer :        begin          F^.Seek(T^.FileOfs);          AddLine('');          OK:=ReadContainer(TCallbackProcParam(@AddToTopic));          RenderTopic(Lines,T);        end;      ng_rtTopic     :        begin          F^.Seek(T^.FileOfs);          AddLine('');          OK:=ReadTopicRec(TCallbackProcParam(@AddTopicLine),TCallbackProcParam(@AddLink));          TranslateLines(Lines);          AddLine('');          { include copyright info }{          AddLine(CharStr('Ä',80));          AddLine(ExtractStr(Header.GuideName,sizeof(Header.GuideName)));          AddLine(ExtractStr(Header.Credits,sizeof(Header.Credits)));}          RenderTopic(Lines,T);        end;  else OK:=false;  end;  Dispose(Lines, Done);  ReadTopic:=OK;end;destructor TNGHelpFile.Done;begin  if Assigned(F) then Dispose(F, Done); F:=nil;  inherited Done;end;function CreateProc(const FileName,Param: string;Index : longint): PHelpFile;begin  CreateProc:=New(PNGHelpFile, Init(FileName,Index));end;procedure RegisterHelpType;begin  RegisterHelpFileType({$ifdef FPC}@{$endif}CreateProc);end;END.
 |