Как с помощью Roslyn перечислить элементы (пространства имен, классы и т. д.) В документе Visual Basic?

Используя Roslyn, единственный механизм для определения членов документа Visual Basic выглядит следующим образом:

var members = SyntaxTree.GetRoot().DescendantNodes().Where(node =>
                          node is ClassStatementSyntax ||
                          node is FunctionAggregationSyntax ||
                          node is IncompleteMemberSyntax ||
                          node is MethodBaseSyntax ||
                          node is ModuleStatementSyntax ||
                          node is NamespaceStatementSyntax ||
                          node is PropertyStatementSyntax ||
                          node is SubNewStatementSyntax
                        );

Как получить член название, StarLineNumber и EndLineNumber каждого члена?

Добро пожаловать в SO. Вы можете использовать обратные кавычки ´, чтобы встроенная пометка работала как код

Neuron 04.06.2018 03:25
Стоит ли изучать PHP в 2023-2024 годах?
Стоит ли изучать PHP в 2023-2024 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
0
1
298
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Существует не только один способ его получить:

1) Как вы попробуете: я не буду показывать этот способ для всех добрых членов (они огромны, и логика аналогична), а только для одного из них, например ClassStatementSyntax:

  • чтобы получить его имя, просто возьмите ClassStatementSyntax.Identifier.ValueText
  • чтобы получить стартовую строку, вы можете использовать Location как один из способов:
var location = Location.Create(SyntaxTree, ClassStatementSyntax.Identifier.Span);
var startLine = location.GetLineSpan().StartLinePosition.Line;
  • логика для получения конечной строки выглядит как логика для получения начальной строки, но зависит от соответствующего закрывающего оператора (какой-то конечный оператор или self)

2) Более удобный способ - использовать SemanticModel для получения нужных вам данных: Таким образом, вам нужно будет получать семантическую информацию только для ClassStatementSyntax, ModuleStatementSyntxt и NamespaceStatementSyntax, а все их члены будут получены при вызове GetMembers():

...

  SemanticModel semanticModel = // usually it is received from the corresponding compilation
  var typeSyntax = // ClassStatementSyntax, ModuleStatementSyntxt or NamespaceStatementSyntax
  string name = null;
  int startLine;
  int endLine;

  var info = semanticModel.GetSymbolInfo(typeSyntax);
  if (info.Symbol is INamespaceOrTypeSymbol typeSymbol)
  {
      name = typeSymbol.Name; // retrieve Name
      startLine = semanticModel.SyntaxTree.GetLineSpan(typeSymbol.DeclaringSyntaxReferences[0].Span).StartLinePosition.Line; //retrieve start line
      endLine = semanticModel.SyntaxTree.GetLineSpan(typeSymbol.DeclaringSyntaxReferences[0].Span).EndLinePosition.Line; //retrieve end line
      foreach (var item in typeSymbol.GetMembers())
      {
          // do the same logic for retrieving name and lines for all others members without calling GetMembers()
      }
  }
  else if (semanticModel.GetDeclaredSymbol(typeSyntax) is INamespaceOrTypeSymbol typeSymbol2)
  {
      name = typeSymbol2.Name; // retrieve Name
      startLine = semanticModel.SyntaxTree.GetLineSpan(typeSymbol2.DeclaringSyntaxReferences[0].Span).StartLinePosition.Line; //retrieve start line
      endLine = semanticModel.SyntaxTree.GetLineSpan(typeSymbol2.DeclaringSyntaxReferences[0].Span).EndLinePosition.Line; //retrieve end line

      foreach (var item in typeSymbol2.GetMembers())
      {
          // do the same logic for retrieving name and lines for all others members without calling GetMembers()
      }
  }

Но обратите внимание, когда у вас есть частичное объявление, ваш DeclaringSyntaxReferences будет иметь пару элементов, поэтому вам нужно отфильтровать SyntaxReference по вашему текущему SyntaxTree.

Большое спасибо Джордж. Примерно на следующий день я реализую ваше решение »

Joginder S Nahil 14.06.2018 15:54

Есть идеи, почему startLine и endLine имеют одно и то же значение?

Joginder S Nahil 14.06.2018 17:56

@JoginderSNahil, это означает, что у вас есть какое-то объявление, которое может располагаться полностью в одной строке, например FieldDeclarationSyntax, PropertyDeclarationSyntax (без Get, Set), MethodStatementSyntax (для метода MustOverride) могут быть расположены в одной строке и т. д., поэтому startLine и endLine будут одинаковыми, но startPosition и endPosition, конечно же, будут разными.

George Alexandria 14.06.2018 22:37

Я проанализировал только объявления пространств имен и классов, и они не находятся в одной строке. У вас работает логика?

Joginder S Nahil 17.06.2018 11:07

@JoginderSNahil, приношу свои извинения, я действительно верил, что ISymbol.Locations содержит полный набор элементов, а не только начало объявления. Я обновил ответ изменил использование Locations на DeclaringSyntaxReferences

George Alexandria 17.06.2018 11:49

Ваши изменения работают отлично. Большое спасибо за такой подробный ответ.

Joginder S Nahil 18.06.2018 19:57

@JoginderSNahil, это вы ошибочно проголосовали против ответа или кто-то другой? :)

George Alexandria 18.06.2018 20:03

Я не осознавал, что сделал это! Как я могу это исправить?

Joginder S Nahil 18.06.2018 23:29

@JoginderSNahil, попробуйте проголосовать, если можете (хотите), иначе я не знаю, как отменить его, и мы не беспокоимся об этом :)

George Alexandria 18.06.2018 23:50

Другие вопросы по теме