home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 9 / 09.iso / l / l040 / 1.ddi / OWLGREP.PAS < prev    next >
Encoding:
Pascal/Delphi Source File  |  1992-10-27  |  12.0 KB  |  472 lines

  1. {************************************************}
  2. {                                                }
  3. {   ObjectWindows Grep Demo                      }
  4. {   Copyright (c) 1992 by Borland International  }
  5. {                                                }
  6. {************************************************}
  7.  
  8. { The grep text search programs consist of 3 binary files:
  9.  
  10.     OWLGREP.EXE   - Windows grep program (uses ObjectWindows)
  11.     TVGREP.EXE    - DOS text mode grep program (uses Turbo Vision)
  12.     REGEXP.DLL    - Text search engine dynamic link library (DLL)
  13.                     that is written in Borland C++ 3.1 and
  14.                     shared by both TVGREP.EXE and OWLGREP.EXE
  15.  
  16.   OWLGREP is a Windows application. To build it, set Compile|Target to
  17.   Windows from inside the IDE or type the following command-line at a
  18.   DOS prompt:
  19.  
  20.     bpc /m /cw owlgrep
  21.  
  22.   TVGREP is a DOS protected mode application (DPMI). To build
  23.   it, set Compile|Target to Protected from inside the IDE or type the
  24.   following command-line at a DOS prompt:
  25.  
  26.     bpc /m /cp tvgrep
  27.  
  28.   REGEXP.DLL is a Windows format DLL and comes already built. To rebuild
  29.   it, make sure Borland C++ 3.1 is on your DOS path, change to the
  30.   \BP\EXAMPLES\GREP\DLL directory and then run MAKE.
  31.  
  32. }
  33.  
  34. program OwlGrep;
  35.  
  36. {$M 20000,8192}
  37.  
  38. uses Strings, WinCrt, WinDos, WinProcs, WinTypes, Objects,
  39.   OWindows, ODialogs, RegExp, OGConst;
  40.  
  41. {$R OwlGrep}
  42.  
  43. const
  44.   Profile = 'OWLGREP.INI';
  45.  
  46. const
  47.   wm_GetPrivateStrings = wm_User;
  48.  
  49.  
  50.   rgb_Yellow      = $0000FFFF;
  51.   rgb_DarkYellow  = $0000C1FF;
  52.   rgb_Blue        = $00FF0000;
  53.   rgb_Red         = $000000FF;
  54.   rgb_Green       = $0000FF00;
  55.   rgb_DarkGreen   = $00007F00;
  56.  
  57. type
  58.   TRequest = record
  59.     Expression: array[0..80] of Char;
  60.     FileMask: array[0..40] of Char;
  61.     StartDir: array[0..128] of Char;
  62.     CaseSensitive,
  63.     UseSubDirs: Boolean;
  64.   end;
  65.  
  66.   PGrep = ^TGrep;
  67.   TGrep = object(TDlgWindow)
  68.     BoxBrush,
  69.     TheBrush: HBrush;
  70.     FileList: PListBox;
  71.     CaseCheck,
  72.     SubDirCheck: PCheckBox;
  73.     StatusText: PStatic;
  74.     FileMask,
  75.     Directory,
  76.     Expression: PEdit;
  77.     FileColl: PStrCollection;
  78.     constructor Init(P: PWindowsObject; N: PChar);
  79.     procedure InitControls;
  80.     destructor Done; virtual;
  81.     procedure SetupWindow; virtual;
  82.     procedure GetWindowClass(var WndClass: TWndClass); virtual;
  83.     function GetClassName: PChar; virtual;
  84.     function RunSearch: Boolean;
  85.     procedure FillBox;
  86.     procedure GetControlData;
  87.     procedure SetControlData;
  88.     procedure ReadProfile;
  89.     procedure WriteProfile;
  90.     procedure ListClick(var Msg: TMessage);
  91.       virtual id_First + idFileList;
  92.     procedure BeginSearch(var Msg: TMessage);
  93.       virtual id_First + idBeginSearch;
  94.     procedure WMControlColor(var Msg: TMessage);
  95.       virtual wm_First + wm_CtlColor;
  96.   end;
  97.  
  98.   TWhereApp = object(TApplication)
  99.     procedure InitMainWindow; virtual;
  100.   end;
  101.  
  102. var
  103.   Request: TRequest;
  104.  
  105. procedure ACenterDlg(HWindow: HWnd);
  106. var
  107.   R: TRect;
  108.   StartX,
  109.   StartY : Integer;
  110.   Frame,
  111.   Caption: Integer;
  112. begin
  113.   Frame := GetSystemMetrics(sm_CxFrame) * 2;
  114.   Caption := GetSystemMetrics(sm_CyCaption);
  115.   GetClientRect(HWindow, R);
  116.   StartX := ((GetSystemMetrics(sm_CxScreen) - (R.Right - R.Left)) div 2);
  117.   StartY := ((GetSystemMetrics(sm_CyScreen) - (R.Bottom - R.Top)) div 2);
  118.   MoveWindow(HWindow, StartX, StartY - ((Caption + Frame) div 2),
  119.     R.Right + Frame, R.Bottom + Frame + Caption, False);
  120. end;
  121.  
  122. { TGrep }
  123.  
  124. constructor TGrep.Init(P: PWindowsObject; N: PChar);
  125. begin
  126.   inherited Init(P, N);
  127.   InitControls;
  128.   FileColl := nil;
  129. end;
  130.  
  131. procedure TGrep.InitControls;
  132. var
  133.   LogBrush: TLogBrush;
  134. begin
  135.   FileList := New(PListBox, InitResource(@Self, idFileList));
  136.   Directory := New(PEdit, InitResource(@Self, idDirectory, 80));
  137.   FileMask := New(PEdit, InitResource(@Self, idFileMask, 80));
  138.   Expression := New(PEdit, InitResource(@Self, idExpression, 80));
  139.  
  140.   StatusText := New(PStatic, InitResource(@Self, idStatus,80));
  141.  
  142.   CaseCheck := New(PCheckBox, InitResource(@Self, idCaseSensitive));
  143.   SubDirCheck := New(PCheckBox, InitResource(@Self, idRecurseDirs));
  144.  
  145.   LogBrush.lbStyle := bs_Solid;
  146.   LogBrush.lbColor := RGB(128,128,128);
  147.   LogBrush.lbHatch := bs_Solid;
  148.   TheBrush  := CreateBrushIndirect(LogBrush);
  149.  
  150.   LogBrush.lbStyle := bs_Solid;
  151.   LogBrush.lbColor := RGB(0,0,64);
  152.   LogBrush.lbHatch := bs_Solid;
  153.   BoxBrush := CreateBrushIndirect(LogBrush);
  154. end;
  155.  
  156. destructor TGrep.Done;
  157. begin
  158.   WriteProfile;
  159.   DeleteObject(TheBrush);
  160.   DeleteObject(BoxBrush);
  161.   inherited Done;
  162. end;
  163.  
  164. function TGrep.RunSearch: Boolean;
  165. var
  166.   RegExp: HRegExp;
  167.   Error: Integer;
  168.   P: PChar;
  169.  
  170.   function Search(Filename: PChar): Boolean;
  171.   var
  172.     TextFile: Text;
  173.     Line: array[0..255] of Char;
  174.     Match: TRegMatch;
  175.   begin
  176.     Search := False;
  177.     StatusText^.SetText(Filename);
  178.     Assign(TextFile, Filename);
  179.     {$I-}
  180.     Reset(TextFile);
  181.     while not Eof(TextFile) do
  182.     begin
  183.       Readln(TextFile, Line);
  184.       if not Request.CaseSensitive then StrUpper(Line);
  185.       if RegExec(RegExp, Line, Match) = 0 then
  186.       begin
  187.         Search := True;
  188.         Break;
  189.       end;
  190.     end;
  191.     Close(TextFile);
  192.   end;
  193.  
  194.   procedure SearchDir(Dir: PChar);
  195.   var
  196.     SR: TSearchRec;
  197.     NewDir: array[0..255] of Char;
  198.   begin
  199.     with Request do
  200.     begin
  201.       StrCopy(StrECopy(NewDir, Dir), FileMask);
  202.       FindFirst(NewDir, faArchive, SR);
  203.       while DosError = 0 do
  204.       begin
  205.         StrCopy(StrECopy(NewDir, Dir), SR.Name);
  206.         if Search(NewDir) then
  207.           FileColl^.Insert(StrNew(NewDir));
  208.         FindNext(SR);
  209.       end;
  210.  
  211.       if Request.UseSubDirs then
  212.       begin
  213.         StrCopy(StrECopy(NewDir, Dir), '*.*');
  214.         FindFirst(NewDir, faDirectory, SR);
  215.         while DosError = 0 do
  216.         begin
  217.           if (SR.Attr and faDirectory <> 0) and (SR.Name[0] <> '.') then
  218.           begin
  219.             StrCopy(StrECopy(StrECopy(NewDir, Dir), SR.Name), '\'); 
  220.             SearchDir(NewDir);
  221.           end;
  222.           FindNext(SR);
  223.         end;
  224.       end;
  225.     end;
  226.   end;
  227.  
  228.   procedure CompileExp;
  229.   var
  230.     Exp: array[0..SizeOf(Request.Expression)] of Char;
  231.   begin
  232.     StrCopy(Exp, Request.Expression);
  233.     if not Request.CaseSensitive then StrUpper(Exp);
  234.     RegExp := RegComp(Exp, Error);
  235.   end;
  236.     
  237. begin
  238.   CompileExp;
  239.   with Request do
  240.   begin
  241.     P := StrEnd(Request.StartDir);
  242.  
  243.     { Force a trailing back-slash }
  244.     if ((P - StartDir > 2) or (StartDir[1] <> ':')) and
  245.         ((P - 1)^ <> '\') then
  246.       StrCopy(P, '\');
  247.   end;
  248.   SearchDir(Request.StartDir);
  249.   P^ := #0; { Undo the backslash }
  250.  
  251.   RegFree(RegExp);
  252. end;
  253.  
  254. procedure TGrep.ReadProfile;
  255. begin
  256.   GetPrivateProfileString('OwlGrep', 'Expression', '',
  257.     Request.Expression, SizeOf(Request.Expression), Profile);
  258.   GetPrivateProfileString('OwlGrep','FileMask','*.PAS',
  259.     Request.FileMask, SizeOf(Request.FileMask), Profile);
  260.   GetCurDir(Request.StartDir, 0);
  261.  
  262.   Request.CaseSensitive := Bool(GetPrivateProfileInt('OwlGrep',
  263.     'CaseSensitive', 1, Profile));
  264.   Request.UseSubDirs := Bool(GetPrivateProfileInt('OwlGrep',
  265.     'SubDirs', 1, Profile));
  266. end;
  267.  
  268. procedure TGrep.WriteProfile;
  269. var
  270.   S: array[0..10] of Char;
  271. begin
  272.   WritePrivateProfileString('OwlGrep', 'Expression', Request.Expression,
  273.     Profile);
  274.   WritePrivateProfileString('OwlGrep', 'FileMask', Request.FileMask,
  275.     Profile);
  276.  
  277.   Str(Integer(Request.CaseSensitive), S);
  278.   WritePrivateProfileString('OwlGrep', 'CaseSensitive', S, Profile);
  279.   Str(Integer(Request.UseSubDirs), S);
  280.   WritePrivateProfileString('OwlGrep', 'SubDirs', S, Profile);
  281. end;
  282.  
  283. procedure TGrep.SetControlData;
  284. begin
  285.   Expression^.SetText(Request.Expression);
  286.   FileMask^.SetText(Request.FileMask);
  287.   Directory^.SetText(Request.StartDir);
  288.  
  289.   CaseCheck^.SetCheck(Integer(Request.CaseSensitive));
  290.   SubDirCheck^.SetCheck(Integer(Request.UseSubDirs));
  291. end;
  292.  
  293. procedure TGrep.GetControlData;
  294. begin
  295.   Expression^.GetText(Request.Expression, SizeOf(Request.Expression));
  296.   FileMask^.GetText(Request.FileMask, SizeOf(Request.FileMask));
  297.   Directory^.GetText(Request.StartDir, SizeOf(Request.StartDir) - 1);
  298.  
  299.   Request.CaseSensitive := CaseCheck^.GetCheck = 1;
  300.   Request.UseSubDirs := SubDirCheck^.GetCheck = 1;
  301. end;
  302.  
  303. procedure TGrep.SetupWindow;
  304. var
  305.   Msg: TMessage;
  306. begin
  307.   inherited SetUpWindow;
  308.  
  309.   ACenterDlg(HWindow);
  310.   ReadProfile;
  311.   SetControlData;
  312. end;
  313.  
  314. procedure TGrep.WMControlColor(var Msg: TMessage);
  315. begin
  316.   case Msg.LParamHi of
  317.     ctlColor_Btn:
  318.       begin
  319.         SetTextColor(Msg.WParam, Rgb_Blue);
  320.         SetBkMode(Msg.WParam, transparent);
  321.         Msg.Result := TheBrush;
  322.       end;
  323.     ctlColor_Static:
  324.       begin
  325.         SetTextColor(Msg.WParam, Rgb_Blue);
  326.         SetBkMode(Msg.WParam, transparent);
  327.         Msg.Result := TheBrush;
  328.       end;
  329.     ctlColor_ListBox, ctlColor_Edit:
  330.       begin
  331.         SetTextColor(Msg.WParam, Rgb_DarkYellow);
  332.         SetBkMode(Msg.WParam, transparent);
  333.         Msg.Result := BoxBrush;
  334.       end;
  335.     ctlcolor_Dlg:
  336.       begin
  337.         SetBkMode(Msg.WParam, Transparent);
  338.         Msg.Result := TheBrush;
  339.       end;
  340.   else
  341.     DefWndProc(Msg);
  342.   end;
  343. end;
  344.  
  345. procedure TGrep.FillBox;
  346. var
  347.   i: Integer;
  348. begin
  349.   FileList^.ClearList;
  350.   if FileColl = nil then Exit;
  351.   for i := 0 to FileColl^.Count - 1 do
  352.     FileList^.AddString(FileColl^.At(i));
  353. end;
  354.  
  355. procedure TGrep.ListClick(var Msg: TMessage);
  356. var
  357.   Choice: Integer;
  358.   Selection: PChar;
  359.   Cmd: array[0..300] of Char;
  360. begin
  361.   if Msg.LParamHi = lbn_DblClk then
  362.   begin
  363.     Choice := FileList^.GetSelIndex;
  364.     Selection := FileColl^.At(Choice);
  365.     StrCopy(StrECopy(Cmd, 'NotePad.Exe '), Selection);
  366.     WinExec(Cmd, sw_ShowNormal);
  367.   end;
  368. end;
  369.  
  370. procedure TGrep.BeginSearch(var Msg: TMessage);
  371. var
  372.   S: array[0..255] of Char;
  373.  
  374.   function DirOk(StartDir: PChar): Boolean;
  375.   var
  376.     P: PChar;
  377.   begin
  378.     StrUpper(StartDir);
  379.     { Strip trailing backslash }
  380.     SetCurDir(StartDir);
  381.     if (DosError <> 0) or (StrLen(StartDir) = 0) then
  382.     begin
  383.       StrCopy(StrECopy(S, 'Could not find directory: '), StartDir);
  384.       MessageBox(HWindow, S, 'Notice',mb_Ok);
  385.       SetFocus(Directory^.HWindow);
  386.       DirOk := False;
  387.     end;
  388.   end;
  389.  
  390.   function FileMaskOk(Mask: PChar): Boolean;
  391.   begin
  392.     if Mask[0] = #0 then
  393.     begin
  394.       FileMaskOk := False;
  395.       MessageBox(HWindow, 'You must provide a file mask. For Instance: *.*',
  396.         'Notice', mb_Ok);
  397.       SetFocus(FileMask^.HWindow);
  398.     end
  399.     else FileMaskOk := True;
  400.   end;
  401.  
  402.   function KeyOk(SrchStr: PChar): Boolean;
  403.   var
  404.     RegExp: HRegExp;
  405.     Error: Integer;
  406.     P: PChar;
  407.   begin
  408.     P := nil;
  409.     if SrchStr[0] = #0 then
  410.       P := 'Search Key can''t be empty'
  411.     else
  412.     begin
  413.       RegExp := RegComp(SrchStr, Error);
  414.       if RegExp = 0 then
  415.         P := 'Invalid regular expression'
  416.       else
  417.         RegFree(RegExp);
  418.     end;
  419.  
  420.     if P <> nil then
  421.     begin
  422.       KeyOk := False;
  423.       MessageBox(HWindow, P, 'Notice', mb_Ok);
  424.       SetFocus(Expression^.HWindow);
  425.     end
  426.     else
  427.       KeyOk := True;
  428.   end;
  429.  
  430. begin
  431.   if FileColl <> nil then Dispose(FileColl, Done);
  432.   FileColl := New(PStrCollection, Init(100, 50));
  433.   GetControlData;
  434.   if not DirOk(Request.StartDir) then Exit;
  435.   if not FileMaskOk(Request.FileMask) then Exit;
  436.   if not KeyOk(Request.Expression) then Exit;
  437.   SetControlData;
  438.  
  439.   StatusText^.SetText('Searching...');
  440.   RunSearch;
  441.   FillBox;
  442.   WVSPrintF(S, 'Number found: %d', FileColl^.Count);
  443.   StatusText^.SetText(S);
  444. end;
  445.  
  446. procedure TGrep.GetWindowClass(var WndClass: TWndClass);
  447. begin
  448.   inherited GetWindowClass(WndClass);
  449.   WndClass.hIcon := LoadIcon(hInstance,'WhereIcon');
  450. end;
  451.  
  452. function TGrep.GetClassName : PChar;
  453. begin
  454.   GetClassName := 'TGrep';
  455. end;
  456.  
  457. { TWhereApp }
  458.  
  459. procedure TWhereApp.InitMainWindow;
  460. begin
  461.   MainWindow := New(PGrep, Init(nil, 'GrepDlg'));
  462. end;
  463.  
  464. var
  465.   WhereApp: TWhereApp;
  466.  
  467. begin
  468.   WhereApp.Init('Search');
  469.   WhereApp.Run;
  470.   WhereApp.Done;
  471. end.
  472.