Delphi-PRAXiS
Seite 2 von 2     12   

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Programmieren allgemein (https://www.delphipraxis.net/40-programmieren-allgemein/)
-   -   Delphi Externes Programm zum durchsuchen von *.dfm Dateien und Speicherung von Actions (https://www.delphipraxis.net/196729-externes-programm-zum-durchsuchen-von-%2A-dfm-dateien-und-speicherung-von-actions.html)

Dennis07 15. Jun 2018 16:32

AW: Externes Programm zum durchsuchen von *.dfm Dateien und Speicherung von Actions
 
Zitat:

Zitat von Axti (Beitrag 1404825)
Delphi-Quellcode:
    for linePath in TempMenuData do
    begin
      if (MenuLevel = 1)
        and (TempMenuData.Strings[TempMenuData.IndexOf(linePath)].Contains('object')
        and TempMenuData.Strings[TempMenuData.IndexOf(linePath)].Contains('TMenuItem'))
      then
      begin
        tempMenuTop := TempMenuData.Strings[TempMenuData.IndexOf(linePath)];
        tempMenuTop := tempMenuTop.Replace(' ', EmptyStr);
        tempMenuTop := tempMenuTop.Replace('object', EmptyStr);
        tempMenuTop := tempMenuTop.Replace(':TMenuItem', EmptyStr);

        MenuLevel := 2;
      end
      else if (MenuLevel = 2)
        and (TempMenuData.Strings[TempMenuData.IndexOf(linePath)].Contains('object')
        and TempMenuData.Strings[TempMenuData.IndexOf(linePath)].Contains('TMenuItem'))
      then
      begin
        tempPath := tempMenuTop + '->' + TempMenuData.Strings[TempMenuData.IndexOf(linePath)];
        tempPath := tempPath.Replace(' ', EmptyStr);
        tempPath := tempPath.Replace('object', EmptyStr);
        tempPath := tempPath.Replace(':TMenuItem', EmptyStr);

        if TempMenuData.Strings[TempMenuData.IndexOf(linePath)+1].Contains('Action') then
        begin
          tempMenuAction := TempMenuData.Strings[TempMenuData.IndexOf(linePath)+1];
          tempMenuAction := tempMenuAction.Replace('Action = ', '');
          tempMenuAction := tempMenuAction.Replace(' ', '');
        end;

        MenuLevel := 3;
      end
      else if (MenuLevel = 3)
        and (TempMenuData.Strings[TempMenuData.IndexOf(linePath)].Contains('object')

        and TempMenuData.Strings[TempMenuData.IndexOf(linePath)].Contains('TMenuItem'))
      then
      begin
        tempPath := tempPath + '->' + TempMenuData.Strings[TempMenuData.IndexOf(linePath)];
        tempPath := tempPath.Replace(' ', EmptyStr);
        tempPath := tempPath.Replace('object', EmptyStr);
        tempPath := tempPath.Replace(':TMenuItem', EmptyStr);

        if TempMenuData.Strings[TempMenuData.IndexOf(linePath)+1].Contains('Action') then
        begin
          tempMenuAction := TempMenuData.Strings[TempMenuData.IndexOf(linePath)+1];
          tempMenuAction := tempMenuAction.Replace('Action = ', '');
          tempMenuAction := tempMenuAction.Replace(' ', '');
        end;
      end;

      if not linePath.Contains('end') then
      begin
        for lineNewPath in FinishedTable do
        begin
          if FinishedTable[FinishedTable.IndexOf(lineNewPath)].Contains(tempMenuAction)
           and not FinishedTable[FinishedFile.IndexOf(lineNewPath)].Contains('Path')
          then
          begin
            FinishedTable.Strings[FinishedTable.IndexOf(lineNewPath)]
                := FinishedTable.Strings[FinishedTable.IndexOf(lineNewPath)] +
                   ' Path: ' + tempPath;
            break;
          end;
        end;
      end
      else if linePath.Contains('end') then
      begin
        MenuLevel := MenuLevel-1;
      end;

      if MenuLevel = 0 then
      begin
        tempPath := EmptyStr;
      end;

    end;

So solltest du bei Parsern aber nicht arbeiten. StringReplace ist sehr unsicher, da du den Kontext nicht beachtest. Das kann in einigen Fällen zu unbeabsichtigten Ergebnissen oder sogar Fehlern führen. Wieso benutzt du nicht einfach TParser? Da brauchst du dir um all das keine Gedanken zu machen, und außerdem ist ein andauerndes Aufrufen von StringReplace auch nicht gerade performant.

himitsu 15. Jun 2018 21:54

AW: Externes Programm zum durchsuchen von *.dfm Dateien und Speicherung von Actions
 
Zitat:

Wieso benutzt du nicht einfach TParser?
Weil er nichts fertiges nutzen will, denn sonst würde er direkt TReader verwenden, welches das Format sogar direkt versteht.

Dennis07 15. Jun 2018 22:10

AW: Externes Programm zum durchsuchen von *.dfm Dateien und Speicherung von Actions
 
Zitat:

Zitat von himitsu (Beitrag 1404931)
sonst würde er direkt TReader verwenden, welches das Format sogar direkt versteht.

Das tut TParser doch auch??

Und falls er es nicht verwenden will (wieso auch immer), kann er es sich doch innerhalb von ner halben Stunde einfach nachbauen...?

Dennis07 15. Jun 2018 22:32

AW: Externes Programm zum durchsuchen von *.dfm Dateien und Speicherung von Actions
 
Übrigens hat er nirgendwo geschrieben, dass er nichts fertiges benutzen will (korrigiere mich bitte, falls ich falsch liege, habe aber nichts gefunden, das darauf hindeutet. Und selbst wenn, TParser ist nichts "fertiges", es ist nur ein Baustein, genau wie TStringList. Dennoch kann es dir einiges an Arbeit hier abnehmen. Habe das ganze mal (außer die Menüstruktur, da ich noch nicht genau verstanden habe, wie das mit dem "auslesen" gemeint ist) hingeschrieben. So in etwa ließe sich das mit TParser lösen:

Delphi-Quellcode:
program DfmAnalyzer;

{$APPTYPE CONSOLE}
{$R *.res}

uses
  System.SysUtils, System.Classes, System.Generics.Collections, Vcl.Controls;

var
  FileName: String;
  Stream: TFileStream;
  Parser: TParser;
  Current: String;

  FormName: String;
  Actions: TDictionary<TComponentName, TCaption>;

procedure ParseObject(const Depth: Byte = 0; const Item: Boolean = False);

  procedure ParseCollection;
  begin
    Parser.NextToken;
    while Parser.Token <> '>' do
    begin
      if Parser.TokenSymbolIs('item') then
      begin
        ParseObject(Succ(Depth), True);
      end;
      Parser.NextToken;
    end;
    Parser.NextToken;
  end;

var
  ComponentName: String;
  ComponentType: String;
begin
  Parser.NextToken;
  if not Item then
  begin
    ComponentName := Parser.TokenString;
    if Depth = 0 then
    begin
      FormName := ComponentName;
    end;
    Parser.NextToken;
    Parser.NextToken;
    ComponentType := Parser.TokenString;
    Parser.NextToken;
  end;
  while not Parser.TokenSymbolIs('end') do
  begin
    if Parser.TokenSymbolIs('object') then
    begin
      ParseObject(Succ(Depth));
    end else
    begin
      if Parser.Token = '<' then
      begin
        ParseCollection;
      end else
      begin
        if not Item and (ComponentType = 'TAction') and
          (Parser.TokenSymbolIs('Caption')) then
        begin
          Parser.NextToken;
          Parser.NextToken;
          Actions.Add(ComponentName, Parser.TokenWideString);
        end;
      end;
    end;
    Parser.NextToken;
  end;
end;

begin
  try
    ReadLn(FileName);
    Stream := TFileStream.Create(FileName, fmOpenRead);
    Stream.Seek(0, soBeginning);
    Parser := TParser.Create(Stream, TFormatSettings.Invariant);
    Actions := TDictionary<TComponentName, TCaption>.Create;
    try
      ParseObject;
      WriteLn('Formularname: ', FormName);
      for Current in Actions.Keys do
      begin
        WriteLn('Caption von ', Current, ': ', QuotedStr(Actions[Current]));
      end;
    finally
      Actions.Free;
      Parser.Free;
      Stream.Free;
      ReadLn;
    end;
  except
    on E: Exception do
      WriteLn(E.ClassName, ': ', E.Message);
  end;

end.
Das nur mal, um evtl. einen Denkanstoß zu geben.

Axti 19. Jun 2018 08:34

AW: Externes Programm zum durchsuchen von *.dfm Dateien und Speicherung von Actions
 
Hab die Vorschläge jetzt ausprobiert und jetzt funktioniert alles super. Danke :-D


Alle Zeitangaben in WEZ +1. Es ist jetzt 18:40 Uhr.
Seite 2 von 2     12   

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz