@Override
 public boolean visit(Expression node) throws Exception {
   for (final PHPSourceElementRequestorExtension visitor : extensions) {
     visitor.visit(node);
   }
   if (node instanceof Assignment) {
     return visit((Assignment) node);
   }
   if (node instanceof ListVariable) {
     return visit((ListVariable) node);
   }
   if (node instanceof TypeReference) {
     return visit((TypeReference) node);
   }
   if (node instanceof Include) {
     return visit((Include) node);
   }
   if (node instanceof PHPCallExpression) {
     return visit((PHPCallExpression) node);
   }
   if (node instanceof LambdaFunctionDeclaration) {
     return visit((LambdaFunctionDeclaration) node);
   }
   return true;
 }
 @Override
 public boolean visit(Statement node) throws Exception {
   for (final PHPSourceElementRequestorExtension visitor : extensions) {
     visitor.visit(node);
   }
   if (node instanceof PHPFieldDeclaration) {
     return visit((PHPFieldDeclaration) node);
   }
   if (node instanceof FieldDeclaration) {
     return visit((FieldDeclaration) node);
   }
   if (node instanceof ConstantDeclaration) {
     return visit((ConstantDeclaration) node);
   }
   if (node instanceof CatchClause) {
     return visit((CatchClause) node);
   }
   if (node instanceof ForEachStatement) {
     return visit((ForEachStatement) node);
   }
   if (node instanceof GlobalStatement) {
     return visit((GlobalStatement) node);
   }
   if (node instanceof UseStatement) {
     return visit((UseStatement) node);
   }
   return true;
 }
  @Override
  public boolean visit(MethodDeclaration method) throws Exception {

    methodGlobalVars.add(new HashSet<String>());

    Declaration parentDeclaration = null;
    if (!declarations.empty()) {
      parentDeclaration = declarations.peek();
    }

    // In case we are entering a nested element - just add to the deferred
    // list
    // and get out of the nested element visiting process
    if (parentDeclaration instanceof MethodDeclaration) {
      deferredDeclarations.add(method);
      return false;
    }

    if (parentDeclaration instanceof InterfaceDeclaration) {
      method.setModifier(Modifiers.AccAbstract);
    }

    declarations.push(method);

    for (final PHPSourceElementRequestorExtension visitor : extensions) {
      visitor.visit(method);
    }

    final boolean visit = super.visit(method);

    if (visit) {
      // Process method argument (local variable) declarations:
      final List<Argument> arguments = method.getArguments();
      for (final Argument arg : arguments) {
        final ISourceElementRequestor.FieldInfo info = new ISourceElementRequestor.FieldInfo();
        info.name = arg.getName();
        info.modifiers = Modifiers.AccPublic;
        info.nameSourceStart = arg.getNameStart();
        info.nameSourceEnd = arg.getNameEnd() - 1;
        info.declarationStart = arg.sourceStart();
        fRequestor.enterField(info);
        fRequestor.exitField(arg.sourceEnd() - 1);
      }
    }
    return visit;
  }
  @Override
  public boolean visit(TypeDeclaration type) throws Exception {
    if (type instanceof NamespaceDeclaration) {
      final NamespaceDeclaration namespaceDecl = (NamespaceDeclaration) type;
      fLastNamespace = namespaceDecl;
      fLastUseParts.clear();
      if (namespaceDecl.isGlobal()) {
        return true;
      }
    }

    // In case we are entering a nested element
    if (!declarations.empty() && declarations.peek() instanceof MethodDeclaration) {
      deferredDeclarations.add(type);
      return false;
    }

    declarations.push(type);

    for (final PHPSourceElementRequestorExtension visitor : extensions) {
      visitor.visit(type);
    }
    return super.visit(type);
  }
  public boolean visit(LambdaFunctionDeclaration lambdaMethod) throws Exception {

    fNodes.push(lambdaMethod);
    methodGlobalVars.add(new HashSet<String>());

    // Declaration parentDeclaration = null;
    // if (!declarations.empty()
    // && declarations.peek() instanceof MethodDeclaration) {
    // parentDeclaration = declarations.peek();
    // // In case we are entering a nested element - just add to the
    // // deferred list and get out of the nested element visiting process
    // deferredDeclarations.add(lambdaMethod);
    // return visitGeneral(lambdaMethod);
    // }

    final Collection<FormalParameter> arguments = lambdaMethod.getArguments();
    final StringBuilder metadata = new StringBuilder();
    final String[] parameters = new String[arguments.size()];
    // if (arguments != null) {
    final Iterator<FormalParameter> i = arguments.iterator();
    int indx = 0;
    while (i.hasNext()) {
      final Argument arg = i.next();
      metadata.append(arg.getName());
      parameters[indx] = arg.getName();
      indx++;
      if (i.hasNext()) {
        metadata.append(",");
      }
    }
    // }

    // Add method declaration:
    for (final PHPSourceElementRequestorExtension visitor : extensions) {
      visitor.visit(lambdaMethod);
    }

    final ISourceElementRequestor.MethodInfo mi = new ISourceElementRequestor.MethodInfo();
    mi.parameterNames = parameters;
    mi.name = PHPCoreConstants.ANONYMOUS;
    mi.modifiers = Modifiers.AccPublic;
    mi.nameSourceStart = lambdaMethod.sourceStart();
    mi.nameSourceEnd = lambdaMethod.sourceEnd();
    mi.declarationStart = mi.nameSourceStart;
    mi.isConstructor = false;

    this.fRequestor.enterMethod(mi);
    this.fInMethod = true;

    for (final Argument arg : arguments) {
      final ISourceElementRequestor.FieldInfo info = new ISourceElementRequestor.FieldInfo();
      info.name = arg.getName();
      info.modifiers = Modifiers.AccPublic;
      info.nameSourceStart = arg.getNameStart();
      info.nameSourceEnd = arg.getNameEnd() - 1;
      info.declarationStart = arg.sourceStart();
      fRequestor.enterField(info);
      fRequestor.exitField(arg.sourceEnd() - 1);
    }

    return true;
  }