// Copyright 2017 Google Inc. Use of this source code is governed by an // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. import 'dart:async'; import 'package:tuple/tuple.dart'; import '../ast/sass.dart'; import '../value.dart'; import 'async.dart'; /// An [AsyncBuiltInCallable]'s callback. typedef FutureOr _Callback(List arguments); /// A callable defined in Dart code. /// /// Unlike user-defined callables, built-in callables support overloads. They /// may declare multiple different callbacks with multiple different sets of /// arguments. When the callable is invoked, the first callback with matching /// arguments is invoked. class AsyncBuiltInCallable implements AsyncCallable { final String name; /// The overloads declared for this callable. final _overloads = >[]; /// Creates a callable with a single [arguments] declaration and a single /// [callback]. /// /// The argument declaration is parsed from [arguments], which should not /// include parentheses. Throws a [SassFormatException] if parsing fails. AsyncBuiltInCallable(this.name, String arguments, FutureOr callback(List arguments)) { _overloads .add(new Tuple2(new ArgumentDeclaration.parse(arguments), callback)); } /// Creates a callable with multiple implementations. /// /// Each key/value pair in [overloads] defines the argument declaration for /// the overload (which should not include parentheses), and the callback to /// execute if that argument declaration matches. Throws a /// [SassFormatException] if parsing fails. AsyncBuiltInCallable.overloaded(this.name, Map overloads) { overloads.forEach((arguments, callback) { _overloads .add(new Tuple2(new ArgumentDeclaration.parse(arguments), callback)); }); } /// Returns the argument declaration and Dart callback for the given /// positional and named arguments. /// /// Note that this doesn't guarantee that [positional] and [names] are valid /// for the returned [ArgumentDeclaration]. Tuple2 callbackFor( int positional, Set names) => _overloads.take(_overloads.length - 1).firstWhere( (overload) => overload.item1.matches(positional, names), orElse: () => _overloads.last); }