| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633 | {    This file is part of the Free Pascal Integrated Development Environment    Copyright (c) 1998 by Berczi Gabor    Code Complete routines    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. **********************************************************************}unit FPCodCmp; { CodeComplete }interfaceuses Objects,Drivers,Dialogs,     WEditor,WUtils,WViews;type     PCodeCompleteWordList = ^TCodeCompleteWordList;     TCodeCompleteWordList = object(TTextCollection)     end;    PCodeCompleteDialog = ^TCodeCompleteDialog;    TCodeCompleteDialog = object(TCenterDialog)      constructor Init;      function    Execute: Word; virtual;      procedure   HandleEvent(var Event: TEvent); virtual;    private      CodeCompleteLB : PAdvancedListBox;      RB : PRadioButtons;      CB : PCheckBoxes;      MinInputL,InputL : PEditorInputLine;      procedure Add;      procedure Edit;      procedure Delete;    end;function FPCompleteCodeWord(const WordS: string; var Text: string): boolean;procedure InitCodeComplete;function  LoadCodeComplete(var S: TStream): boolean;procedure AddStandardUnitsToCodeComplete;procedure AddAvailableUnitsToCodeComplete(OnlyStandard : boolean);function  StoreCodeComplete(var S: TStream): boolean;procedure DoneCodeComplete;const CodeCompleteWords : PCodeCompleteWordList = nil;type      TCodeCompleteCase = (ccc_unchanged, ccc_lower, ccc_upper, ccc_mixed);const     CodeCompleteCase : TCodeCompleteCase = ccc_unchanged;     UnitsCodeCompleteWords : PCodeCompleteWordList = nil;procedure RegisterCodeComplete;implementationuses App,Views,MsgBox,Validate,     FVConsts,     systems, BrowCol,     FPSwitch, FPCompil,     FPVars, FPSymbol,     FPConst,FPString,FPViews;{$ifndef NOOBJREG}const  RCodeCompleteWordList: TStreamRec = (     ObjType: 14401;     VmtLink: Ofs(TypeOf(TCodeCompleteWordList)^);     Load:    @TCodeCompleteWordList.Load;     Store:   @TCodeCompleteWordList.Store  );{$endif}function FPCompleteCodeWord(const WordS: string; var Text: string): boolean;var OK: boolean;    CIndex, Index, i : sw_integer;    St, UpWordS : string;begin  if ShowOnlyUnique then    UpWordS:=UpCaseStr(WordS);  OK:=Assigned(CodeCompleteWords);  if OK then  begin    Text:=CodeCompleteWords^.Lookup(WordS,CIndex);    OK:=(CIndex<>-1) and (length(Text)<>length(WordS));    Index:=-1;    if OK and ShowOnlyUnique and (CIndex<CodeCompleteWords^.Count-1) then      begin        St:=PString(CodeCompleteWords^.At(CIndex+1))^;        if (UpCaseStr(Copy(St,1,length(WordS)))=UpWordS) then          begin            {if UpCase(st[Length(UpWordS)+1])<>Upcase(Text[Length(UpWordS)+1]) then}              begin                Text:='';                FPCompleteCodeWord:=false;                exit;            (*  end            else              { only give the common part }              begin                i:=Length(UpWordS)+1;                while (i<=length(st)) and (i<=length(text)) and (UpCase(st[i])=Upcase(Text[i])) do                  inc(i);                SetLength(Text,i-1);    *)              end;          end;      end;  end;  if (ShowOnlyUnique or not OK) and Assigned(UnitsCodeCompleteWords) then  begin    Text:=UnitsCodeCompleteWords^.Lookup(WordS,Index);    OK:=(Index<>-1) and (length(Text)<>length(WordS));    if ShowOnlyUnique and (Index<UnitsCodeCompleteWords^.Count-1) then      begin        St:=PString(UnitsCodeCompleteWords^.At(Index+1))^;        if UpCaseStr(Copy(St,1,length(WordS)))=UpWordS then          begin            {if UpCase(st[Length(UpWordS)+1])<>Upcase(Text[Length(UpWordS)+1]) then}              begin                Text:='';                FPCompleteCodeWord:=false;                exit;            (*  end            else              { only give the common part }              begin                i:=Length(UpWordS)+1;                while (i<=length(st)) and (i<=length(text)) and (UpCase(st[i])=Upcase(Text[i])) do                  inc(i);                SetLength(Text,i-1); *)              end;          end;      end;  end;  if ShowOnlyUnique and (Index<>-1) and (CIndex<>-1) then    begin      {St:=PString(CodeCompleteWords^.At(CIndex+1))^;       Was wrong, CIndex+1 could be above count => collection.error       generated RTE 213      if UpCase(st[Length(UpWordS)+1])<>Upcase(Text[Length(UpWordS)+1]) then}        begin          Text:='';          FPCompleteCodeWord:=false;          exit;      (*  end      else        { only give the common part }        begin          i:=Length(UpWordS)+1;          while (i<=length(st)) and (i<=length(text)) and (UpCase(st[i])=Upcase(Text[i])) do            inc(i);          SetLength(Text,i-1); *)        end;    end;  if OK=false then Text:=''  else case CodeCompleteCase of    ccc_upper : Text:=UpcaseStr(Text);    ccc_lower : Text:=LowcaseStr(Text);    ccc_mixed : Text:=UpCase(Text[1])+LowCaseStr(Copy(Text,2,High(Text)));  end;  FPCompleteCodeWord:=OK;end;procedure InitCodeComplete;var I:integer;    S: string;begin  if Assigned(CodeCompleteWords) then    Dispose(CodeCompleteWords, Done);  New(CodeCompleteWords, Init(10,10));  for I:=0 to GetReservedWordCount-1 do    begin      S:=LowCaseStr(GetReservedWord(I));      if length(S)>=CodeCompleteMinLen then        CodeCompleteWords^.Insert(NewStr(S));    end;  {    there should be also a user front-end for customizing CodeComplete !     any volunteers to implement? ;) - Gabor  }end;procedure AddAvailableUnitsToCodeComplete(OnlyStandard : boolean);var  I : sw_integer;  Overflow: boolean;  Level : longint;  UpStandardUnits : string;  procedure InsertInS(P: PSymbol); {$ifndef FPC}far;{$endif}    procedure InsertItemsInS(P: PSymbolCollection);    var I: Sw_integer;    begin      for I:=0 to P^.Count-1 do        InsertInS(P^.At(I));    end;  Var    st : string;  begin    Inc(level);    if UnitsCodeCompleteWords^.Count=MaxCollectionSize then       begin Overflow:=true; Exit; end;    st:=P^.GetName;    if Length(st)>=CodeCompleteMinLen then      if not ((level=1) and OnlyStandard and (st=UpCaseStr(CodeCompleteUnitName))) then        UnitsCodeCompleteWords^.Insert(NewStr(Lowcasestr(st)));    { this is wrong because it inserted args or locals of proc      in the globals list !! PM}    if (P^.Items<>nil) and (level=1) and        ((not OnlyStandard or (Pos(P^.GetName+',',UpStandardUnits)>0) or        { don't exclude system unit ... }        (Pos('SYS',P^.GetName)>0))) then      InsertItemsInS(P^.Items);    Dec(level);  end;begin  if OnlyStandard then    UpStandardunits:=UpCaseStr(StandardUnits)+',';  if IsSymbolInfoAvailable then    begin      if Assigned(UnitsCodeCompleteWords) then        begin          Dispose(UnitsCodeCompleteWords,done);          UnitsCodeCompleteWords:=nil;        end;      New(UnitsCodeCompleteWords, Init(10,10));      level:=0;      Overflow:=false;      BrowCol.Modules^.ForEach(@InsertInS);      { if Overflow then        WarningBox(msg_toomanysymbolscantdisplayall,nil); }    end;end;procedure AddStandardUnitsToCodeComplete;var  HiddenSource : PSourceWindow;  R : TRect;  StoreBrowserSwitchesConfig : string;begin  Desktop^.GetExtent(R);  New(HiddenSource,init(R,'*'));  HiddenSource^.NoNameCount:=0;  HiddenSource^.UpdateTitle;  HiddenSource^.Hide;  CompilingHiddenFile:=HiddenSource;  { compile a dummy file to get symbol info }  with HiddenSource^.Editor^ do    begin      FileName:=CodeCompleteUnitName+'.pp';      Addline('unit '+CodeCompleteUnitName+';');      Addline('interface');      if StandardUnits<>'' then        begin          AddLine('uses');          Addline(StandardUnits);          Addline('  ;');        end;      Addline('implementation');      Addline('end.');      SetModified(true);      // SaveFile;    end;  StoreBrowserSwitchesConfig:=BrowserSwitches^.GetCurrSelParam;  BrowserSwitches^.ReadItemsCfg('+');  DoCompile(cCompile);  BrowserSwitches^.SetCurrSelParam(StoreBrowserSwitchesConfig);  AddAvailableUnitsToCodeComplete(true);  { Now add the interface declarations to the Code Complete list }  CompilingHiddenFile:=nil;  Dispose(HiddenSource,Done);end;function LoadCodeComplete(var S: TStream): boolean;var C: PCodeCompleteWordList;    OK: boolean;    NewCodeCompleteMinLen : byte;    NewUseStandardUnitsInCodeComplete,    NewUseAllUnitsInCodeComplete,    NewShowOnlyUnique : boolean;    NewCodeCompleteCase : TCodeCompleteCase;    StPtr : PString;begin  New(C, Load(S));  OK:=Assigned(C) and (S.Status=stOk);  if OK then    begin      if Assigned(CodeCompleteWords) then Dispose(CodeCompleteWords, Done);      CodeCompleteWords:=C;      S.Read(NewCodeCompleteCase,Sizeof(TCodeCompleteCase));      OK:=(S.Status=stOk);      if OK then        CodeCompleteCase:=NewCodeCompleteCase;      { Old version of Code complete, also OK PM }      if not OK or (S.getPos=S.getSize) then        begin          LoadCodeComplete:=OK;          exit;        end;      if S.Status=stOK then        S.Read(NewUseStandardUnitsInCodeComplete,Sizeof(UseStandardUnitsInCodeComplete));      if S.Status=stOK then        UseStandardUnitsInCodeComplete:=NewUseStandardUnitsInCodeComplete;      if S.Status=stOK then        S.Read(NewUseAllUnitsInCodeComplete,Sizeof(UseAllUnitsInCodeComplete));      if S.Status=stOK then        UseAllUnitsInCodeComplete:=NewUseAllUnitsInCodeComplete;      if S.Status=stOK then        S.Read(NewShowOnlyUnique,Sizeof(ShowOnlyUnique));      if S.Status=stOK then        ShowOnlyUnique:=NewShowOnlyUnique;      if S.Status=stOK then        S.Read(NewCodeCompleteMinLen,Sizeof(CodeCompleteMinLen));      if S.Status=stOK then        CodeCompleteMinLen:=NewCodeCompleteMinLen;      if S.Status=stOK then        StPtr:=S.ReadStr      else        StPtr:=nil;      if (S.Status=stOK) then        StandardUnits:=GetStr(StPtr);      if assigned(StPtr) then        FreeMem(StPtr,Length(StandardUnits)+1);      OK:=S.Status=stOK;    end  else    if Assigned(C) then      Dispose(C, Done);  LoadCodeComplete:=OK;end;function StoreCodeComplete(var S: TStream): boolean;var OK: boolean;begin  OK:=Assigned(CodeCompleteWords);  if OK then  begin    CodeCompleteWords^.Store(S);    S.Write(CodeCompleteCase,Sizeof(TCodeCompleteCase));    { New fields added }    S.Write(UseStandardUnitsInCodeComplete,Sizeof(UseStandardUnitsInCodeComplete));    S.Write(UseAllUnitsInCodeComplete,Sizeof(UseAllUnitsInCodeComplete));    S.Write(ShowOnlyUnique,Sizeof(ShowOnlyUnique));    S.Write(CodeCompleteMinLen,Sizeof(CodeCompleteMinLen));    S.WriteStr(@StandardUnits);    OK:=OK and (S.Status=stOK);  end;  StoreCodeComplete:=OK;end;procedure DoneCodeComplete;begin  if Assigned(CodeCompleteWords) then    begin      Dispose(CodeCompleteWords, Done);      CodeCompleteWords:=nil;    end;  if Assigned(UnitsCodeCompleteWords) then    begin      Dispose(UnitsCodeCompleteWords,done);      UnitsCodeCompleteWords:=nil;    end;end;constructor TCodeCompleteDialog.Init;var R,R2,R3: TRect;    Items: PSItem;    SB: PScrollBar;begin  R.Assign(0,0,50,22);  inherited Init(R,dialog_codecomplete);  HelpCtx:=hcCodeCompleteOptions;  { name list dialog }  GetExtent(R); R.Grow(-3,-2); Inc(R.A.Y); R3.Copy(R); Dec(R.B.X,12);  Dec(R.B.Y,7);  R2.Copy(R); R2.Move(1,0); R2.A.X:=R2.B.X-1;  New(SB, Init(R2)); Insert(SB);  New(CodeCompleteLB, Init(R,1,SB));  Insert(CodeCompleteLB);  R2.Copy(R); R2.Move(0,-1); R2.B.Y:=R2.A.Y+1; Dec(R2.A.X);  Insert(New(PLabel, Init(R2, label_codecomplete_keywords, CodeCompleteLB)));  { Case choice }  R.Copy(R3); Dec(R.B.Y,2); R.A.Y:=R.B.Y-4; Inc(R.A.X); R.B.X:=R.A.X+15;  Items:=NewSItem('Unc~h~anged',           NewSItem('~L~ower',           NewSItem('~U~pper',           NewSItem('~M~ixed',nil))));  RB:=New(PRadioButtons,Init(R,Items));  RB^.SetData(ord(CodeCompleteCase));  R2.Copy(R); R2.Move(0,-1); R2.B.Y:=R2.A.Y+1; Dec(R2.A.X);  Insert(New(PLabel, Init(R2, 'Case handling', RB)));  Insert(RB);  { Mininum length inputline }  R.Copy(R3); R.A.Y:=R.B.Y-7;R.B.Y:=R.A.Y+1; Dec(R.B.X); R.A.X:=R.B.X -5;  New(MinInputL, Init(R,5));  MinInputL^.SetValidator(New(PRangeValidator, Init(1,255)));  Insert(MinInputL);  R2.Copy(R); R2.A.X:=20;Dec(R2.B.X,5);  Insert(New(PLabel, Init(R2, 'Min. length', MinInputL)));  { Standard/all units booleans }  Items:=nil;  Items:=NewSItem('Add standard units', Items);  Items:=NewSItem('Add all units', Items);  Items:=NewSItem('Show only unique', Items);  R.Copy(R3); R.A.Y:=R.B.Y-5;R.B.Y:=R.A.Y+3; Inc(R.A.X,18); Dec(R.B.X);  New(CB, Init(R, Items));  Insert(CB);  R2.Copy(R); R2.Move(0,-1); R2.B.Y:=R2.A.Y+1; Dec(R2.A.X);  Insert(New(PLabel, Init(R2, 'Unit handling', CB)));  R2.Copy(R); R2.Move(0,-1); R2.B.Y:=R2.A.Y+1;  If ShowOnlyUnique then    CB^.Press(0);  If UseAllUnitsInCodeComplete then    CB^.Press(1);  If UseStandardUnitsInCodeComplete then    CB^.Press(2);  { Standard unit name boolean }  R.Copy(R3); R.A.Y:=R.B.Y-1; Inc(R.A.X); Dec(R.B.X);  New(InputL,Init(R,255));  Insert(InputL);  InputL^.SetValidator(New(PFilterValidator,Init(NumberChars+AlphaChars+[','])));  R2.Copy(R); R2.Move(0,-1); R2.B.Y:=R2.A.Y+1; Dec(R2.A.X);R2.B.X:=R2.A.X+25;  Insert(New(PLabel, Init(R2, '~S~tandard unit list', InputL)));  R.Copy(R3); R.A.X:=R.B.X-10; R.B.Y:=R.A.Y+2;  Insert(New(PButton, Init(R, button_OK, cmOK, bfNormal)));  R.Move(0,2);  Insert(New(PButton, Init(R, button_Edit, cmEditItem, bfDefault)));  R.Move(0,2);  Insert(New(PButton, Init(R, button_New, cmAddItem, bfNormal)));  R.Move(0,2);  Insert(New(PButton, Init(R, button_Delete, cmDeleteItem, bfNormal)));  R.Move(0,2);  Insert(New(PButton, Init(R, button_Cancel, cmCancel, bfNormal)));  SelectNext(false);end;procedure TCodeCompleteDialog.HandleEvent(var Event: TEvent);var DontClear: boolean;begin  case Event.What of    evKeyDown :      begin        DontClear:=false;        case Event.KeyCode of          kbIns  :            Message(@Self,evCommand,cmAddItem,nil);          kbDel  :            Message(@Self,evCommand,cmDeleteItem,nil);        else DontClear:=true;        end;        if DontClear=false then ClearEvent(Event);      end;    evBroadcast :      case Event.Command of        cmListItemSelected :          if Event.InfoPtr=pointer(CodeCompleteLB) then            Message(@Self,evCommand,cmEditItem,nil);      end;    evCommand :      begin        DontClear:=false;        case Event.Command of          cmAddItem    : Add;          cmDeleteItem : Delete;          cmEditItem   : Edit;        else DontClear:=true;        end;        if DontClear=false then ClearEvent(Event);      end;  end;  inherited HandleEvent(Event);end;function TCodeCompleteDialog.Execute: Word;var R: word;    C: PCodeCompleteWordList;    NewVal, I: integer;    NewValStr : string;begin  New(C, Init(10,20));  if Assigned(CodeCompleteWords) then  for I:=0 to CodeCompleteWords^.Count-1 do    C^.Insert(NewStr(GetStr(CodeCompleteWords^.At(I))));  CodeCompleteLB^.NewList(C);  InputL^.SetData(StandardUnits);  NewValStr:=IntToStr(CodeCompleteMinLen);  MinInputL^.SetData(NewValStr);  R:=inherited Execute;  if R=cmOK then    begin      if Assigned(CodeCompleteWords) then Dispose(CodeCompleteWords, Done);      CodeCompleteWords:=C;      CodeCompleteCase:=TCodeCompleteCase(RB^.Value);      MinInputL^.GetData(NewValStr);      NewVal:=StrToInt(NewValStr);      if (NewVal>0) and (NewVal<>CodeCompleteMinLen) then        begin          CodeCompleteMinLen:=NewVal;          InitCodeComplete;        end;      ShowOnlyUnique:=CB^.Mark(0);      UseAllUnitsInCodeComplete:=CB^.Mark(1);      UseStandardUnitsInCodeComplete:=CB^.Mark(2);      if UseStandardUnitsInCodeComplete and (not UseAllUnitsInCodeComplete or not assigned(UnitsCodeCompleteWords)) and         ((StandardUnits<>GetStr(InputL^.Data)) or not assigned(UnitsCodeCompleteWords)) then        begin          InputL^.GetData(StandardUnits);          AddStandardUnitsToCodeComplete;        end      else        InputL^.GetData(StandardUnits);    end  else    Dispose(C, Done);  Execute:=R;end;procedure TCodeCompleteDialog.Add;var IC: boolean;    S: string;    P: PString;    Cmd: word;    CanExit: boolean;    I: sw_integer;begin  IC:=CodeCompleteLB^.Range=0;  if IC=false then    S:=GetStr(CodeCompleteLB^.List^.At(CodeCompleteLB^.Focused))  else    S:='';  repeat    Cmd:=InputBox(dialog_codecomplete_add,label_codecomplete_add_keyword,S,255);    CanExit:=Cmd<>cmOK;    if CanExit=false then      begin        CanExit:=PCodeCompleteWordList(CodeCompleteLB^.List)^.Search(@S,I)=false;        if CanExit=false then        begin          ClearFormatParams; AddFormatParamStr(S);          ErrorBox(msg_codecomplete_alreadyinlist,@FormatParams);        end;      end;  until CanExit;  if Cmd=cmOK then    begin      P:=NewStr(S);      with CodeCompleteLB^ do      begin        List^.Insert(P);        SetRange(List^.Count);        SetFocusedItem(P);      end;      ReDraw;    end;end;procedure TCodeCompleteDialog.Edit;var S: string;    I,T: sw_integer;    Cmd: word;    CanExit: boolean;    P: PString;begin  if CodeCompleteLB^.Range=0 then Exit;  I:=CodeCompleteLB^.Focused;  S:=GetStr(CodeCompleteLB^.List^.At(I));  repeat    Cmd:=InputBox(dialog_codecomplete_edit,label_codecomplete_edit_keyword,S,255);    CanExit:=Cmd<>cmOK;    if CanExit=false then      begin        CanExit:=PCodeCompleteWordList(CodeCompleteLB^.List)^.Search(@S,T)=false;        CanExit:=CanExit or (T=I);        if CanExit=false then        begin          ClearFormatParams; AddFormatParamStr(S);          ErrorBox(msg_codecomplete_alreadyinlist,@FormatParams);        end;      end;  until CanExit;  if Cmd=cmOK then    begin      P:=NewStr(S);      with CodeCompleteLB^ do      begin        List^.AtFree(I);        List^.Insert(P);        SetFocusedItem(P);      end;      ReDraw;    end;end;procedure TCodeCompleteDialog.Delete;begin  if CodeCompleteLB^.Range=0 then Exit;  CodeCompleteLB^.List^.AtFree(CodeCompleteLB^.Focused);  CodeCompleteLB^.SetRange(CodeCompleteLB^.List^.Count);  ReDraw;end;procedure RegisterCodeComplete;begin{$ifndef NOOBJREG}  RegisterType(RCodeCompleteWordList);{$endif}end;END.
 |