import 'package:source_span/source_span.dart';
import 'package:symbol_table/symbol_table.dart';
import 'ast_node.dart';
import 'expression.dart';
import 'identifier.dart';
import 'token.dart';

class Call extends Expression {
  final Expression target;
  final Token lParen, rParen;
  final List<Expression> arguments;
  final List<NamedArgument> namedArguments;

  Call(this.target, this.lParen, this.rParen, this.arguments,
      this.namedArguments);

  @override
  FileSpan get span {
    return arguments
        .fold<FileSpan>(lParen.span, (out, a) => out.expand(a.span))
        .expand(namedArguments.fold<FileSpan>(
            lParen.span, (out, a) => out.expand(a.span)))
        .expand(rParen.span);
  }

  List computePositional(SymbolTable scope) => arguments.map((e) => e.compute(scope)).toList();

  Map<Symbol, dynamic> computeNamed(SymbolTable scope) {
    return namedArguments.fold<Map<Symbol, dynamic>>({}, (out, a) {
      return out..[new Symbol(a.name.name)] = a.value.compute(scope);
    });
  }

  @override
  compute(scope) {
    var callee = target.compute(scope);
    var args = computePositional(scope);
    var named = computeNamed(scope);

    return Function.apply(callee, args, named);
  }
}

class NamedArgument extends AstNode {
  final Identifier name;
  final Token colon;
  final Expression value;

  NamedArgument(this.name, this.colon, this.value);

  @override
  FileSpan get span {
    return name.span.expand(colon.span).expand(value.span);
  }
}