ViewVC Help
View File | Revision Log | View Changeset | Root Listing
root/Oni2/oup/current/Helper/LevelDB.pas
(Generate patch)

Comparing oup/current/Helper/LevelDB.pas (file contents):
Revision 120 by alloc, Fri Mar 23 00:59:01 2007 UTC vs.
Revision 173 by alloc, Wed May 2 13:19:40 2007 UTC

# Line 1 | Line 1
1 {$A8,B-,C+,D+,E-,F-,G+,H+,I+,J-,K-,L+,M-,N+,O+,P+,Q-,R-,S-,T-,U-,V+,W-,X+,Y+,Z1}
2 {$MINSTACKSIZE $00004000}
3 {$MAXSTACKSIZE $00100000}
4 {$IMAGEBASE $00400000}
5 {$APPTYPE GUI}
6 {$WARN SYMBOL_DEPRECATED ON}
7 {$WARN SYMBOL_LIBRARY ON}
8 {$WARN SYMBOL_PLATFORM ON}
9 {$WARN SYMBOL_EXPERIMENTAL ON}
10 {$WARN UNIT_LIBRARY ON}
11 {$WARN UNIT_PLATFORM ON}
12 {$WARN UNIT_DEPRECATED ON}
13 {$WARN UNIT_EXPERIMENTAL ON}
14 {$WARN HRESULT_COMPAT ON}
15 {$WARN HIDING_MEMBER ON}
16 {$WARN HIDDEN_VIRTUAL ON}
17 {$WARN GARBAGE ON}
18 {$WARN BOUNDS_ERROR ON}
19 {$WARN ZERO_NIL_COMPAT ON}
20 {$WARN STRING_CONST_TRUNCED ON}
21 {$WARN FOR_LOOP_VAR_VARPAR ON}
22 {$WARN TYPED_CONST_VARPAR ON}
23 {$WARN ASG_TO_TYPED_CONST ON}
24 {$WARN CASE_LABEL_RANGE ON}
25 {$WARN FOR_VARIABLE ON}
26 {$WARN CONSTRUCTING_ABSTRACT ON}
27 {$WARN COMPARISON_FALSE ON}
28 {$WARN COMPARISON_TRUE ON}
29 {$WARN COMPARING_SIGNED_UNSIGNED ON}
30 {$WARN COMBINING_SIGNED_UNSIGNED ON}
31 {$WARN UNSUPPORTED_CONSTRUCT ON}
32 {$WARN FILE_OPEN ON}
33 {$WARN FILE_OPEN_UNITSRC ON}
34 {$WARN BAD_GLOBAL_SYMBOL ON}
35 {$WARN DUPLICATE_CTOR_DTOR ON}
36 {$WARN INVALID_DIRECTIVE ON}
37 {$WARN PACKAGE_NO_LINK ON}
38 {$WARN PACKAGED_THREADVAR ON}
39 {$WARN IMPLICIT_IMPORT ON}
40 {$WARN HPPEMIT_IGNORED ON}
41 {$WARN NO_RETVAL ON}
42 {$WARN USE_BEFORE_DEF ON}
43 {$WARN FOR_LOOP_VAR_UNDEF ON}
44 {$WARN UNIT_NAME_MISMATCH ON}
45 {$WARN NO_CFG_FILE_FOUND ON}
46 {$WARN IMPLICIT_VARIANTS ON}
47 {$WARN UNICODE_TO_LOCALE ON}
48 {$WARN LOCALE_TO_UNICODE ON}
49 {$WARN IMAGEBASE_MULTIPLE ON}
50 {$WARN SUSPICIOUS_TYPECAST ON}
51 {$WARN PRIVATE_PROPACCESSOR ON}
52 {$WARN UNSAFE_TYPE OFF}
53 {$WARN UNSAFE_CODE OFF}
54 {$WARN UNSAFE_CAST OFF}
55 {$WARN OPTION_TRUNCATED ON}
56 {$WARN WIDECHAR_REDUCED ON}
57 {$WARN DUPLICATES_IGNORED ON}
58 {$WARN UNIT_INIT_SEQ ON}
59 {$WARN LOCAL_PINVOKE ON}
60 {$WARN MESSAGE_DIRECTIVE ON}
1   unit LevelDB;
2   interface
3   uses
# Line 72 | Line 12 | type
12      btn_abortok:  TButton;
13      lbl_estimation: TLabel;
14      procedure btn_abortokClick(Sender: TObject);
75  private
76    procedure HandleFile(Ext: String; FileID: Integer);
77    procedure StopConvert;
15    public
16      procedure CreateDatabase(Source, Target: String);
17      procedure CreateLevel(Source, Target: String);
# Line 87 | Line 24 | var
24   implementation
25   {$R *.dfm}
26   uses ABSMain, ABSDecUtil, Main,
27 <    ConnectionManager, TypeDefs, DataAccess, OniImgClass, Data;
28 <
92 < type
93 <  THandler = procedure(FileID: Integer);
94 <  TConvertHandler = record
95 <    Ext:     String[4];
96 <    Handler: THandler;
97 <  end;
27 >    ConnectionManager, TypeDefs, DataAccess, OniImgClass, Data, RawList,
28 >  Access_OniArchive;
29  
30   var
100  ConvertHandlers: array of TConvertHandler;
101 //  loaded_filename: String;
31    Converting:  Boolean = False;
32    Abort:       Boolean = False;
33  
# Line 116 | Line 45 | end;
45  
46  
47   procedure TForm_LevelDB.CreateLevel(Source, Target: String);
48 + const
49 +  EmptyBytes: Array[0..31] of Byte = (
50 +      0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 );
51   var
52    DatHeader:        THeader;
53    FilesHeader:      TFilesMap;
# Line 125 | Line 57 | var
57    Stream_Body, Stream_Names:          TMemoryStream;
58    Stream_Dat, Stream_Raw, Stream_Sep: TFileStream;
59  
128 //  Data, rawdata: Tdata;
60    BeginTime, FileTime: Double;
61    Step:     Integer;
131 //  rawlist:  TRawDataList;
132 //  datlinks: TDatLinks;
133  OniImage:   TOniImage;
62    LevelID:    Integer;
63    TimeFormat: TFormatSettings;
64  
# Line 164 | Line 92 | const
92        group_progress.Caption := 'Creating Dat (FINISHED)';
93    end;
94  
95 +  procedure StopConvert;
96 +  begin
97 +    btn_abortok.Caption := '&Close';
98 +    btn_abortok.Default := True;
99 +    converting := False;
100 +    lbl_estimation.Caption := 'ABORTED';
101 +    group_progress.Caption := 'Creating Level (ABORTED)';
102 +
103 +    Stream_Body.Free;
104 +    Stream_Names.Free;
105 +    DatFileStream.Free;
106 +    RawFileStream.Free;
107 +    
108 +    Stream_Dat.Free;
109 +    Stream_Raw.Free;
110 +    if Connection.DataOS in [DOS_WINDEMO, DOS_MAC, DOS_MACBETA] then
111 +      Stream_Sep.Free;
112 +    
113 +    if MessageBox(Self.Handle, PChar('Delete the unfinished level-files?'),
114 +      PChar('Delete files?'), MB_YESNO) = idYes then
115 +    begin
116 +      DeleteFile(target);
117 +      DeleteFile(AnsiReplaceStr(Target, '.dat', '.raw'));
118 +      if Connection.DataOS in [DOS_WINDEMO, DOS_MAC, DOS_MACBETA] then
119 +        DeleteFile(AnsiReplaceStr(Target, '.dat', '.sep'));
120 +    end;
121 +  end;
122 +
123   begin
124  
125    //
# Line 211 | Line 167 | begin
167        begin
168          ShowMessage('Couldn''t delete file. Aborting');
169          Exit;
214      end else if not DeleteFile(AnsiReplaceStr(Target, '.dat', '.raw')) then
215      begin
216        ShowMessage('Couldn''t delete file. Aborting');
217        Exit;
218      end else if not DeleteFile(AnsiReplaceStr(Target, '.dat', '.sep')) then
219      begin
220        ShowMessage('Couldn''t delete file. Aborting');
221        Exit;
170        end;
171 +      if FileExists(AnsiReplaceStr(Target, '.dat', '.raw')) then
172 +        if not DeleteFile(AnsiReplaceStr(Target, '.dat', '.raw')) then
173 +        begin
174 +          ShowMessage('Couldn''t delete file. Aborting');
175 +          Exit;
176 +        end;
177 +      if FileExists(AnsiReplaceStr(Target, '.dat', '.sep')) then
178 +        if Connection.DataOS in [DOS_WINDEMO, DOS_MAC, DOS_MACBETA] then
179 +          if not DeleteFile(AnsiReplaceStr(Target, '.dat', '.sep')) then
180 +          begin
181 +            ShowMessage('Couldn''t delete file. Aborting');
182 +            Exit;
183 +          end;
184      end else begin
185        ShowMessage('Aborting');
186        Exit;
# Line 228 | Line 189 | begin
189  
190    LevelID  := Connection.LevelNumber;
191    LevelID  := (LevelID * 2) * 256 * 256 * 256 + $01;
231  OniImage := TOniImage.Create;
192  
193    Self.Visible := True;
194    Form_Main.Visible := False;
# Line 243 | Line 203 | begin
203    Stream_Names := TMemoryStream.Create;
204    Stream_Dat   := TFileStream.Create(Target, fmCreate);
205    Stream_Raw   := TFileStream.Create(AnsiReplaceStr(Target, '.dat', '.raw'), fmCreate);
206 +  Stream_Raw.Write(EmptyBytes[0], 32);
207    if Connection.DataOS in [DOS_WINDEMO, DOS_MAC, DOS_MACBETA] then
208 +  begin
209      Stream_Sep := TFileStream.Create(AnsiReplaceStr(Target, '.dat', '.sep'), fmCreate);
210 +    Stream_Sep.Write(EmptyBytes[0], 32);
211 +  end;
212 +
213  
214    DoStep('Creating header');
215    progress.Position      := 0;
# Line 273 | Line 238 | begin
238    for i := 0 to High(DatHeader.OSIdent) do
239      case Connection.DataOS of
240        DOS_WIN: DatHeader.OSIdent[i] := HeaderOSIdentWin[i];
241 <      DOS_MAC: DatHeader.OSIdent[i] := HeaderOSIdentMac[i];
241 >      DOS_MAC, DOS_WINDEMO: DatHeader.OSIdent[i] := HeaderOSIdentMac[i];
242        DOS_MACBETA: DatHeader.OSIdent[i] := HeaderOSIdentMacBeta[i];
243      end;
244    for i := 0 to High(DatHeader.GlobalIdent) do
# Line 314 | Line 279 | begin
279      begin
280        case Connection.DataOS of
281          DOS_WIN:     ExtensionsHeader[i].Ident := FileTypes[j].IdentWin;
282 <        DOS_WINDEMO: ExtensionsHeader[i].Ident := FileTypes[j].IdentMac;
282 >        DOS_WINDEMO:
283 >            if FileTypes[j].Extension = 'SNDD' then
284 >              ExtensionsHeader[i].Ident := FileTypes[j].IdentWin
285 >            else
286 >              ExtensionsHeader[i].Ident := FileTypes[j].IdentMac;
287          DOS_MAC:     ExtensionsHeader[i].Ident := FileTypes[j].IdentMac;
288          DOS_MACBETA: ExtensionsHeader[i].Ident := FileTypes[j].IdentMac;
289        end;
# Line 346 | Line 315 | begin
315        FilesHeader[FileID].DataAddr := Stream_Body.Size + 8;
316        DatFileStream := TMemoryStream.Create;
317        Connection.LoadDatFile(FileID, TStream(DatFileStream));
318 <      DatFileStream.Seek(4, soFromBeginning);
318 >      DatFileStream.Seek(0, soFromBeginning);
319 >      tempi := FileID * 256 + 1;
320 >      DatFileStream.Write(tempi, 4);
321        DatFileStream.Write(LevelID, 4);
322  
323        DatLinks := Connection.GetDatLinks(FileID);
# Line 371 | Line 342 | begin
342            if RawLinks[i].RawSize > 0 then
343            begin
344              RawFileStream := TMemoryStream.Create;
345 <            if UpperCase(fileinfo.Extension) = 'TXMP' then
345 >            Connection.LoadRawFile(FileID, RawLinks[i].SrcOffset, TStream(RawFileStream));
346 >            RawFileStream.Seek(0, soFromBeginning);
347 >            if RawLinks[i].LocSep then
348              begin
349 <              DatFileStream.Seek($88, soFromBeginning);
350 <              DatFileStream.Read(tempb, 1);
351 <              if (tempb and $01) > 0 then
352 <              begin
380 <                OniImage.Load(Connection.ConnectionID, FileID);
381 <                if not OniImage.GetMipMappedImage(TStream(RawFileStream)) then
382 <                begin
383 <                  ShowMessage('MipMapping-Error');
384 <                  RawFileStream.Seek(0, soFromBeginning);
385 <                  tempb := tempb and $FE;
386 <                  DatFileStream.Seek($88, soFromBeginning);
387 <                  DatFileStream.Write(tempb, 1);
388 <                  OniImage.Load(Connection.ConnectionID, FileID);
389 <                  OniImage.GetAsData(TStream(RawFileStream));
390 <                end else
391 <                begin
392 <                  // Change of Depth($89), Storetype ($90)
393 <                  DatFileStream.Seek($89, soFromBeginning);
394 <                  DatFileStream.Read(tempb, 1);
395 <
396 <                  DatFileStream.Seek($90, soFromBeginning);
397 <                  DatFileStream.Read(tempb, 1);
398 <                  Exit;
399 <                end;
400 <              end else
401 <                OniImage.GetAsData(TStream(RawFileStream));
349 >              RawLinks[i].RawAddr := Stream_Sep.Size;
350 >              Stream_sep.CopyFrom(RawFileStream, RawFileStream.Size);
351 >              if (Stream_Sep.Size mod 32) > 0 then
352 >                Stream_Sep.Write(EmptyBytes[0], 32 - (Stream_Sep.Size mod 32));
353              end else begin
354 <              Connection.LoadRawFile(FileID, RawLinks[i].SrcOffset, TStream(RawFileStream));
354 >              RawLinks[i].RawAddr := Stream_Raw.Size;
355 >              Stream_Raw.CopyFrom(RawFileStream, RawFileStream.Size);
356 >              if (Stream_Raw.Size mod 32) > 0 then
357 >                Stream_Raw.Write(EmptyBytes[0], 32 - (Stream_Raw.Size mod 32));
358              end;
405            RawFileStream.Seek(0, soFromBeginning);
359            end else
360              RawLinks[i].RawAddr := 0;
361 +          DatFileStream.Seek(RawLinks[i].SrcOffset, soFromBeginning);
362 +          DatFileStream.Write(RawLinks[i].RawAddr, 4);
363          end;
364        end;
365 <
366 < {
367 <            if rawlist[j].raw_size > 0 then
368 <            begin
369 <              if (UpperCase(fileinfo.Extension) = 'TXMP') and
370 <                ((Data[$88] and $01) > 0) then
371 <              begin
372 <                OniImage.LoadFromTXMP(Connection, i);
373 <                OniImage.GetMipMappedImage(rawdata);
374 <                rawlist[j].raw_size := OniImage.GetImageDataSize(True);
420 <                Data[$90] := $08;
421 <                Data[$89] := 32;
422 < (*                  if data[$90]<>OniImage.StoreType then begin
423 <                    data[$90]:=OniImage.StoreType;
424 <                    data[$89]:=(data[$89] and $CF) or $20;
425 <                  end;
426 < *)                end
427 <              else
428 <              begin
429 <                SetLength(rawdata, rawlist[j].raw_size);
430 <                OniDataConnection.LoadRawFile(i, rawlist[j].src_offset, @rawdata[0]);
431 <              end;
432 <              //                data[$88]:=data[$88] and $FE;
433 <
434 <              if rawlist[j].loc_sep then
435 <              begin
436 <                rawlist[j].raw_addr := Stream_Sep.Size;
437 <                Stream_Sep.Write(rawdata[0], Length(rawdata));
438 <              end
439 <              else
440 <              begin
441 <                rawlist[j].raw_addr := Stream_Raw.Size;
442 <                Stream_Raw.Write(rawdata[0], Length(rawdata));
443 <              end;
444 <            end
445 <            else
446 <              rawlist[j].raw_addr := 0;
447 <            Data[rawlist[j].src_offset + 0] := (rawlist[j].raw_addr) and $FF;
448 <            Data[rawlist[j].src_offset + 1] := (rawlist[j].raw_addr shr 8) and $FF;
449 <            Data[rawlist[j].src_offset + 2] := (rawlist[j].raw_addr shr 16) and $FF;
450 <            Data[rawlist[j].src_offset + 3] := (rawlist[j].raw_addr shr 24) and $FF;
451 < }
452 <      Stream_Body.Write(Data[0], Length(Data));
453 <      //
365 >      DatFileStream.Seek(0, soFromBeginning);
366 >      Stream_Body.CopyFrom(DatFileStream, DatFileStream.Size);
367 >      if (Stream_Body.Size mod 32) > 0 then
368 >      begin
369 >        ShowMessage(
370 >            IntToStr(FileID) + '-' + FileInfo.Name + '.' + FileInfo.Extension + #13#10 +
371 >            IntToStr(FileInfo.Size) + ' - 0x' + IntToHex(FileInfo.Size, 6) + ' - real: ' + IntToStr(DatFileStream.Size) + ' - 0x' + IntToHex(DatFileStream.Size, 6) + #13#10 +
372 >            IntToStr(Stream_Body.Size) + ' - 0x' + IntToHex(Stream_Body.Size, 6) );
373 >        Stream_Body.Write(EmptyBytes[0], 32 - (Stream_Body.Size mod 32));
374 >      end;
375      end
376      else
377 <      FilesHeader[i].DataAddr := 0;
377 >      FilesHeader[FileID].DataAddr := 0;
378      if Length(fileinfo.Name) > 0 then
379      begin
380 <      FilesHeader[i].NameAddr := Stream_Names.Size;
380 >      FilesHeader[FileID].NameAddr := Stream_Names.Size;
381        temps := fileinfo.Extension + fileinfo.Name + Chr(0);
382        Stream_Names.Write(temps[1], Length(temps));
383      end
384      else
385 <      FilesHeader[i].NameAddr := 0;
386 <    FilesHeader[i].FileSize := fileinfo.Size;
387 <    FilesHeader[i].FileType := fileinfo.FileType;
385 >      FilesHeader[FileID].NameAddr := 0;
386 >    FilesHeader[FileID].FileSize := fileinfo.Size;
387 >    FilesHeader[FileID].FileType := fileinfo.FileType;
388  
389 <    if ((i mod 10) = 0) and (i >= 100) then
389 >    if ((FileID mod 10) = 0) and (FileID >= 100) then
390        lbl_estimation.Caption := 'Estimated time left: ' + TimeToStr(
391 <        (Time - FileTime) / i * (progress.Max - i + 1) * 1.1, TimeFormat );
392 <    progress.Position := i + 1;
393 <    lbl_progress.Caption := 'Files done: ' + IntToStr(i + 1) + '/' + IntToStr(progress.Max);
391 >        (Time - FileTime) / FileID * (progress.Max - FileID + 1) * 1.1, TimeFormat );
392 >    progress.Position := FileID + 1;
393 >    lbl_progress.Caption := 'Files done: ' + IntToStr(FileID + 1) + '/' + IntToStr(progress.Max);
394      Application.ProcessMessages;
395    end;
396  
# Line 481 | Line 402 | begin
402    for i := 0 to High(ExtensionsHeader) do
403      Stream_Dat.Write(ExtensionsHeader[i], SizeOf(ExtensionsHeader[i]));
404  
405 +  if (Stream_Dat.Size mod 32) > 0 then
406 +    Stream_Dat.Write(EmptyBytes[0], 32 - (Stream_Dat.Size mod 32));
407 +
408    DatHeader.DataSize  := Stream_Body.Size;
409    DatHeader.NamesSize := Stream_Names.Size;
410    DatHeader.DataAddr  := Stream_Dat.Size;
411 +
412    Stream_Body.Seek(0, soFromBeginning);
413    Stream_Dat.CopyFrom(Stream_Body, Stream_Body.Size);
414 +
415 +  if (Stream_Dat.Size mod 32) > 0 then
416 +    Stream_Dat.Write(EmptyBytes[0], 32 - (Stream_Dat.Size mod 32));
417 +
418    DatHeader.NamesAddr := Stream_Dat.Size;
419    Stream_Names.Seek(0, soFromBeginning);
420    Stream_Dat.CopyFrom(Stream_Names, Stream_Names.Size);
# Line 497 | Line 426 | begin
426    Stream_Body.Free;
427    Stream_Names.Free;
428    Stream_Raw.Free;
429 <  if OniDataConnection.OSisMac then
429 >
430 >  if Connection.DataOS in [DOS_WINDEMO, DOS_MAC, DOS_MACBETA] then
431      Stream_Sep.Free;
432  
433    progress.Position      := progress.Max;
# Line 509 | Line 439 | begin
439    btn_abortok.Caption := '&OK';
440    btn_abortok.Default := True;
441  
512  OniImage.Free;
513
442    converting := False;
443  
444 <  CloseDataConnection(DataConnections[conIndex]);
517 < end;
518 <
519 <
520 <
521 <
522 < procedure TForm_LevelDB.HandleFile;
523 < var
524 <  i: Byte;
525 < begin
526 <  for i := 1 to Length(ConvertHandlers) do
527 <    if UpperCase(ConvertHandlers[i].Ext) = UpperCase(ext) then
528 <      if ConvertHandlers[i].needed then
529 <      begin
530 <        ConvertHandlers[i].Handler(fileid, dir_dat2db);
531 <        Break;
532 <      end
533 <      else
534 <        Break;
444 > //  CloseDataConnection(DataConnections[conIndex]);
445   end;
446  
447  
# Line 539 | Line 449 | end;
449  
450   procedure TForm_LevelDB.CreateDatabase(Source, target: String);
451   var
452 <  DataBase:    TABSDatabase;
453 <  Query:       TABSQuery;
454 <  MimeCoder:   TStringFormat_MIME64;
545 <
546 <  i, j:     LongWord;
547 <  temps, temps2: String;
548 <  Data:     Tdata;
549 <  absolutebegintime, begintime: Double;
550 <  step:     Byte;
551 <  rawlist:  TRawList;
552 <  extlist:  TExtensionsMap;
553 <  fileinfo: TFileInfo;
554 <  timeformat: TFormatSettings;
452 >  DataBase:  TABSDatabase;
453 >  Query:     TABSQuery;
454 >  MimeCoder: TStringFormat_MIME64;
455  
456 <  conIndex: Integer;
457 < const
458 <  steps: Byte = 4;
456 >  BeginTime, FileTime: Double;
457 >  Step:       Integer;
458 >  TimeFormat: TFormatSettings;
459  
460 +  ConID:      Integer;
461 +  Connection: TDataAccess;
462 +  ConRepMsg:  TStatusMessages;
463  
464 +  FileID:     Integer;
465  
466 +  i:          Integer;
467 +  temps:      String;
468 +  tempdata:   TByteData;
469 +  FileInfo:   TFileInfo;
470 +  DatLinks:   TDatLinkList;
471 +  RawLinks:   TRawDataList;
472 + const
473 +  steps: Byte = 2;
474  
475    procedure DoStep(stepname: String);
476    begin
# Line 570 | Line 482 | const
482        group_progress.Caption := 'Creating DB (FINISHED)';
483    end;
484  
485 +  procedure StopConvert;
486 +  begin
487 +    btn_abortok.Caption := '&Close';
488 +    btn_abortok.Default := True;
489 +    converting := False;
490 +    lbl_estimation.Caption := 'ABORTED';
491 +    group_progress.Caption := 'Creating DB (ABORTED)';
492 +    DataBase.Close;
493 +    if MessageBox(Self.Handle, PChar('Delete the unfinished DB-file?'),
494 +      PChar('Delete file?'), MB_YESNO) = idYes then
495 +    begin
496 +      DeleteFile(target);
497 +    end;
498 +  end;
499 +
500 +
501 +
502   begin
503 <  if CreateDataConnection(Source, ODB_Dat) = nil then
503 >
504 >  //
505 >  // FILE EXISTS CHECK FÜR DAT/RAW/SEP!!!
506 >  //
507 >
508 >  TimeFormat.ShortTimeFormat := 'hh:nn:ss';
509 >  TimeFormat.LongTimeFormat  := 'hh:nn:ss';
510 >  TimeFormat.TimeSeparator   := ':';
511 >
512 >  ConID := ConManager.OpenConnection(Source, ConRepMsg);
513 >  if not (ConRepMsg in [SM_OK, SM_AlreadyOpened]) then
514    begin
515 <    ShowMessage('Could not connect to .dat-file');
515 >    ShowMessage('Source-file couldn''t be opened! Aborting' + CrLf + GetOpenMsg(ConRepMsg));
516      Exit;
517 <  end
518 <  else
517 >  end else
518 >    Connection := ConManager.Connection[ConID];
519 >
520 >  ConID := ConManager.FileOpened(Target);
521 >  if ConID >= 0 then
522    begin
523 <    TOniDataDat(OniDataConnection).UnloadWhenUnused := False;
523 >    if MessageBox(Self.Handle, PChar('Destination-file is opened, close it in ' +
524 >          'order to proceed conversion?'), PChar('Destination-file opened'),
525 >          MB_YESNO + MB_ICONQUESTION) = ID_YES then
526 >    begin
527 >      if Form_Main.CheckConnectionCloseable(ConID) then
528 >        if not ConManager.CloseConnection(ConID, ConRepMsg) then
529 >        begin
530 >          ShowMessage('Couldn''t close destination-file. Aborting');
531 >          Exit;
532 >        end;
533 >    end else begin
534 >      ShowMessage('Aborting');
535 >      Exit;
536 >    end;
537    end;
538  
539 <  timeformat.LongTimeFormat := 'hh:nn:ss';
540 <  timeformat.ShortTimeFormat := 'hh:nn:ss';
541 <  timeformat.TimeSeparator := ':';
539 >  if FileExists(Target) then
540 >  begin
541 >    if MessageBox(Self.Handle, PChar('Destination-file exists. ' +
542 >          'Overwrite it?'), PChar('Destination-file exists'),
543 >          MB_YESNO + MB_ICONWARNING) = ID_YES then
544 >    begin
545 >      if not DeleteFile(Target) then
546 >      begin
547 >        ShowMessage('Couldn''t delete file. Aborting');
548 >        Exit;
549 >      end;
550 >    end else begin
551 >      ShowMessage('Aborting');
552 >      Exit;
553 >    end;
554 >  end;
555  
556    Self.Visible := True;
557    Form_Main.Visible := False;
# Line 592 | Line 560 | begin
560    abort := False;
561    btn_abortok.Caption := '&Abort...';
562    btn_abortok.Default := False;
595  loaded_filename := target;
563  
564 <  absolutebegintime := Time;
564 >  BeginTime := Time;
565  
566    DataBase := TABSDatabase.Create(Self);
567 +  DataBase.MaxConnections := 1;
568 +  DataBase.PageSize := 8112;
569 +  DataBase.PageCountInExtent := 8;
570 +
571    DataBase.DatabaseName := 'OLDB';
572    DataBase.DatabaseFileName := target;
573    DataBase.CreateDatabase;
# Line 610 | Line 581 | begin
581    Query := TABSQuery.Create(Self);
582    Query.DatabaseName := 'OLDB';
583    Query.SQL.Text :=
584 <    'CREATE TABLE globals  ( id AUTOINC PRIMARY KEY, name STRING(128), value STRING(128) );';
584 >    'CREATE TABLE globals  ( id AUTOINC PRIMARY KEY, name STRING(128), ' +
585 >    'value STRING(128) );';
586    Query.ExecSQL;
587    Query.SQL.Text :=
588 <    'CREATE TABLE linkmap  ( id AUTOINC PRIMARY KEY, src_id INTEGER, src_link_offset INTEGER, target_id INTEGER );';
588 >    'CREATE TABLE linkmap  ( id AUTOINC PRIMARY KEY, src_id INTEGER, ' +
589 >    'src_link_offset INTEGER, target_id INTEGER);';
590 >  Query.ExecSQL;
591 >  Query.SQL.Text := 'CREATE INDEX idsrcid ON linkmap (src_id);';
592 >  Query.ExecSQL;
593 >  Query.SQL.Text := 'CREATE INDEX idtargetid ON linkmap (target_id);';
594    Query.ExecSQL;
595    Query.SQL.Text :=
596 <    'CREATE TABLE rawmap  ( id AUTOINC PRIMARY KEY, src_id INTEGER, src_link_offset INTEGER, sep BOOLEAN, size INTEGER, data BLOB BlobCompressionMode 9 BlobBlockSize 1024 BlobCompressionAlgorithm ZLib );';
596 >    'CREATE TABLE rawmap  ( id AUTOINC PRIMARY KEY, src_id INTEGER, ' +
597 >    'src_link_offset INTEGER, sep BOOLEAN, size INTEGER, ' +
598 >    'data BLOB BlobCompressionMode 9 BlobBlockSize 1024 BlobCompressionAlgorithm ZLib);';
599    //    Query.SQL.Text:='CREATE TABLE rawmap  ( id AUTOINC PRIMARY KEY, src_id INTEGER, src_link_offset INTEGER, size INTEGER, data BLOB BlobCompressionAlgorithm None );';
600    Query.ExecSQL;
601 <  Query.SQL.Text :=
623 <    'CREATE TABLE datfiles  ( id INTEGER PRIMARY KEY, extension CHAR(4), name STRING(128), contenttype INTEGER, size INTEGER, data BLOB BlobCompressionMode 9 BlobBlockSize 1024 BlobCompressionAlgorithm ZLib );';
624 <  //    Query.SQL.Text:='CREATE TABLE datfiles  ( id INTEGER PRIMARY KEY, extension CHAR(4), name STRING(128), contenttype INTEGER, size INTEGER, data BLOB BlobCompressionAlgorithm None );';
601 >  Query.SQL.Text := 'CREATE INDEX idsrcid ON rawmap (src_id);';
602    Query.ExecSQL;
603    Query.SQL.Text :=
604 <    'CREATE TABLE extlist  ( id AUTOINC PRIMARY KEY, ext CHAR(4), ident CHAR(16) );';
604 >    'CREATE TABLE datfiles  ( id INTEGER PRIMARY KEY, extension CHAR(4), ' +
605 >    'name STRING(128), contenttype INTEGER, size INTEGER, ' +
606 >    'data BLOB BlobCompressionMode 9 BlobBlockSize 1024 BlobCompressionAlgorithm ZLib );';
607 >  //    Query.SQL.Text:='CREATE TABLE datfiles  ( id INTEGER PRIMARY KEY, extension CHAR(4), name STRING(128), contenttype INTEGER, size INTEGER, data BLOB BlobCompressionAlgorithm None );';
608    Query.ExecSQL;
609 + //  Query.SQL.Text :=
610 + //    'CREATE TABLE extlist  ( id AUTOINC PRIMARY KEY, ext CHAR(4), ident CHAR(16) );';
611 + //  Query.ExecSQL;
612  
613    Query.SQL.Text := 'INSERT INTO globals (name,value) VALUES ("dbversion","' +
614      dbversion + '");';
615    Query.ExecSQL;
616 <  SetLength(Data, Length(OniDataConnection.LevelInfo.Ident));
634 <  for i := 0 to High(OniDataConnection.LevelInfo.Ident) do
635 <    Data[i] := OniDataConnection.LevelInfo.Ident[i];
636 <  temps := CreateHexString(Data, True);
637 <  Query.SQL.Text := 'INSERT INTO globals (name,value) VALUES ("ident","' + temps + '");';
638 <  Query.ExecSQL;
616 >
617    Query.SQL.Text := 'INSERT INTO globals (name,value) VALUES ("lvl","' +
618 <    IntToStr(OniDataConnection.LevelInfo.LevelNumber) + '");';
618 >    IntToStr(Connection.LevelNumber) + '");';
619    Query.ExecSQL;
620 <  if OniDataConnection.OSisMAC then
621 <    Query.SQL.Text := 'INSERT INTO globals (name,value) VALUES ("os","MAC");'
622 <  else
623 <    Query.SQL.Text := 'INSERT INTO globals (name,value) VALUES ("os","PC");';
624 <  Query.ExecSQL;
647 <
648 <  DoStep('Writing extensionslist');
649 <  progress.Max := Length(OniDataConnection.GetExtensionsList);
650 <  Application.ProcessMessages;
651 <
652 <  extlist := OniDataConnection.GetExtendedExtensionsList;
653 <  for i := 0 to High(extlist) do
654 <  begin
655 <    SetLength(Data, Length(extlist[i].Ident));
656 <    for j := 0 to High(extlist[i].Ident) do
657 <      Data[j] := extlist[i].Ident[j];
658 <    temps := CreateHexString(Data, True);
659 <    temps2 := extlist[i].Extension[3] + extlist[i].Extension[2] +
660 <      extlist[i].Extension[1] + extlist[i].Extension[0];
661 <    Query.SQL.Text := 'INSERT INTO extlist (ext,ident) VALUES ("' +
662 <      temps2 + '","' + temps + '");';
663 <    Query.ExecSQL;
664 <    progress.Position    := i;
665 <    lbl_progress.Caption := 'Extensions done: ' + IntToStr(i) + '/' +
666 <      IntToStr(Length(extlist));
667 <    Application.ProcessMessages;
668 <    if abort then
669 <    begin
670 <      stop_convert;
671 <      Exit;
672 <    end;
620 >  case Connection.DataOS of
621 >    DOS_WIN: temps := 'WIN';
622 >    DOS_WINDEMO: temps := 'WINDEMO';
623 >    DOS_MAC: temps := 'MAC';
624 >    DOS_MACBETA: temps := 'MACBETA';
625    end;
626 <  lbl_progress.Caption := '';
626 >  Query.SQL.Text := 'INSERT INTO globals (name,value) VALUES ("os","' + temps + '");';
627 >  Query.ExecSQL;
628  
629    progress.Position      := 0;
630    lbl_progress.Caption   := 'Files done: ' + IntToStr(0) + '/' + IntToStr(
631 <    OniDataConnection.GetFilesCount);
631 >    Connection.GetFileCount);
632    lbl_estimation.Caption := 'Estimated finishing time: unknown';
633  
634 <  DoStep('Loading .dat into memory');
682 <  Application.ProcessMessages;
683 <
684 <  progress.Max := OniDataConnection.GetFilesCount;
634 >  progress.Max := Connection.GetFileCount;
635    begintime    := Time;
636    DoStep('Writing .dat-fileslist');
637    Application.ProcessMessages;
638  
639 +  TAccess_OniArchive(Connection).UnloadWhenUnused := False;
640 +
641 +  FileTime := Time;
642    Database.StartTransaction;
643 <  for i := 0 to OniDataConnection.GetFilesCount - 1 do
643 >  for FileID := 0 to Connection.GetFileCount - 1 do
644    begin
645 <    fileinfo := OniDataConnection.GetFileInfo(i);
645 >    fileinfo := Connection.GetFileInfo(FileID);
646      if (fileinfo.FileType and $02) = 0 then
647      begin
648        mimecoder := TStringFormat_MIME64.Create;
649 <      Data      := OniDataConnection.LoadDatFile(i);
649 >      Connection.LoadDatFile(FileID, tempdata);
650        Query.SQL.Text :=
651          'INSERT INTO datfiles (id,extension,name,contenttype,size,data) VALUES (' +
652 <        IntToStr(i) + ',"' + fileinfo.Extension + '","' + fileinfo.Name + '","' + IntToHex(
652 >        IntToStr(FileID) + ',"' + fileinfo.Extension + '","' + fileinfo.Name + '","' + IntToHex(
653          fileinfo.FileType, 8) + '",' + IntToStr(fileinfo.Size) + ',MimeToBin("' +
654 <        MimeCoder.StrTo(@Data[0], Length(Data)) + '") );';
654 >        MimeCoder.StrTo(@tempdata[0], Length(tempdata)) + '") );';
655        Query.ExecSQL;
656        mimecoder.Free;
657  
658 <      rawlist := OniDataConnection.GetRawList(i);
659 <      if Length(rawlist) > 0 then
658 >      RawLinks := Connection.GetRawList(FileID);
659 >      if Length(RawLinks) > 0 then
660        begin
661 <        for j := 0 to High(rawlist) do
661 >        for i := 0 to High(RawLinks) do
662          begin
663 <          if rawlist[j].raw_size > 0 then
663 >          if RawLinks[i].RawSize > 0 then
664            begin
665 <            SetLength(Data, rawlist[j].raw_size);
666 <            OniDataConnection.LoadRawFile(i, rawlist[j].src_offset, Data);
665 >            SetLength(tempdata, RawLinks[i].RawSize);
666 >            Connection.LoadRawFile(FileID, RawLinks[i].SrcOffset, tempdata);
667              mimecoder      := TStringFormat_MIME64.Create;
668              Query.SQL.Text :=
669                'INSERT INTO rawmap (src_id,src_link_offset,sep,size,data) VALUES (' +
670 <              IntToStr(i) + ',' + IntToStr(rawlist[j].src_offset) + ',' + BoolToStr(
671 <              rawlist[j].loc_sep) + ',' + IntToStr(rawlist[j].raw_size) + ',MimeToBin("' +
672 <              MimeCoder.StrTo(@Data[0], rawlist[j].raw_size) + '") );';
670 >              IntToStr(FileID) + ', ' + IntToStr(RawLinks[i].SrcOffset) + ',' +
671 >              BoolToStr(RawLinks[i].LocSep) + ', ' +
672 >              IntToStr(RawLinks[i].RawSize) + ', ' +
673 >              'MimeToBin("' + MimeCoder.StrTo(@tempdata[0], RawLinks[i].RawSize) + '") );';
674              Query.ExecSQL;
675              mimecoder.Free;
676            end
# Line 724 | Line 678 | begin
678            begin
679              Query.SQL.Text :=
680                'INSERT INTO rawmap (src_id,src_link_offset,sep,size) VALUES (' +
681 <              IntToStr(i) + ',' + IntToStr(rawlist[j].src_offset) + ',' + BoolToStr(rawlist[j].loc_sep) + ',0);';
681 >              IntToStr(FileID) + ', ' + IntToStr(RawLinks[i].SrcOffset) + ', ' +
682 >              BoolToStr(RawLinks[i].LocSep) + ', 0);';
683              Query.ExecSQL;
684            end;
685          end;
686        end;
687  
688 <      HandleFile(fileinfo.Extension, i, True);
688 >      DatLinks := Connection.GetDatLinks(FileID);
689 >      if Length(DatLinks) > 0 then
690 >      begin
691 >        for i := 0 to High(DatLinks) do
692 >        begin
693 >          Query.SQL.Text :=
694 >            'INSERT INTO linkmap (src_id, src_link_offset, target_id) VALUES (' +
695 >            IntToStr(FileID) + ', ' + IntToStr(DatLinks[i].SrcOffset) + ', ' +
696 >            IntToStr(DatLinks[i].DestID) + ');';
697 >          Query.ExecSQL;
698 >        end;
699 >      end;
700      end
701      else
702      begin
703        Query.SQL.Text :=
704          'INSERT INTO datfiles (id,extension,name,contenttype,size) VALUES (' +
705 <        IntToStr(i) + ',"' + fileinfo.Extension + '","' + fileinfo.Name + '","' + IntToHex(
706 <        fileinfo.FileType, 8) + '",0);';
705 >        IntToStr(FileID) + ', "' + fileinfo.Extension + '", ' +
706 >        '"' + fileinfo.Name + '", "' + IntToHex(fileinfo.FileType, 8) + '", 0);';
707        Query.ExecSQL;
708      end;
709 <    if ((i mod 100) = 0) and (i > 0) then
709 >    if ((FileID mod 100) = 0) and (FileID > 0) then
710      begin
711        Database.Commit(False);
712        Database.StartTransaction;
713      end;
714 <    if ((i mod 10) = 0) and (i >= 100) then
714 >    if ((FileID mod 10) = 0) and (FileID >= 100) then
715        lbl_estimation.Caption := 'Estimated time left: ' + TimeToStr(
716 <        (Time - begintime) / i * (progress.Max - i + 1) * 1.1, timeformat );
717 <    progress.Position := i;
718 <    lbl_progress.Caption := 'Files done: ' + IntToStr(i) + '/' + IntToStr(progress.Max);
716 >        (Time - FileTime) / FileID * (progress.Max - FileID + 1) * 1.1, timeformat );
717 >    progress.Position := FileID;
718 >    lbl_progress.Caption := 'Files done: ' + IntToStr(FileID) + '/' + IntToStr(progress.Max);
719      Application.ProcessMessages;
720      if abort then
721      begin
722 <      stop_convert;
722 >      StopConvert;
723        Exit;
724      end;
725    end;
# Line 762 | Line 728 | begin
728    lbl_progress.Caption   := 'Files done: ' + IntToStr(progress.Max) + '/' +
729      IntToStr(progress.Max);
730  
731 <  lbl_estimation.Caption := 'FINISHED (duration: ' + TimeToStr(Time - absolutebegintime, timeformat) + ')';
731 >  lbl_estimation.Caption := 'FINISHED (duration: ' + TimeToStr(Time - BeginTime, timeformat) + ')';
732  
733    DoStep('FIN');
734    btn_abortok.Caption := '&OK';
735    btn_abortok.Default := True;
736  
737    converting := False;
738 +  TAccess_OniArchive(Connection).UnloadWhenUnused := True;
739  
740 <  database.Close;
741 <  database.Free;
775 <
776 <  CloseDataConnection(DataConnections[conIndex]);
777 < end;
778 <
779 <
780 <
781 <
782 < procedure TForm_LevelDB.stop_convert;
783 < begin
784 <  btn_abortok.Caption := '&Close';
785 <  btn_abortok.Default := True;
786 <  converting := False;
787 <  lbl_estimation.Caption := 'ABORTED';
788 <  group_progress.Caption := 'Creating DB (ABORTED)';
740 >  Query.Close;
741 >  Query.Free;
742    DataBase.Close;
743 <  if MessageBox(Self.Handle, PChar('Delete the unfinished DB-file?'),
791 <    PChar('Delete file?'), MB_YESNO) = idYes then
792 <  begin
793 <    DeleteFile(loaded_filename);
794 <  end;
743 >  DataBase.Free;
744   end;
745  
746  
# Line 814 | Line 763 | begin
763   end;
764  
765  
817
818
819 procedure InsertDatLinkToDB(fileid: LongWord; offset: LongWord);
820 var
821  link: LongWord;
822 begin
823  OniDataConnection.LoadDatFilePart(fileid, offset, 4, @link);
824  if link = 0 then
825    link := $FFFFFFFF
826  else
827    link := link div 256;
828  Query.SQL.Text := 'INSERT INTO linkmap (src_id,src_link_offset,target_id) VALUES (' +
829    IntToStr(fileid) + ',' + IntToStr(offset) + ',' + IntToStr(link) + ');';
830  Query.ExecSQL;
831 end;
832
833
834
835
836 procedure AISA(fileid: LongWord; dir_dat2db: Boolean);
837 var
838  packages: Word;
839  i: LongWord;
840 begin
841  if dir_dat2db then
842  begin
843    OniDataConnection.LoadDatFilePart(fileid, $1E, 2, @packages);
844    if packages > 0 then
845    begin
846      for i := 0 to packages - 1 do
847        InsertDatLinkToDB(fileid, $20 + i * $160 + $28);
848      for i := 0 to packages - 1 do
849        InsertDatLinkToDB(fileid, $20 + i * $160 + $150);
850    end;
851  end
852  else
853  begin
854  end;
855 end;
856
857
858
859
860 procedure AKEV(fileid: LongWord; dir_dat2db: Boolean);
861 var
862  i: LongWord;
863 begin
864  if dir_dat2db then
865  begin
866    for i := 0 to 16 do
867      InsertDatLinkToDB(fileid, $8 + i * 4);
868  end
869  else
870  begin
871  end;
872 end;
873
874
875
876
877 procedure AKOT(fileid: LongWord; dir_dat2db: Boolean);
878 var
879  i: LongWord;
880 begin
881  if dir_dat2db then
882  begin
883    for i := 0 to 4 do
884      InsertDatLinkToDB(fileid, $8 + i * 4);
885  end
886  else
887  begin
888  end;
889 end;
890
891
892
893
894 procedure CBPI(fileid: LongWord; dir_dat2db: Boolean);
895 var
896  i: LongWord;
897 begin
898  if dir_dat2db then
899  begin
900    for i := 0 to 56 do
901      InsertDatLinkToDB(fileid, $8 + i * 4);
902  end
903  else
904  begin
905  end;
906 end;
907
908
909
910
911 procedure CBPM(fileid: LongWord; dir_dat2db: Boolean);
912 var
913  i: LongWord;
914 begin
915  if dir_dat2db then
916  begin
917    for i := 0 to 18 do
918      InsertDatLinkToDB(fileid, $8 + i * 4);
919  end
920  else
921  begin
922  end;
923 end;
924
925
926
927
928 procedure CONS(fileid: LongWord; dir_dat2db: Boolean);
929 var
930  i: LongWord;
931 begin
932  if dir_dat2db then
933  begin
934    for i := 0 to 1 do
935      InsertDatLinkToDB(fileid, $24 + i * 4);
936  end
937  else
938  begin
939  end;
940 end;
941
942
943
944
945 procedure CRSA(fileid: LongWord; dir_dat2db: Boolean);
946 var
947  packages: LongWord;
948  i: LongWord;
949 begin
950  if dir_dat2db then
951  begin
952    OniDataConnection.LoadDatFilePart(fileid, $14, 4, @packages);
953    if packages > 0 then
954      for i := 0 to packages - 1 do
955        InsertDatLinkToDB(fileid, $20 + i * 1100 + $A0);
956  end
957  else
958  begin
959  end;
960 end;
961
962
963
964
965 procedure DOOR(fileid: LongWord; dir_dat2db: Boolean);
966 begin
967  if dir_dat2db then
968  begin
969    InsertDatLinkToDB(fileid, $08);
970    InsertDatLinkToDB(fileid, $10);
971  end
972  else
973  begin
974  end;
975 end;
976
977
978
979
980 procedure DPGE(fileid: LongWord; dir_dat2db: Boolean);
981 begin
982  if dir_dat2db then
983  begin
984    InsertDatLinkToDB(fileid, $40);
985  end
986  else
987  begin
988  end;
989 end;
990
991
992
993
994 procedure HPGE(fileid: LongWord; dir_dat2db: Boolean);
995 begin
996  if dir_dat2db then
997  begin
998    InsertDatLinkToDB(fileid, $0C);
999  end
1000  else
1001  begin
1002  end;
1003 end;
1004
1005
1006
1007
1008 procedure IGHH(fileid: LongWord; dir_dat2db: Boolean);
1009 begin
1010  if dir_dat2db then
1011  begin
1012    InsertDatLinkToDB(fileid, $24);
1013    InsertDatLinkToDB(fileid, $28);
1014  end
1015  else
1016  begin
1017  end;
1018 end;
1019
1020
1021
1022
1023 procedure IGPA(fileid: LongWord; dir_dat2db: Boolean);
1024 var
1025  links: LongWord;
1026  i:     LongWord;
1027 begin
1028  if dir_dat2db then
1029  begin
1030    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @links);
1031    if links > 0 then
1032      for i := 0 to links - 1 do
1033        InsertDatLinkToDB(fileid, $20 + i * 4);
1034  end
1035  else
1036  begin
1037  end;
1038 end;
1039
1040
1041
1042
1043 procedure IGPG(fileid: LongWord; dir_dat2db: Boolean);
1044 var
1045  i: LongWord;
1046 begin
1047  if dir_dat2db then
1048  begin
1049    for i := 0 to 1 do
1050      InsertDatLinkToDB(fileid, $1C + i * 4);
1051  end
1052  else
1053  begin
1054  end;
1055 end;
1056
1057
1058
1059
1060 procedure IGSA(fileid: LongWord; dir_dat2db: Boolean);
1061 var
1062  links: LongWord;
1063  i:     LongWord;
1064 begin
1065  if dir_dat2db then
1066  begin
1067    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @links);
1068    if links > 0 then
1069      for i := 0 to links - 1 do
1070        InsertDatLinkToDB(fileid, $20 + i * 4);
1071  end
1072  else
1073  begin
1074  end;
1075 end;
1076
1077
1078
1079
1080 procedure IMPT(fileid: LongWord; dir_dat2db: Boolean);
1081 begin
1082  if dir_dat2db then
1083  begin
1084    InsertDatLinkToDB(fileid, $10);
1085  end
1086  else
1087  begin
1088  end;
1089 end;
1090
1091
1092
1093
1094 procedure IPGE(fileid: LongWord; dir_dat2db: Boolean);
1095 begin
1096  if dir_dat2db then
1097  begin
1098    InsertDatLinkToDB(fileid, $0C);
1099  end
1100  else
1101  begin
1102  end;
1103 end;
1104
1105
1106
1107
1108 procedure KEYI(fileid: LongWord; dir_dat2db: Boolean);
1109 var
1110  i: LongWord;
1111 begin
1112  if dir_dat2db then
1113  begin
1114    for i := 0 to 9 do
1115      InsertDatLinkToDB(fileid, $08 + i * 4);
1116  end
1117  else
1118  begin
1119  end;
1120 end;
1121
1122
1123
1124
1125 procedure M3GA(fileid: LongWord; dir_dat2db: Boolean);
1126 var
1127  links: LongWord;
1128  i:     LongWord;
1129 begin
1130  if dir_dat2db then
1131  begin
1132    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @links);
1133    if links > 0 then
1134      for i := 0 to links - 1 do
1135        InsertDatLinkToDB(fileid, $20 + i * 4);
1136  end
1137  else
1138  begin
1139  end;
1140 end;
1141
1142
1143
1144
1145 procedure M3GM(fileid: LongWord; dir_dat2db: Boolean);
1146 var
1147  i: LongWord;
1148 begin
1149  if dir_dat2db then
1150  begin
1151    for i := 0 to 6 do
1152      InsertDatLinkToDB(fileid, $0C + i * 4);
1153  end
1154  else
1155  begin
1156  end;
1157 end;
1158
1159
1160
1161
1162 procedure MTRL(fileid: LongWord; dir_dat2db: Boolean);
1163 begin
1164  if dir_dat2db then
1165  begin
1166    InsertDatLinkToDB(fileid, $10);
1167  end
1168  else
1169  begin
1170  end;
1171 end;
1172
1173
1174
1175
1176 procedure OBDC(fileid: LongWord; dir_dat2db: Boolean);
1177 var
1178  packages: Word;
1179  i: LongWord;
1180 begin
1181  if dir_dat2db then
1182  begin
1183    OniDataConnection.LoadDatFilePart(fileid, $1E, 2, @packages);
1184    if packages > 0 then
1185      for i := 0 to packages - 1 do
1186        InsertDatLinkToDB(fileid, $20 + i * $18 + $4);
1187  end
1188  else
1189  begin
1190  end;
1191 end;
1192
1193
1194
1195
1196 procedure OBOA(fileid: LongWord; dir_dat2db: Boolean);
1197 var
1198  packages: Word;
1199  i: LongWord;
1200 begin
1201  if dir_dat2db then
1202  begin
1203    OniDataConnection.LoadDatFilePart(fileid, $1E, 2, @packages);
1204    if packages > 0 then
1205      for i := 0 to packages - 1 do
1206      begin
1207        InsertDatLinkToDB(fileid, $20 + i * 240 + $0);
1208        InsertDatLinkToDB(fileid, $20 + i * 240 + $4);
1209        InsertDatLinkToDB(fileid, $20 + i * 240 + $8);
1210      end;
1211  end
1212  else
1213  begin
1214  end;
1215 end;
1216
1217
1218
1219
1220 procedure OFGA(fileid: LongWord; dir_dat2db: Boolean);
1221 var
1222  packages: LongWord;
1223  i: LongWord;
1224 begin
1225  if dir_dat2db then
1226  begin
1227    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1228    if packages > 0 then
1229      for i := 0 to packages - 1 do
1230        InsertDatLinkToDB(fileid, $20 + i * 12 + $04);
1231  end
1232  else
1233  begin
1234  end;
1235 end;
1236
1237
1238
1239
1240 procedure ONCC(fileid: LongWord; dir_dat2db: Boolean);
1241 var
1242  i: LongWord;
1243 begin
1244  if dir_dat2db then
1245  begin
1246    InsertDatLinkToDB(fileid, $28);
1247    InsertDatLinkToDB(fileid, $434);
1248    InsertDatLinkToDB(fileid, $438);
1249    InsertDatLinkToDB(fileid, $43C);
1250    InsertDatLinkToDB(fileid, $C3C);
1251    InsertDatLinkToDB(fileid, $C40);
1252    InsertDatLinkToDB(fileid, $C44);
1253    InsertDatLinkToDB(fileid, $C48);
1254    InsertDatLinkToDB(fileid, $C88);
1255    InsertDatLinkToDB(fileid, $C8C);
1256  end
1257  else
1258  begin
1259  end;
1260 end;
1261
1262
1263
1264
1265 procedure ONCV(fileid: LongWord; dir_dat2db: Boolean);
1266 begin
1267  if dir_dat2db then
1268  begin
1269    InsertDatLinkToDB(fileid, $08);
1270  end
1271  else
1272  begin
1273  end;
1274 end;
1275
1276
1277
1278
1279 procedure ONLV(fileid: LongWord; dir_dat2db: Boolean);
1280 var
1281  i: LongWord;
1282 begin
1283  if dir_dat2db then
1284  begin
1285    for i := 0 to 5 do
1286      InsertDatLinkToDB(fileid, $48 + i * 4);
1287    for i := 0 to 5 do
1288      InsertDatLinkToDB(fileid, $64 + i * 4);
1289    InsertDatLinkToDB(fileid, $300);
1290  end
1291  else
1292  begin
1293  end;
1294 end;
1295
1296
1297
1298
1299 procedure ONOA(fileid: LongWord; dir_dat2db: Boolean);
1300 var
1301  packages: LongWord;
1302  i: LongWord;
1303 begin
1304  if dir_dat2db then
1305  begin
1306    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1307    if packages > 0 then
1308      for i := 0 to packages - 1 do
1309        InsertDatLinkToDB(fileid, $20 + i * 8 + $04);
1310  end
1311  else
1312  begin
1313  end;
1314 end;
1315
1316
1317
1318
1319 procedure ONSK(fileid: LongWord; dir_dat2db: Boolean);
1320 begin
1321  if dir_dat2db then
1322  begin
1323    InsertDatLinkToDB(fileid, $08);
1324    InsertDatLinkToDB(fileid, $0C);
1325    InsertDatLinkToDB(fileid, $10);
1326    InsertDatLinkToDB(fileid, $14);
1327    InsertDatLinkToDB(fileid, $18);
1328    InsertDatLinkToDB(fileid, $20);
1329    InsertDatLinkToDB(fileid, $44);
1330  end
1331  else
1332  begin
1333  end;
1334 end;
1335
1336
1337
1338
1339 procedure ONVL(fileid: LongWord; dir_dat2db: Boolean);
1340 var
1341  packages: LongWord;
1342  i: LongWord;
1343 begin
1344  if dir_dat2db then
1345  begin
1346    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1347    if packages > 0 then
1348      for i := 0 to packages - 1 do
1349        InsertDatLinkToDB(fileid, $20 + i * 4);
1350  end
1351  else
1352  begin
1353  end;
1354 end;
1355
1356
1357
1358
1359 procedure ONWC(fileid: LongWord; dir_dat2db: Boolean);
1360 begin
1361  if dir_dat2db then
1362  begin
1363    InsertDatLinkToDB(fileid, $28);
1364    InsertDatLinkToDB(fileid, $34);
1365    InsertDatLinkToDB(fileid, $40);
1366    InsertDatLinkToDB(fileid, $54);
1367    InsertDatLinkToDB(fileid, $58);
1368    InsertDatLinkToDB(fileid, $5C);
1369    InsertDatLinkToDB(fileid, $60);
1370    InsertDatLinkToDB(fileid, $6FC);
1371    InsertDatLinkToDB(fileid, $700);
1372  end
1373  else
1374  begin
1375  end;
1376 end;
1377
1378
1379
1380
1381 procedure OPGE(fileid: LongWord; dir_dat2db: Boolean);
1382 begin
1383  if dir_dat2db then
1384  begin
1385    InsertDatLinkToDB(fileid, $0C);
1386  end
1387  else
1388  begin
1389  end;
1390 end;
1391
1392
1393
1394
1395 procedure PSPC(fileid: LongWord; dir_dat2db: Boolean);
1396 begin
1397  if dir_dat2db then
1398  begin
1399    InsertDatLinkToDB(fileid, $50);
1400  end
1401  else
1402  begin
1403  end;
1404 end;
1405
1406
1407
1408
1409 procedure PSPL(fileid: LongWord; dir_dat2db: Boolean);
1410 var
1411  packages: LongWord;
1412  i: LongWord;
1413 begin
1414  if dir_dat2db then
1415  begin
1416    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1417    if packages > 0 then
1418      for i := 0 to packages - 1 do
1419        InsertDatLinkToDB(fileid, $20 + i * 8 + $4);
1420  end
1421  else
1422  begin
1423  end;
1424 end;
1425
1426
1427
1428
1429 procedure PSUI(fileid: LongWord; dir_dat2db: Boolean);
1430 var
1431  i: LongWord;
1432 begin
1433  if dir_dat2db then
1434  begin
1435    for i := 0 to 43 do
1436      InsertDatLinkToDB(fileid, $08 + i * 4);
1437  end
1438  else
1439  begin
1440  end;
1441 end;
1442
1443
1444
1445
1446 procedure STNA(fileid: LongWord; dir_dat2db: Boolean);
1447 var
1448  packages: Word;
1449  i: LongWord;
1450 begin
1451  if dir_dat2db then
1452  begin
1453    OniDataConnection.LoadDatFilePart(fileid, $1E, 2, @packages);
1454    if packages > 0 then
1455      for i := 0 to packages - 1 do
1456        InsertDatLinkToDB(fileid, $20 + i * 4);
1457  end
1458  else
1459  begin
1460  end;
1461 end;
1462
1463
1464
1465
1466 procedure TRAC(fileid: LongWord; dir_dat2db: Boolean);
1467 var
1468  packages: Word;
1469  i: LongWord;
1470 begin
1471  if dir_dat2db then
1472  begin
1473    InsertDatLinkToDB(fileid, $18);
1474    OniDataConnection.LoadDatFilePart(fileid, $1E, 2, @packages);
1475    if packages > 0 then
1476      for i := 0 to packages - 1 do
1477        InsertDatLinkToDB(fileid, $20 + i * 12 + 8);
1478  end
1479  else
1480  begin
1481  end;
1482 end;
1483
1484
1485
1486
1487 procedure TRAM(fileid: LongWord; dir_dat2db: Boolean);
1488 begin
1489  if dir_dat2db then
1490  begin
1491    InsertDatLinkToDB(fileid, $40);
1492    InsertDatLinkToDB(fileid, $44);
1493  end
1494  else
1495  begin
1496  end;
1497 end;
1498
1499
1500
1501
1502 procedure TRAS(fileid: LongWord; dir_dat2db: Boolean);
1503 begin
1504  if dir_dat2db then
1505  begin
1506    InsertDatLinkToDB(fileid, $08);
1507  end
1508  else
1509  begin
1510  end;
1511 end;
1512
1513
1514
1515
1516 procedure TRBS(fileid: LongWord; dir_dat2db: Boolean);
1517 var
1518  i: LongWord;
1519 begin
1520  if dir_dat2db then
1521  begin
1522    for i := 0 to 4 do
1523      InsertDatLinkToDB(fileid, $08 + i * 4);
1524  end
1525  else
1526  begin
1527  end;
1528 end;
1529
1530
1531
1532
1533 procedure TRCM(fileid: LongWord; dir_dat2db: Boolean);
1534 var
1535  i: LongWord;
1536 begin
1537  if dir_dat2db then
1538  begin
1539    for i := 0 to 2 do
1540      InsertDatLinkToDB(fileid, $5C + i * 4);
1541  end
1542  else
1543  begin
1544  end;
1545 end;
1546
1547
1548
1549
1550 procedure TRGA(fileid: LongWord; dir_dat2db: Boolean);
1551 var
1552  i: LongWord;
1553  packages: Word;
1554 begin
1555  if dir_dat2db then
1556  begin
1557    OniDataConnection.LoadDatFilePart(fileid, $1E, 2, @packages);
1558    if packages > 0 then
1559      for i := 0 to packages - 1 do
1560        InsertDatLinkToDB(fileid, $20 + i * 4);
1561  end
1562  else
1563  begin
1564  end;
1565 end;
1566
1567
1568
1569
1570 procedure TRGE(fileid: LongWord; dir_dat2db: Boolean);
1571 begin
1572  if dir_dat2db then
1573  begin
1574    InsertDatLinkToDB(fileid, $20);
1575  end
1576  else
1577  begin
1578  end;
1579 end;
1580
1581
1582
1583
1584 procedure TRIG(fileid: LongWord; dir_dat2db: Boolean);
1585 begin
1586  if dir_dat2db then
1587  begin
1588    InsertDatLinkToDB(fileid, $18);
1589    InsertDatLinkToDB(fileid, $24);
1590    InsertDatLinkToDB(fileid, $28);
1591  end
1592  else
1593  begin
1594  end;
1595 end;
1596
1597
1598
1599
1600 procedure TRMA(fileid: LongWord; dir_dat2db: Boolean);
1601 var
1602  i: LongWord;
1603  packages: Word;
1604 begin
1605  if dir_dat2db then
1606  begin
1607    OniDataConnection.LoadDatFilePart(fileid, $1E, 2, @packages);
1608    if packages > 0 then
1609      for i := 0 to packages - 1 do
1610        InsertDatLinkToDB(fileid, $20 + i * 4);
1611  end
1612  else
1613  begin
1614  end;
1615 end;
1616
1617
1618
1619
1620 procedure TRSC(fileid: LongWord; dir_dat2db: Boolean);
1621 var
1622  i: LongWord;
1623  packages: Word;
1624 begin
1625  if dir_dat2db then
1626  begin
1627    OniDataConnection.LoadDatFilePart(fileid, $1E, 2, @packages);
1628    if packages > 0 then
1629      for i := 0 to packages - 1 do
1630        InsertDatLinkToDB(fileid, $20 + i * 4);
1631  end
1632  else
1633  begin
1634  end;
1635 end;
1636
1637
1638
1639
1640 procedure TSFF(fileid: LongWord; dir_dat2db: Boolean);
1641 var
1642  i: LongWord;
1643  packages: LongWord;
1644 begin
1645  if dir_dat2db then
1646  begin
1647    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1648    if packages > 0 then
1649      for i := 0 to packages - 1 do
1650        InsertDatLinkToDB(fileid, $20 + i * 4);
1651  end
1652  else
1653  begin
1654  end;
1655 end;
1656
1657
1658
1659
1660 procedure TSFT(fileid: LongWord; dir_dat2db: Boolean);
1661 begin
1662  if dir_dat2db then
1663  begin
1664    InsertDatLinkToDB(fileid, $1C);
1665  end
1666  else
1667  begin
1668  end;
1669 end;
1670
1671
1672
1673
1674 procedure TURR(fileid: LongWord; dir_dat2db: Boolean);
1675 begin
1676  if dir_dat2db then
1677  begin
1678    InsertDatLinkToDB(fileid, $60);
1679    InsertDatLinkToDB(fileid, $6C);
1680    InsertDatLinkToDB(fileid, $74);
1681  end
1682  else
1683  begin
1684  end;
1685 end;
1686
1687
1688
1689
1690 procedure TXAN(fileid: LongWord; dir_dat2db: Boolean);
1691 var
1692  i: LongWord;
1693  packages: LongWord;
1694 begin
1695  if dir_dat2db then
1696  begin
1697    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1698    if packages > 0 then
1699      for i := 0 to packages - 1 do
1700        InsertDatLinkToDB(fileid, $20 + i * 4);
1701  end
1702  else
1703  begin
1704  end;
1705 end;
1706
1707
1708
1709
1710 procedure TXMA(fileid: LongWord; dir_dat2db: Boolean);
1711 var
1712  i: LongWord;
1713  packages: LongWord;
1714 begin
1715  if dir_dat2db then
1716  begin
1717    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1718    if packages > 0 then
1719      for i := 0 to packages - 1 do
1720        InsertDatLinkToDB(fileid, $20 + i * 4);
1721  end
1722  else
1723  begin
1724  end;
1725 end;
1726
1727
1728
1729
1730 procedure TXMB(fileid: LongWord; dir_dat2db: Boolean);
1731 var
1732  i: LongWord;
1733  packages: LongWord;
1734 begin
1735  if dir_dat2db then
1736  begin
1737    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1738    if packages > 0 then
1739      for i := 0 to packages - 1 do
1740        InsertDatLinkToDB(fileid, $20 + i * 4);
1741  end
1742  else
1743  begin
1744  end;
1745 end;
1746
1747
1748
1749
1750 procedure TXMP(fileid: LongWord; dir_dat2db: Boolean);
1751 begin
1752  if dir_dat2db then
1753  begin
1754    InsertDatLinkToDB(fileid, $94);
1755    InsertDatLinkToDB(fileid, $98);
1756  end
1757  else
1758  begin
1759  end;
1760 end;
1761
1762
1763
1764
1765 procedure TXTC(fileid: LongWord; dir_dat2db: Boolean);
1766 begin
1767  if dir_dat2db then
1768  begin
1769    InsertDatLinkToDB(fileid, $08);
1770  end
1771  else
1772  begin
1773  end;
1774 end;
1775
1776
1777
1778
1779 procedure WMCL(fileid: LongWord; dir_dat2db: Boolean);
1780 var
1781  i: LongWord;
1782  packages: LongWord;
1783 begin
1784  if dir_dat2db then
1785  begin
1786    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1787    if packages > 0 then
1788      for i := 0 to packages - 1 do
1789        InsertDatLinkToDB(fileid, $20 + i * 8 + $4);
1790  end
1791  else
1792  begin
1793  end;
1794 end;
1795
1796
1797
1798
1799 procedure WMDD(fileid: LongWord; dir_dat2db: Boolean);
1800 var
1801  i: LongWord;
1802  packages: LongWord;
1803 begin
1804  if dir_dat2db then
1805  begin
1806    OniDataConnection.LoadDatFilePart(fileid, $11C, 4, @packages);
1807    if packages > 0 then
1808      for i := 0 to packages - 1 do
1809        InsertDatLinkToDB(fileid, $120 + i * $124 + $114);
1810  end
1811  else
1812  begin
1813  end;
1814 end;
1815
1816
1817
1818
1819 procedure WMMB(fileid: LongWord; dir_dat2db: Boolean);
1820 var
1821  i: LongWord;
1822  packages: LongWord;
1823 begin
1824  if dir_dat2db then
1825  begin
1826    OniDataConnection.LoadDatFilePart(fileid, $1C, 4, @packages);
1827    if packages > 0 then
1828      for i := 0 to packages - 1 do
1829        InsertDatLinkToDB(fileid, $20 + i * 4);
1830  end
1831  else
1832  begin
1833  end;
1834 end;
1835
1836
1837
1838
1839 procedure WPGE(fileid: LongWord; dir_dat2db: Boolean);
1840 begin
1841  if dir_dat2db then
1842  begin
1843    InsertDatLinkToDB(fileid, $08);
1844    InsertDatLinkToDB(fileid, $0C);
1845  end
1846  else
1847  begin
1848  end;
1849 end;
1850
1851
1852
1853
1854 procedure InsertHandler(ext: String; needed: Boolean; handler: THandler);
1855 begin
1856  SetLength(ConvertHandlers, Length(ConvertHandlers) + 1);
1857  ConvertHandlers[High(ConvertHandlers)].Ext     := ext;
1858  ConvertHandlers[High(ConvertHandlers)].needed  := needed;
1859  ConvertHandlers[High(ConvertHandlers)].handler := handler;
1860 end;
1861
1862 begin
1863  InsertHandler('ABNA', False, nil);
1864  //  InsertHandler('AGDB',True,AGDB);
1865  InsertHandler('AGDB', False, nil);
1866  InsertHandler('AGQC', False, nil);
1867  InsertHandler('AGQG', False, nil);
1868  InsertHandler('AGQR', False, nil);
1869  InsertHandler('AISA', True, AISA);
1870  InsertHandler('AITR', False, nil);
1871  InsertHandler('AKAA', False, nil);
1872  InsertHandler('AKBA', False, nil);
1873  InsertHandler('AKBP', False, nil);
1874  InsertHandler('AKDA', False, nil);
1875  InsertHandler('AKEV', True, AKEV);
1876  InsertHandler('AKOT', True, AKOT);
1877  InsertHandler('AKVA', False, nil);
1878  InsertHandler('BINA', False, nil);
1879  InsertHandler('CBPI', True, CBPI);
1880  InsertHandler('CBPM', True, CBPM);
1881  InsertHandler('CONS', True, CONS);
1882  InsertHandler('CRSA', True, CRSA);
1883  InsertHandler('DOOR', True, DOOR);
1884  InsertHandler('DPGE', True, DPGE);
1885  InsertHandler('ENVP', False, nil);
1886  InsertHandler('FILM', False, nil);
1887  InsertHandler('HPGE', True, HPGE);
1888  InsertHandler('IDXA', False, nil);
1889  InsertHandler('IGHH', True, IGHH);
1890  InsertHandler('IGPA', True, IGPA);
1891  InsertHandler('IGPG', True, IGPG);
1892  InsertHandler('IGSA', True, IGSA);
1893  InsertHandler('IMPT', True, IMPT);
1894  InsertHandler('IPGE', True, IPGE);
1895  InsertHandler('KEYI', True, KEYI);
1896  InsertHandler('M3GA', True, M3GA);
1897  InsertHandler('M3GM', True, M3GM);
1898  InsertHandler('MTRL', True, MTRL);
1899  InsertHandler('OBAN', False, nil);
1900  InsertHandler('OBDC', True, OBDC);
1901  InsertHandler('OBOA', True, OBOA);
1902  InsertHandler('OFGA', True, OFGA);
1903  InsertHandler('ONCC', True, ONCC);
1904  InsertHandler('ONCP', False, nil);
1905  InsertHandler('ONCV', True, ONCV);
1906  InsertHandler('ONFA', False, nil);
1907  InsertHandler('ONGS', False, nil);
1908  InsertHandler('ONIA', False, nil);
1909  InsertHandler('ONLD', False, nil);
1910  InsertHandler('ONLV', True, ONLV);
1911  InsertHandler('ONMA', False, nil);
1912  InsertHandler('ONOA', True, ONOA);
1913  InsertHandler('ONSA', False, nil);
1914  InsertHandler('ONSK', True, ONSK);
1915  InsertHandler('ONTA', False, nil);
1916  InsertHandler('ONVL', True, ONVL);
1917  InsertHandler('ONWC', True, ONWC);
1918  InsertHandler('OPGE', True, OPGE);
1919  InsertHandler('OSBD', False, nil);
1920  InsertHandler('OTIT', False, nil);
1921  InsertHandler('OTLF', False, nil);
1922  InsertHandler('PLEA', False, nil);
1923  InsertHandler('PNTA', False, nil);
1924  InsertHandler('PSPC', True, PSPC);
1925  InsertHandler('PSPL', True, PSPL);
1926  InsertHandler('PSUI', True, PSUI);
1927  InsertHandler('QTNA', False, nil);
1928  InsertHandler('SNDD', False, nil);
1929  InsertHandler('STNA', True, STNA);
1930  InsertHandler('SUBT', False, nil);
1931  InsertHandler('TRAC', True, TRAC);
1932  InsertHandler('TRAM', True, TRAM);
1933  InsertHandler('TRAS', True, TRAS);
1934  InsertHandler('TRBS', True, TRBS);
1935  InsertHandler('TRCM', True, TRCM);
1936  InsertHandler('TRGA', True, TRGA);
1937  InsertHandler('TRGE', True, TRGE);
1938  InsertHandler('TRIA', False, nil);
1939  InsertHandler('TRIG', True, TRIG);
1940  InsertHandler('TRMA', True, TRMA);
1941  InsertHandler('TRSC', True, TRSC);
1942  InsertHandler('TRTA', False, nil);
1943  InsertHandler('TSFF', True, TSFF);
1944  InsertHandler('TSFL', False, nil);
1945  InsertHandler('TSFT', True, TSFT);
1946  InsertHandler('TSGA', False, nil);
1947  InsertHandler('TSTR', False, nil);
1948  InsertHandler('TURR', True, TURR);
1949  InsertHandler('TXAN', True, TXAN);
1950  InsertHandler('TXCA', False, nil);
1951  InsertHandler('TXMA', True, TXMA);
1952  InsertHandler('TXMB', True, TXMB);
1953  InsertHandler('TXMP', True, TXMP);
1954  InsertHandler('TXTC', True, TXTC);
1955  InsertHandler('VCRA', False, nil);
1956  InsertHandler('WMCL', True, WMCL);
1957  InsertHandler('WMDD', True, WMDD);
1958  InsertHandler('WMM_', False, nil);
1959  InsertHandler('WMMB', True, WMMB);
1960  InsertHandler('WPGE', True, WPGE);
766   end.

Diff Legend

Removed lines
+ Added lines
< Changed lines (old)
> Changed lines (new)