Skip to content

Commit e86b342

Browse files
committed
Merge branch 'master' into release-2.7
2 parents 4c8b979 + 7a1deae commit e86b342

File tree

113 files changed

+1927
-294
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

113 files changed

+1927
-294
lines changed

src/compiler/binder.ts

+30-24
Original file line numberDiff line numberDiff line change
@@ -2289,30 +2289,13 @@ namespace ts {
22892289
declareSymbol(file.symbol.exports, file.symbol, <PropertyAccessExpression>node.left, SymbolFlags.Property | SymbolFlags.ExportValue, SymbolFlags.None);
22902290
}
22912291

2292-
function isExportsOrModuleExportsOrAlias(node: Node): boolean {
2293-
return isExportsIdentifier(node) ||
2294-
isModuleExportsPropertyAccessExpression(node) ||
2295-
isIdentifier(node) && isNameOfExportsOrModuleExportsAliasDeclaration(node);
2296-
}
2297-
2298-
function isNameOfExportsOrModuleExportsAliasDeclaration(node: Identifier): boolean {
2299-
const symbol = lookupSymbolForName(node.escapedText);
2300-
return symbol && symbol.valueDeclaration && isVariableDeclaration(symbol.valueDeclaration) &&
2301-
symbol.valueDeclaration.initializer && isExportsOrModuleExportsOrAliasOrAssignment(symbol.valueDeclaration.initializer);
2302-
}
2303-
2304-
function isExportsOrModuleExportsOrAliasOrAssignment(node: Node): boolean {
2305-
return isExportsOrModuleExportsOrAlias(node) ||
2306-
(isAssignmentExpression(node, /*excludeCompoundAssignements*/ true) && (isExportsOrModuleExportsOrAliasOrAssignment(node.left) || isExportsOrModuleExportsOrAliasOrAssignment(node.right)));
2307-
}
2308-
23092292
function bindModuleExportsAssignment(node: BinaryExpression) {
23102293
// A common practice in node modules is to set 'export = module.exports = {}', this ensures that 'exports'
23112294
// is still pointing to 'module.exports'.
23122295
// We do not want to consider this as 'export=' since a module can have only one of these.
23132296
// Similarly we do not want to treat 'module.exports = exports' as an 'export='.
23142297
const assignedExpression = getRightMostAssignedExpression(node.right);
2315-
if (isEmptyObjectLiteral(assignedExpression) || isExportsOrModuleExportsOrAlias(assignedExpression)) {
2298+
if (isEmptyObjectLiteral(assignedExpression) || container === file && isExportsOrModuleExportsOrAlias(file, assignedExpression)) {
23162299
// Mark it as a module in case there are no other exports in the file
23172300
setCommonJsModuleIndicator(node);
23182301
return;
@@ -2393,7 +2376,7 @@ namespace ts {
23932376
if (node.kind === SyntaxKind.BinaryExpression) {
23942377
leftSideOfAssignment.parent = node;
23952378
}
2396-
if (isNameOfExportsOrModuleExportsAliasDeclaration(target)) {
2379+
if (container === file && isNameOfExportsOrModuleExportsAliasDeclaration(file, target)) {
23972380
// This can be an alias for the 'exports' or 'module.exports' names, e.g.
23982381
// var util = module.exports;
23992382
// util.property = function ...
@@ -2406,11 +2389,7 @@ namespace ts {
24062389
}
24072390

24082391
function lookupSymbolForName(name: __String) {
2409-
const local = container.locals && container.locals.get(name);
2410-
if (local) {
2411-
return local.exportSymbol || local;
2412-
}
2413-
return container.symbol && container.symbol.exports && container.symbol.exports.get(name);
2392+
return lookupSymbolForNameWorker(container, name);
24142393
}
24152394

24162395
function bindPropertyAssignment(functionName: __String, propertyAccess: PropertyAccessExpression, isPrototypeProperty: boolean) {
@@ -2649,6 +2628,33 @@ namespace ts {
26492628
}
26502629
}
26512630

2631+
/* @internal */
2632+
export function isExportsOrModuleExportsOrAlias(sourceFile: SourceFile, node: Expression): boolean {
2633+
return isExportsIdentifier(node) ||
2634+
isModuleExportsPropertyAccessExpression(node) ||
2635+
isIdentifier(node) && isNameOfExportsOrModuleExportsAliasDeclaration(sourceFile, node);
2636+
}
2637+
2638+
function isNameOfExportsOrModuleExportsAliasDeclaration(sourceFile: SourceFile, node: Identifier): boolean {
2639+
const symbol = lookupSymbolForNameWorker(sourceFile, node.escapedText);
2640+
return symbol && symbol.valueDeclaration && isVariableDeclaration(symbol.valueDeclaration) &&
2641+
symbol.valueDeclaration.initializer && isExportsOrModuleExportsOrAliasOrAssignment(sourceFile, symbol.valueDeclaration.initializer);
2642+
}
2643+
2644+
function isExportsOrModuleExportsOrAliasOrAssignment(sourceFile: SourceFile, node: Expression): boolean {
2645+
return isExportsOrModuleExportsOrAlias(sourceFile, node) ||
2646+
(isAssignmentExpression(node, /*excludeCompoundAssignements*/ true) && (
2647+
isExportsOrModuleExportsOrAliasOrAssignment(sourceFile, node.left) || isExportsOrModuleExportsOrAliasOrAssignment(sourceFile, node.right)));
2648+
}
2649+
2650+
function lookupSymbolForNameWorker(container: Node, name: __String): Symbol | undefined {
2651+
const local = container.locals && container.locals.get(name);
2652+
if (local) {
2653+
return local.exportSymbol || local;
2654+
}
2655+
return container.symbol && container.symbol.exports && container.symbol.exports.get(name);
2656+
}
2657+
26522658
/**
26532659
* Computes the transform flags for a node, given the transform flags of its subtree
26542660
*

src/compiler/checker.ts

+40-20
Original file line numberDiff line numberDiff line change
@@ -268,8 +268,8 @@ namespace ts {
268268
getSuggestionForNonexistentSymbol: (location, name, meaning) => getSuggestionForNonexistentSymbol(location, escapeLeadingUnderscores(name), meaning),
269269
getBaseConstraintOfType,
270270
getDefaultFromTypeParameter: type => type && type.flags & TypeFlags.TypeParameter ? getDefaultFromTypeParameter(type as TypeParameter) : undefined,
271-
resolveName(name, location, meaning) {
272-
return resolveName(location, escapeLeadingUnderscores(name), meaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false);
271+
resolveName(name, location, meaning, excludeGlobals) {
272+
return resolveName(location, escapeLeadingUnderscores(name), meaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false, excludeGlobals);
273273
},
274274
getJsxNamespace: () => unescapeLeadingUnderscores(getJsxNamespace()),
275275
getAccessibleSymbolChain,
@@ -952,8 +952,9 @@ namespace ts {
952952
nameNotFoundMessage: DiagnosticMessage | undefined,
953953
nameArg: __String | Identifier,
954954
isUse: boolean,
955+
excludeGlobals = false,
955956
suggestedNameNotFoundMessage?: DiagnosticMessage): Symbol {
956-
return resolveNameHelper(location, name, meaning, nameNotFoundMessage, nameArg, isUse, getSymbol, suggestedNameNotFoundMessage);
957+
return resolveNameHelper(location, name, meaning, nameNotFoundMessage, nameArg, isUse, excludeGlobals, getSymbol, suggestedNameNotFoundMessage);
957958
}
958959

959960
function resolveNameHelper(
@@ -963,6 +964,7 @@ namespace ts {
963964
nameNotFoundMessage: DiagnosticMessage,
964965
nameArg: __String | Identifier,
965966
isUse: boolean,
967+
excludeGlobals: boolean,
966968
lookup: typeof getSymbol,
967969
suggestedNameNotFoundMessage?: DiagnosticMessage): Symbol {
968970
const originalLocation = location; // needed for did-you-mean error reporting, which gathers candidates starting from the original location
@@ -1209,7 +1211,9 @@ namespace ts {
12091211
}
12101212
}
12111213

1212-
result = lookup(globals, name, meaning);
1214+
if (!excludeGlobals) {
1215+
result = lookup(globals, name, meaning);
1216+
}
12131217
}
12141218

12151219
if (!result) {
@@ -11277,6 +11281,9 @@ namespace ts {
1127711281
}
1127811282
diagnostic = Diagnostics._0_which_lacks_return_type_annotation_implicitly_has_an_1_return_type;
1127911283
break;
11284+
case SyntaxKind.MappedType:
11285+
error(declaration, Diagnostics.Mapped_object_type_implicitly_has_an_any_template_type);
11286+
return;
1128011287
default:
1128111288
diagnostic = Diagnostics.Variable_0_implicitly_has_an_1_type;
1128211289
}
@@ -11889,6 +11896,7 @@ namespace ts {
1188911896
Diagnostics.Cannot_find_name_0,
1189011897
node,
1189111898
!isWriteOnlyAccess(node),
11899+
/*excludeGlobals*/ false,
1189211900
Diagnostics.Cannot_find_name_0_Did_you_mean_1) || unknownSymbol;
1189311901
}
1189411902
return links.resolvedSymbol;
@@ -15154,7 +15162,7 @@ namespace ts {
1515415162
* element is not a class element, or the class element type cannot be determined, returns 'undefined'.
1515515163
* For example, in the element <MyClass>, the element instance type is `MyClass` (not `typeof MyClass`).
1515615164
*/
15157-
function getJsxElementInstanceType(node: JsxOpeningLikeElement, valueType: Type, sourceAttributesType: Type) {
15165+
function getJsxElementInstanceType(node: JsxOpeningLikeElement, valueType: Type, sourceAttributesType: Type | undefined) {
1515815166
Debug.assert(!(valueType.flags & TypeFlags.Union));
1515915167
if (isTypeAny(valueType)) {
1516015168
// Short-circuit if the class tag is using an element type 'any'
@@ -15173,20 +15181,27 @@ namespace ts {
1517315181
}
1517415182
}
1517515183

15176-
const instantiatedSignatures = [];
15177-
for (const signature of signatures) {
15178-
if (signature.typeParameters) {
15179-
const isJavascript = isInJavaScriptFile(node);
15180-
const inferenceContext = createInferenceContext(signature, /*flags*/ isJavascript ? InferenceFlags.AnyDefault : 0);
15181-
const typeArguments = inferJsxTypeArguments(signature, sourceAttributesType, inferenceContext);
15182-
instantiatedSignatures.push(getSignatureInstantiation(signature, typeArguments, isJavascript));
15183-
}
15184-
else {
15185-
instantiatedSignatures.push(signature);
15184+
if (sourceAttributesType) {
15185+
// Instantiate in context of source type
15186+
const instantiatedSignatures = [];
15187+
for (const signature of signatures) {
15188+
if (signature.typeParameters) {
15189+
const isJavascript = isInJavaScriptFile(node);
15190+
const inferenceContext = createInferenceContext(signature, /*flags*/ isJavascript ? InferenceFlags.AnyDefault : 0);
15191+
const typeArguments = inferJsxTypeArguments(signature, sourceAttributesType, inferenceContext);
15192+
instantiatedSignatures.push(getSignatureInstantiation(signature, typeArguments, isJavascript));
15193+
}
15194+
else {
15195+
instantiatedSignatures.push(signature);
15196+
}
1518615197
}
15187-
}
1518815198

15189-
return getUnionType(map(instantiatedSignatures, getReturnTypeOfSignature), UnionReduction.Subtype);
15199+
return getUnionType(map(instantiatedSignatures, getReturnTypeOfSignature), UnionReduction.Subtype);
15200+
}
15201+
else {
15202+
// Do not instantiate if no source type is provided - type parameters and their constraints will be used by contextual typing
15203+
return getUnionType(map(signatures, getReturnTypeOfSignature), UnionReduction.Subtype);
15204+
}
1519015205
}
1519115206

1519215207
/**
@@ -15410,7 +15425,7 @@ namespace ts {
1541015425
}
1541115426

1541215427
// Get the element instance type (the result of newing or invoking this tag)
15413-
const elemInstanceType = getJsxElementInstanceType(openingLikeElement, elementType, sourceAttributesType || emptyObjectType);
15428+
const elemInstanceType = getJsxElementInstanceType(openingLikeElement, elementType, sourceAttributesType);
1541415429

1541515430
// If we should include all stateless attributes type, then get all attributes type from all stateless function signature.
1541615431
// Otherwise get only attributes type from the signature picked by choose-overload logic.
@@ -16068,7 +16083,7 @@ namespace ts {
1606816083

1606916084
function getSuggestionForNonexistentSymbol(location: Node, outerName: __String, meaning: SymbolFlags): string {
1607016085
Debug.assert(outerName !== undefined, "outername should always be defined");
16071-
const result = resolveNameHelper(location, outerName, meaning, /*nameNotFoundMessage*/ undefined, outerName, /*isUse*/ false, (symbols, name, meaning) => {
16086+
const result = resolveNameHelper(location, outerName, meaning, /*nameNotFoundMessage*/ undefined, outerName, /*isUse*/ false, /*excludeGlobals*/ false, (symbols, name, meaning) => {
1607216087
Debug.assertEqual(outerName, name, "name should equal outerName");
1607316088
const symbol = getSymbol(symbols, name, meaning);
1607416089
// Sometimes the symbol is found when location is a return type of a function: `typeof x` and `x` is declared in the body of the function
@@ -20307,6 +20322,11 @@ namespace ts {
2030720322
function checkMappedType(node: MappedTypeNode) {
2030820323
checkSourceElement(node.typeParameter);
2030920324
checkSourceElement(node.type);
20325+
20326+
if (noImplicitAny && !node.type) {
20327+
reportImplicitAnyError(node, anyType);
20328+
}
20329+
2031020330
const type = <MappedType>getTypeFromMappedTypeNode(node);
2031120331
const constraintType = getConstraintTypeFromMappedType(type);
2031220332
checkTypeAssignableTo(constraintType, stringType, node.typeParameter.constraint);
@@ -22985,7 +23005,7 @@ namespace ts {
2298523005
Diagnostics.Property_0_in_type_1_is_not_assignable_to_the_same_property_in_base_type_2,
2298623006
unescapeLeadingUnderscores(declaredProp.escapedName),
2298723007
typeToString(typeWithThis),
22988-
typeToString(getTypeOfSymbol(baseProp))
23008+
typeToString(baseWithThis)
2298923009
);
2299023010
if (!checkTypeAssignableTo(getTypeOfSymbol(prop), getTypeOfSymbol(baseProp), member.name || member, /*message*/ undefined, rootChain)) {
2299123011
issuedMemberError = true;

src/compiler/core.ts

+20-7
Original file line numberDiff line numberDiff line change
@@ -412,12 +412,14 @@ namespace ts {
412412
return result;
413413
}
414414

415+
415416
export function mapIterator<T, U>(iter: Iterator<T>, mapFn: (x: T) => U): Iterator<U> {
416-
return { next };
417-
function next(): { value: U, done: false } | { value: never, done: true } {
418-
const iterRes = iter.next();
419-
return iterRes.done ? iterRes : { value: mapFn(iterRes.value), done: false };
420-
}
417+
return {
418+
next() {
419+
const iterRes = iter.next();
420+
return iterRes.done ? iterRes : { value: mapFn(iterRes.value), done: false };
421+
}
422+
};
421423
}
422424

423425
// Maps from T to T and avoids allocation if all elements map to themselves
@@ -551,12 +553,23 @@ namespace ts {
551553
return result || array;
552554
}
553555

556+
export function mapAllOrFail<T, U>(array: ReadonlyArray<T>, mapFn: (x: T, i: number) => U | undefined): U[] | undefined {
557+
const result: U[] = [];
558+
for (let i = 0; i < array.length; i++) {
559+
const mapped = mapFn(array[i], i);
560+
if (mapped === undefined) {
561+
return undefined;
562+
}
563+
result.push(mapped);
564+
}
565+
return result;
566+
}
567+
554568
export function mapDefined<T, U>(array: ReadonlyArray<T> | undefined, mapFn: (x: T, i: number) => U | undefined): U[] {
555569
const result: U[] = [];
556570
if (array) {
557571
for (let i = 0; i < array.length; i++) {
558-
const item = array[i];
559-
const mapped = mapFn(item, i);
572+
const mapped = mapFn(array[i], i);
560573
if (mapped !== undefined) {
561574
result.push(mapped);
562575
}

src/compiler/diagnosticMessages.json

+12-1
Original file line numberDiff line numberDiff line change
@@ -3571,7 +3571,10 @@
35713571
"category": "Error",
35723572
"code": 7038
35733573
},
3574-
3574+
"Mapped object type implicitly has an 'any' template type.": {
3575+
"category": "Error",
3576+
"code": 7039
3577+
},
35753578
"You cannot rename this element.": {
35763579
"category": "Error",
35773580
"code": 8000
@@ -3874,6 +3877,10 @@
38743877
"category": "Message",
38753878
"code": 90028
38763879
},
3880+
"Add async modifier to containing function": {
3881+
"category": "Message",
3882+
"code": 90029
3883+
},
38773884
"Convert function to an ES2015 class": {
38783885
"category": "Message",
38793886
"code": 95001
@@ -3937,5 +3944,9 @@
39373944
"Use synthetic 'default' member.": {
39383945
"category": "Message",
39393946
"code": 95016
3947+
},
3948+
"Convert to ES6 module": {
3949+
"category": "Message",
3950+
"code": 95017
39403951
}
39413952
}

src/compiler/emitter.ts

+8-20
Original file line numberDiff line numberDiff line change
@@ -1194,27 +1194,15 @@ namespace ts {
11941194
//
11951195

11961196
function emitObjectBindingPattern(node: ObjectBindingPattern) {
1197-
const elements = node.elements;
1198-
if (elements.length === 0) {
1199-
write("{}");
1200-
}
1201-
else {
1202-
write("{");
1203-
emitList(node, elements, ListFormat.ObjectBindingPatternElements);
1204-
write("}");
1205-
}
1197+
write("{");
1198+
emitList(node, node.elements, ListFormat.ObjectBindingPatternElements);
1199+
write("}");
12061200
}
12071201

12081202
function emitArrayBindingPattern(node: ArrayBindingPattern) {
1209-
const elements = node.elements;
1210-
if (elements.length === 0) {
1211-
write("[]");
1212-
}
1213-
else {
1214-
write("[");
1215-
emitList(node, node.elements, ListFormat.ArrayBindingPatternElements);
1216-
write("]");
1217-
}
1203+
write("[");
1204+
emitList(node, node.elements, ListFormat.ArrayBindingPatternElements);
1205+
write("]");
12181206
}
12191207

12201208
function emitBindingElement(node: BindingElement) {
@@ -3167,8 +3155,8 @@ namespace ts {
31673155
TupleTypeElements = CommaDelimited | SpaceBetweenSiblings | SingleLine | Indented,
31683156
UnionTypeConstituents = BarDelimited | SpaceBetweenSiblings | SingleLine,
31693157
IntersectionTypeConstituents = AmpersandDelimited | SpaceBetweenSiblings | SingleLine,
3170-
ObjectBindingPatternElements = SingleLine | AllowTrailingComma | SpaceBetweenBraces | CommaDelimited | SpaceBetweenSiblings,
3171-
ArrayBindingPatternElements = SingleLine | AllowTrailingComma | CommaDelimited | SpaceBetweenSiblings,
3158+
ObjectBindingPatternElements = SingleLine | AllowTrailingComma | SpaceBetweenBraces | CommaDelimited | SpaceBetweenSiblings | NoSpaceIfEmpty,
3159+
ArrayBindingPatternElements = SingleLine | AllowTrailingComma | CommaDelimited | SpaceBetweenSiblings | NoSpaceIfEmpty,
31723160
ObjectLiteralExpressionProperties = PreserveLines | CommaDelimited | SpaceBetweenSiblings | SpaceBetweenBraces | Indented | Braces | NoSpaceIfEmpty,
31733161
ArrayLiteralExpressionElements = PreserveLines | CommaDelimited | SpaceBetweenSiblings | AllowTrailingComma | Indented | SquareBrackets,
31743162
CommaListElements = CommaDelimited | SpaceBetweenSiblings | SingleLine,

0 commit comments

Comments
 (0)