blob: 68efd6cf33df3f98bd1a8d777f39ad3ac49a7337 [file] [log] [blame]
// Copyright (c) 2020, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/dart/element/type.dart';
import 'package:analyzer/src/dart/ast/ast.dart';
import 'package:analyzer/src/dart/constant/utilities.dart';
import 'package:analyzer/src/dart/element/element.dart';
import 'package:analyzer/src/dart/element/type.dart';
import 'package:analyzer/src/error/codes.dart';
import 'package:analyzer/src/generated/resolver.dart';
/// Helper for resolving [VariableDeclaration]s.
class VariableDeclarationResolver {
final ResolverVisitor _resolver;
final bool _strictInference;
VariableDeclarationResolver({
required ResolverVisitor resolver,
required bool strictInference,
}) : _resolver = resolver,
_strictInference = strictInference;
void resolve(VariableDeclarationImpl node) {
var parent = node.parent as VariableDeclarationList;
var initializer = node.initializer;
if (initializer == null) {
if (_strictInference && parent.type == null) {
_resolver.errorReporter.reportErrorForNode(
HintCode.INFERENCE_FAILURE_ON_UNINITIALIZED_VARIABLE,
node,
[node.name.name],
);
}
return;
}
var element = node.declaredElement!;
var isTopLevel =
element is FieldElement || element is TopLevelVariableElement;
InferenceContext.setTypeFromNode(initializer, node);
if (isTopLevel) {
_resolver.flowAnalysis?.topLevelDeclaration_enter(node, null, null);
} else if (element.isLate) {
_resolver.flowAnalysis?.flow?.lateInitializer_begin(node);
}
initializer.accept(_resolver);
initializer = node.initializer;
if (parent.type == null) {
_setInferredType(element, initializer!.staticType!);
}
if (isTopLevel) {
_resolver.flowAnalysis?.topLevelDeclaration_exit();
} else if (element.isLate) {
_resolver.flowAnalysis?.flow?.lateInitializer_end();
}
// Note: in addition to cloning the initializers for const variables, we
// have to clone the initializers for non-static final fields (because if
// they occur in a class with a const constructor, they will be needed to
// evaluate the const constructor).
if (element is ConstVariableElement) {
(element as ConstVariableElement).constantInitializer =
ConstantAstCloner().cloneNullableNode(initializer);
}
}
void _setInferredType(VariableElement element, DartType initializerType) {
if (element is LocalVariableElementImpl) {
if (initializerType.isDartCoreNull) {
initializerType = DynamicTypeImpl.instance;
}
var inferredType = _resolver.typeSystem.demoteType(initializerType);
element.type = inferredType;
}
}
}