mirror of
https://github.com/discordjs/discord.js.git
synced 2026-03-12 01:23:31 +01:00
698 lines
24 KiB
TypeScript
698 lines
24 KiB
TypeScript
/* eslint-disable @typescript-eslint/no-loop-func */
|
|
// Copyright (c) Microsoft Corporation. All rights reserved. Licensed under the MIT license.
|
|
// See LICENSE in the project root for license information.
|
|
|
|
import {
|
|
type DocNode,
|
|
type DocPlainText,
|
|
DocDeclarationReference,
|
|
DocNodeKind,
|
|
TSDocConfiguration,
|
|
DocMemberReference,
|
|
DocMemberIdentifier,
|
|
} from '@microsoft/tsdoc';
|
|
import type { DeclarationReference } from '@microsoft/tsdoc/lib-commonjs/beta/DeclarationReference.js';
|
|
import { InternalError } from '@rushstack/node-core-library';
|
|
import type { IExcerptToken, IExcerptTokenRange } from '../index.js';
|
|
import { ApiDeclaredItem } from '../index.js';
|
|
import type { IApiDeclaredItemJson } from '../items/ApiDeclaredItem.js';
|
|
import {
|
|
ApiItem,
|
|
apiItem_onParentChanged,
|
|
type IApiItemJson,
|
|
type IApiItemOptions,
|
|
type IApiItemConstructor,
|
|
ApiItemKind,
|
|
} from '../items/ApiItem.js';
|
|
import type { ApiClass } from '../model/ApiClass.js';
|
|
import type { ApiInterface } from '../model/ApiInterface.js';
|
|
import type { ApiModel } from '../model/ApiModel.js';
|
|
import { ApiJsonSchemaVersion, type DeserializerContext } from '../model/DeserializerContext.js';
|
|
import type { HeritageType } from '../model/HeritageType.js';
|
|
import type { IResolveDeclarationReferenceResult } from '../model/ModelReferenceResolver.js';
|
|
import { ApiNameMixin } from './ApiNameMixin.js';
|
|
import type { ExcerptToken } from './Excerpt.js';
|
|
import { ExcerptTokenKind } from './Excerpt.js';
|
|
import { type IFindApiItemsResult, type IFindApiItemsMessage, FindApiItemsMessageId } from './IFindApiItemsResult.js';
|
|
|
|
/**
|
|
* Constructor options for {@link (ApiItemContainerMixin:interface)}.
|
|
*
|
|
* @public
|
|
*/
|
|
export interface IApiItemContainerMixinOptions extends IApiItemOptions {
|
|
members?: ApiItem[] | undefined;
|
|
preserveMemberOrder?: boolean | undefined;
|
|
}
|
|
|
|
export interface IApiItemContainerJson extends IApiItemJson {
|
|
members: IApiItemJson[];
|
|
preserveMemberOrder?: boolean;
|
|
}
|
|
|
|
interface Mixin {
|
|
declarationReference: DocDeclarationReference;
|
|
typeParameters: IExcerptTokenRange[];
|
|
}
|
|
|
|
interface ExcerptTokenRangeInDeclaredItem {
|
|
item: ApiDeclaredItem;
|
|
range: IExcerptTokenRange;
|
|
}
|
|
interface IMappedTypeParameters {
|
|
item: ApiItem;
|
|
mappedTypeParameters: Map<string, ExcerptTokenRangeInDeclaredItem>;
|
|
}
|
|
|
|
const _members: unique symbol = Symbol('ApiItemContainerMixin._members');
|
|
const _membersSorted: unique symbol = Symbol('ApiItemContainerMixin._membersSorted');
|
|
const _membersByContainerKey: unique symbol = Symbol('ApiItemContainerMixin._membersByContainerKey');
|
|
const _membersByName: unique symbol = Symbol('ApiItemContainerMixin._membersByName');
|
|
const _membersByKind: unique symbol = Symbol('ApiItemContainerMixin._membersByKind');
|
|
const _preserveMemberOrder: unique symbol = Symbol('ApiItemContainerMixin._preserveMemberOrder');
|
|
|
|
/**
|
|
* The mixin base class for API items that act as containers for other child items.
|
|
*
|
|
* @remarks
|
|
*
|
|
* This is part of the {@link ApiModel} hierarchy of classes, which are serializable representations of
|
|
* API declarations. The non-abstract classes (e.g. `ApiClass`, `ApiEnum`, `ApiInterface`, etc.) use
|
|
* TypeScript "mixin" functions (e.g. `ApiDeclaredItem`, `ApiItemContainerMixin`, etc.) to add various
|
|
* features that cannot be represented as a normal inheritance chain (since TypeScript does not allow a child class
|
|
* to extend more than one base class). The "mixin" is a TypeScript merged declaration with three components:
|
|
* the function that generates a subclass, an interface that describes the members of the subclass, and
|
|
* a namespace containing static members of the class.
|
|
*
|
|
* Examples of `ApiItemContainerMixin` child classes include `ApiModel`, `ApiPackage`, `ApiEntryPoint`,
|
|
* and `ApiEnum`. But note that `Parameter` is not considered a "member" of an `ApiMethod`; this relationship
|
|
* is modeled using {@link (ApiParameterListMixin:interface).parameters} instead
|
|
* of {@link ApiItem.members}.
|
|
* @public
|
|
*/
|
|
|
|
export interface ApiItemContainerMixin extends ApiItem {
|
|
/**
|
|
* For a given member of this container, return its `ApiItem.getMergedSiblings()` list.
|
|
*
|
|
* @internal
|
|
*/
|
|
_getMergedSiblingsForMember(memberApiItem: ApiItem): readonly ApiItem[];
|
|
|
|
/**
|
|
* Adds a new member to the container.
|
|
*
|
|
* @remarks
|
|
* An ApiItem cannot be added to more than one container.
|
|
*/
|
|
addMember(member: ApiItem): void;
|
|
|
|
/**
|
|
* Returns a list of members with the specified name.
|
|
*/
|
|
findMembersByName(name: string): readonly ApiItem[];
|
|
|
|
/**
|
|
* Finds all of the ApiItem's immediate and inherited members by walking up the inheritance tree.
|
|
*
|
|
* @remarks
|
|
*
|
|
* Given the following class heritage:
|
|
*
|
|
* ```
|
|
* export class A {
|
|
* public a: number|boolean;
|
|
* }
|
|
*
|
|
* export class B extends A {
|
|
* public a: number;
|
|
* public b: string;
|
|
* }
|
|
*
|
|
* export class C extends B {
|
|
* public c: boolean;
|
|
* }
|
|
* ```
|
|
*
|
|
* Calling `findMembersWithInheritance` on `C` will return `B.a`, `B.b`, and `C.c`. Calling the
|
|
* method on `B` will return `B.a` and `B.b`. And calling the method on `A` will return just
|
|
* `A.a`.
|
|
*
|
|
* The inherited members returned by this method may be incomplete. If so, there will be a flag
|
|
* on the result object indicating this as well as messages explaining the errors in more detail.
|
|
* Some scenarios include:
|
|
*
|
|
* - Interface extending from a type alias.
|
|
*
|
|
* - Class extending from a variable.
|
|
*
|
|
* - Extending from a declaration not present in the model (e.g. external package).
|
|
*
|
|
* - Extending from an unexported declaration (e.g. ae-forgotten-export). Common in mixin
|
|
* patterns.
|
|
*
|
|
* - Unexpected runtime errors...
|
|
*
|
|
* Lastly, be aware that the types of inherited members are returned with respect to their
|
|
* defining class as opposed to with respect to the inheriting class. For example, consider
|
|
* the following:
|
|
*
|
|
* ```
|
|
* export class A<T> {
|
|
* public a: T;
|
|
* }
|
|
*
|
|
* export class B extends A<number> {}
|
|
* ```
|
|
*
|
|
* When called on `B`, this method will return `B.a` with type `T` as opposed to type
|
|
* `number`, although the latter is more accurate.
|
|
*/
|
|
findMembersWithInheritance(): IFindApiItemsResult;
|
|
|
|
/**
|
|
* Disables automatic sorting of {@link ApiItem.members}.
|
|
*
|
|
* @remarks
|
|
* By default `ApiItemContainerMixin` will automatically sort its members according to their
|
|
* {@link ApiItem.getSortKey} string, which provides a standardized mostly alphabetical ordering
|
|
* that is appropriate for most API items. When loading older .api.json files the automatic sorting
|
|
* is reapplied and may update the ordering.
|
|
*
|
|
* Set `preserveMemberOrder` to true to disable automatic sorting for this container; instead, the
|
|
* members will retain whatever ordering appeared in the {@link IApiItemContainerMixinOptions.members} array.
|
|
* The `preserveMemberOrder` option is saved in the .api.json file.
|
|
*/
|
|
readonly preserveMemberOrder: boolean;
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
serializeInto(jsonObject: Partial<IApiItemJson>): void;
|
|
|
|
/**
|
|
* Attempts to retrieve a member using its containerKey, or returns `undefined` if no matching member was found.
|
|
*
|
|
* @remarks
|
|
* Use the `getContainerKey()` static member to construct the key. Each subclass has a different implementation
|
|
* of this function, according to the aspects that are important for identifying it.
|
|
*
|
|
* See {@link ApiItem.containerKey} for more information.
|
|
*/
|
|
tryGetMemberByKey(containerKey: string): ApiItem | undefined;
|
|
}
|
|
|
|
/**
|
|
* Mixin function for {@link ApiDeclaredItem}.
|
|
*
|
|
* @param baseClass - The base class to be extended
|
|
* @returns A child class that extends baseClass, adding the {@link (ApiItemContainerMixin:interface)} functionality.
|
|
* @public
|
|
*/
|
|
export function ApiItemContainerMixin<TBaseClass extends IApiItemConstructor>(
|
|
baseClass: TBaseClass,
|
|
): TBaseClass & (new (...args: any[]) => ApiItemContainerMixin) {
|
|
class MixedClass extends baseClass implements ApiItemContainerMixin {
|
|
public readonly [_members]: ApiItem[];
|
|
|
|
public [_membersSorted]: boolean;
|
|
|
|
public [_membersByContainerKey]: Map<string, ApiItem>;
|
|
|
|
public [_preserveMemberOrder]: boolean;
|
|
|
|
// For members of this container that extend ApiNameMixin, this stores the list of members with a given name.
|
|
// Examples include merged declarations, overloaded functions, etc.
|
|
public [_membersByName]: Map<string, ApiItem[]> | undefined;
|
|
|
|
// For members of this container that do NOT extend ApiNameMixin, this stores the list of members
|
|
// that share a common ApiItemKind. Examples include overloaded constructors or index signatures.
|
|
public [_membersByKind]: Map<string, ApiItem[]> | undefined; // key is ApiItemKind
|
|
|
|
public constructor(...args: any[]) {
|
|
super(...args);
|
|
const options: IApiItemContainerMixinOptions = args[0] as IApiItemContainerMixinOptions;
|
|
|
|
this[_members] = [];
|
|
this[_membersSorted] = false;
|
|
this[_membersByContainerKey] = new Map<string, ApiItem>();
|
|
this[_preserveMemberOrder] = options.preserveMemberOrder ?? false;
|
|
|
|
if (options.members) {
|
|
for (const member of options.members) {
|
|
this.addMember(member);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
public static override onDeserializeInto(
|
|
options: Partial<IApiItemContainerMixinOptions>,
|
|
context: DeserializerContext,
|
|
jsonObject: IApiItemContainerJson,
|
|
): void {
|
|
baseClass.onDeserializeInto(options, context, jsonObject);
|
|
options.preserveMemberOrder = jsonObject.preserveMemberOrder;
|
|
options.members = [];
|
|
for (const memberObject of jsonObject.members) {
|
|
options.members.push(ApiItem.deserialize(memberObject, context));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
public override get members(): readonly ApiItem[] {
|
|
if (!this[_membersSorted] && !this[_preserveMemberOrder]) {
|
|
this[_members].sort((x, y) => x.getSortKey().localeCompare(y.getSortKey()));
|
|
this[_membersSorted] = true;
|
|
}
|
|
|
|
return this[_members];
|
|
}
|
|
|
|
public get preserveMemberOrder(): boolean {
|
|
return this[_preserveMemberOrder];
|
|
}
|
|
|
|
public addMember(member: ApiItem): void {
|
|
if (this[_membersByContainerKey].has(member.containerKey)) {
|
|
throw new Error(
|
|
`Another member has already been added with the same name (${member.displayName})` +
|
|
` and containerKey (${member.containerKey})`,
|
|
);
|
|
}
|
|
|
|
const existingParent: ApiItem | undefined = member.parent;
|
|
if (existingParent !== undefined) {
|
|
throw new Error(`This item has already been added to another container: "${existingParent.displayName}"`);
|
|
}
|
|
|
|
this[_members].push(member);
|
|
this[_membersByName] = undefined; // invalidate the lookup
|
|
this[_membersByKind] = undefined; // invalidate the lookup
|
|
this[_membersSorted] = false;
|
|
this[_membersByContainerKey].set(member.containerKey, member);
|
|
|
|
member[apiItem_onParentChanged](this);
|
|
}
|
|
|
|
public tryGetMemberByKey(containerKey: string): ApiItem | undefined {
|
|
return this[_membersByContainerKey].get(containerKey);
|
|
}
|
|
|
|
public findMembersByName(name: string): readonly ApiItem[] {
|
|
this._ensureMemberMaps();
|
|
return this[_membersByName]!.get(name) ?? [];
|
|
}
|
|
|
|
public findMembersWithInheritance(): IFindApiItemsResult {
|
|
const messages: IFindApiItemsMessage[] = [];
|
|
let maybeIncompleteResult = false;
|
|
|
|
// For API items that don't support inheritance, this method just returns the item's
|
|
// immediate members.
|
|
switch (this.kind) {
|
|
case ApiItemKind.Class:
|
|
case ApiItemKind.Interface:
|
|
break;
|
|
default: {
|
|
return {
|
|
items: this.members.concat(),
|
|
messages,
|
|
maybeIncompleteResult,
|
|
};
|
|
}
|
|
}
|
|
|
|
// The Deserializer class is coupled with a ton of other classes, so we delay loading it
|
|
// to avoid ES5 circular imports.
|
|
// *eslint-disable-next-line @typescript-eslint/consistent-type-imports, @typescript-eslint/no-require-imports, @typescript-eslint/no-var-requires
|
|
// const deserializerModule: typeof import('../model/Deserializer') = require('../model/Deserializer');
|
|
|
|
const membersByName: Map<string, ApiItem[]> = new Map();
|
|
const membersByKind: Map<ApiItemKind, ApiItem[]> = new Map();
|
|
|
|
const toVisit: IMappedTypeParameters[] = [];
|
|
let next: IMappedTypeParameters | undefined = { item: this, mappedTypeParameters: new Map() };
|
|
|
|
while (next?.item) {
|
|
const membersToAdd: ApiItem[] = []; //*
|
|
const typeParams = next.mappedTypeParameters;
|
|
const context: DeserializerContext = {
|
|
apiJsonFilename: '',
|
|
toolPackage: '',
|
|
toolVersion: '',
|
|
versionToDeserialize: ApiJsonSchemaVersion.LATEST,
|
|
tsdocConfiguration: new TSDocConfiguration(),
|
|
}; // */
|
|
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-imports, @typescript-eslint/no-require-imports, @typescript-eslint/no-var-requires
|
|
const deserializerModule: typeof import('../model/Deserializer') = require('../model/Deserializer');
|
|
|
|
// For each member, check to see if we've already seen a member with the same name
|
|
// previously in the inheritance tree. If so, we know we won't inherit it, and thus
|
|
// do not add it to our `membersToAdd` array.
|
|
for (let member of next.item.members) {
|
|
// We add the to-be-added members to an intermediate array instead of immediately
|
|
// to the maps themselves to support method overloads with the same name.
|
|
|
|
// This was supposed to replace type parameters with their assigned values in inheritance, but doesn't work yet
|
|
//*
|
|
if (member instanceof ApiDeclaredItem && member.excerptTokens.some((token) => typeParams.has(token.text))) {
|
|
const jsonObject: Partial<IApiItemJson> = {};
|
|
member.serializeInto(jsonObject);
|
|
const excerptTokens = (jsonObject as IApiDeclaredItemJson).excerptTokens.map((token) => {
|
|
let x: ExcerptToken | undefined;
|
|
if (typeParams.has(token.text) && next?.item instanceof ApiDeclaredItem) {
|
|
const originalValue = typeParams.get(token.text)!;
|
|
x = originalValue.item.excerptTokens[originalValue.range.startIndex];
|
|
}
|
|
|
|
const excerptToken: IExcerptToken = x ? { kind: x.kind, text: x.text } : token;
|
|
if (x?.canonicalReference !== undefined) {
|
|
excerptToken.canonicalReference = x.canonicalReference.toString();
|
|
}
|
|
|
|
return excerptToken;
|
|
});
|
|
member = deserializerModule.Deserializer.deserialize(context, {
|
|
...jsonObject,
|
|
excerptTokens,
|
|
} as IApiDeclaredItemJson);
|
|
member[apiItem_onParentChanged](next.item);
|
|
}
|
|
|
|
if (ApiNameMixin.isBaseClassOf(member)) {
|
|
if (!membersByName.has(member.name)) {
|
|
membersToAdd.push(member);
|
|
}
|
|
} else if (!membersByKind.has(member.kind)) {
|
|
membersToAdd.push(member);
|
|
}
|
|
}
|
|
|
|
for (const member of membersToAdd) {
|
|
if (ApiNameMixin.isBaseClassOf(member)) {
|
|
const members: ApiItem[] = membersByName.get(member.name) ?? [];
|
|
members.push(member);
|
|
membersByName.set(member.name, members);
|
|
} else {
|
|
const members: ApiItem[] = membersByKind.get(member.kind) ?? [];
|
|
members.push(member);
|
|
membersByKind.set(member.kind, members);
|
|
}
|
|
}
|
|
|
|
const findPlainTextNode = (node: DocNode): DocPlainText | undefined => {
|
|
switch (node.kind) {
|
|
case DocNodeKind.PlainText:
|
|
return node as DocPlainText;
|
|
default:
|
|
for (const child of node.getChildNodes()) {
|
|
const result = findPlainTextNode(child);
|
|
if (result) return result;
|
|
}
|
|
}
|
|
|
|
return undefined;
|
|
};
|
|
|
|
// Interfaces can extend multiple interfaces, so iterate through all of them.
|
|
// Also Classes can have multiple mixins
|
|
const extendedItems: IMappedTypeParameters[] = [];
|
|
let extendsTypes: readonly (HeritageType | Mixin)[] | undefined;
|
|
|
|
switch (next.item.kind) {
|
|
case ApiItemKind.Class: {
|
|
const apiClass: ApiClass = next.item as ApiClass;
|
|
const configuration = apiClass.tsdocComment?.configuration ?? new TSDocConfiguration();
|
|
const mixins =
|
|
apiClass.tsdocComment?.customBlocks
|
|
.filter(
|
|
(block) => block.blockTag.tagName === '@mixes', // &&
|
|
// block.getChildNodes().some((node) => node.kind === DocNodeKind.PlainText),
|
|
)
|
|
.map(findPlainTextNode)
|
|
.filter((block) => block !== undefined)
|
|
.map((block) => ({
|
|
declarationReference: new DocDeclarationReference({
|
|
configuration,
|
|
memberReferences: block.text.split('.').map(
|
|
(part, index) =>
|
|
new DocMemberReference({
|
|
configuration,
|
|
hasDot: index > 0,
|
|
memberIdentifier: new DocMemberIdentifier({ configuration, identifier: part }),
|
|
}),
|
|
),
|
|
}),
|
|
typeParameters: [] as IExcerptTokenRange[],
|
|
})) ?? [];
|
|
extendsTypes = apiClass.extendsType ? [apiClass.extendsType, ...mixins] : [...mixins];
|
|
break;
|
|
}
|
|
|
|
case ApiItemKind.Interface: {
|
|
const apiInterface: ApiInterface = next.item as ApiInterface;
|
|
extendsTypes = apiInterface.extendsTypes;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (extendsTypes === undefined) {
|
|
messages.push({
|
|
messageId: FindApiItemsMessageId.UnsupportedKind,
|
|
text: `Unable to analyze references of API item ${next.item.displayName} because it is of unsupported kind ${next.item.kind}`,
|
|
});
|
|
maybeIncompleteResult = true;
|
|
next = toVisit.shift();
|
|
continue;
|
|
}
|
|
|
|
for (const extendsType of extendsTypes) {
|
|
let canonicalReference: DeclarationReference | DocDeclarationReference;
|
|
if ('excerpt' in extendsType) {
|
|
// We want to find the reference token associated with the actual inherited declaration.
|
|
// In every case we support, this is the first reference token. For example:
|
|
//
|
|
// ```
|
|
// export class A extends B {}
|
|
// ^
|
|
// export class A extends B<C> {}
|
|
// ^
|
|
// export class A extends B.C {}
|
|
// ^^^
|
|
// ```
|
|
const firstReferenceToken: ExcerptToken | undefined = extendsType.excerpt.spannedTokens.find(
|
|
(token: ExcerptToken) => {
|
|
return token.kind === ExcerptTokenKind.Reference && token.canonicalReference;
|
|
},
|
|
);
|
|
|
|
if (!firstReferenceToken) {
|
|
messages.push({
|
|
messageId: FindApiItemsMessageId.ExtendsClauseMissingReference,
|
|
text: `Unable to analyze extends clause ${extendsType.excerpt.text} of API item ${next.item.displayName} because no canonical reference was found`,
|
|
});
|
|
maybeIncompleteResult = true;
|
|
continue;
|
|
}
|
|
|
|
canonicalReference = firstReferenceToken.canonicalReference!;
|
|
} else {
|
|
// extendsType is a Mixin
|
|
canonicalReference = extendsType.declarationReference;
|
|
}
|
|
|
|
const apiModel: ApiModel | undefined = this.getAssociatedModel();
|
|
if (!apiModel) {
|
|
messages.push({
|
|
messageId: FindApiItemsMessageId.NoAssociatedApiModel,
|
|
text: `Unable to analyze references of API item ${next.item.displayName} because it is not associated with an ApiModel`,
|
|
});
|
|
maybeIncompleteResult = true;
|
|
continue;
|
|
}
|
|
|
|
const apiItemResult: IResolveDeclarationReferenceResult = apiModel.resolveDeclarationReference(
|
|
canonicalReference,
|
|
this,
|
|
);
|
|
|
|
const apiItem: ApiItem | undefined = apiItemResult.resolvedApiItem;
|
|
if (!apiItem) {
|
|
messages.push({
|
|
messageId: FindApiItemsMessageId.DeclarationResolutionFailed,
|
|
text: `Unable to resolve declaration reference within API item ${next.item.displayName}: ${apiItemResult.errorMessage}`,
|
|
});
|
|
maybeIncompleteResult = true;
|
|
continue;
|
|
}
|
|
|
|
const mappedTypeParameters: Map<string, ExcerptTokenRangeInDeclaredItem> = new Map();
|
|
if (
|
|
(apiItem.kind === ApiItemKind.Class || apiItem.kind === ApiItemKind.Interface) &&
|
|
next.item.kind === ApiItemKind.Class
|
|
) {
|
|
for (const [index, key] of (apiItem as ApiClass | ApiInterface).typeParameters.entries() ?? []) {
|
|
const typeParameter = extendsType.typeParameters?.[index];
|
|
if (typeParameter)
|
|
mappedTypeParameters.set(key.name, { item: next.item as ApiDeclaredItem, range: typeParameter });
|
|
else if (key.defaultTypeExcerpt)
|
|
mappedTypeParameters.set(key.name, {
|
|
item: apiItem as ApiDeclaredItem,
|
|
range: key.defaultTypeExcerpt.tokenRange,
|
|
});
|
|
}
|
|
}
|
|
|
|
extendedItems.push({ item: apiItem, mappedTypeParameters });
|
|
}
|
|
|
|
// For classes, this array will only have one item. For interfaces, there may be multiple items. Sort the array
|
|
// into alphabetical order before adding to our list of API items to visit. This ensures that in the case
|
|
// of multiple interface inheritance, a member inherited from multiple interfaces is attributed to the interface
|
|
// earlier in alphabetical order (as opposed to source order).
|
|
//
|
|
// For example, in the code block below, `Bar.x` is reported as the inherited item, not `Foo.x`.
|
|
//
|
|
// ```
|
|
// interface Foo {
|
|
// public x: string;
|
|
// }
|
|
//
|
|
// interface Bar {
|
|
// public x: string;
|
|
// }
|
|
//
|
|
// interface FooBar extends Foo, Bar {}
|
|
// ```
|
|
extendedItems.sort((x: IMappedTypeParameters, y: IMappedTypeParameters) =>
|
|
x.item.getSortKey().localeCompare(y.item.getSortKey()),
|
|
);
|
|
|
|
toVisit.push(...extendedItems);
|
|
next = toVisit.shift();
|
|
}
|
|
|
|
const items: ApiItem[] = [];
|
|
for (const members of membersByName.values()) {
|
|
items.push(...members);
|
|
}
|
|
|
|
for (const members of membersByKind.values()) {
|
|
items.push(...members);
|
|
}
|
|
|
|
items.sort((x: ApiItem, y: ApiItem) => x.getSortKey().localeCompare(y.getSortKey()));
|
|
|
|
return {
|
|
items,
|
|
messages,
|
|
maybeIncompleteResult,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
public _getMergedSiblingsForMember(memberApiItem: ApiItem): readonly ApiItem[] {
|
|
this._ensureMemberMaps();
|
|
let result: ApiItem[] | undefined;
|
|
if (ApiNameMixin.isBaseClassOf(memberApiItem)) {
|
|
result = this[_membersByName]!.get(memberApiItem.name);
|
|
} else {
|
|
result = this[_membersByKind]!.get(memberApiItem.kind);
|
|
}
|
|
|
|
if (!result) {
|
|
throw new InternalError('Item was not found in the _membersByName/_membersByKind lookup');
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
public _ensureMemberMaps(): void {
|
|
// Build the _membersByName and _membersByKind tables if they don't already exist
|
|
if (this[_membersByName] === undefined) {
|
|
const membersByName: Map<string, ApiItem[]> = new Map<string, ApiItem[]>();
|
|
const membersByKind: Map<string, ApiItem[]> = new Map<string, ApiItem[]>();
|
|
|
|
for (const member of this[_members]) {
|
|
let map: Map<ApiItemKind, ApiItem[]> | Map<string, ApiItem[]>;
|
|
let key: ApiItemKind | string;
|
|
|
|
if (ApiNameMixin.isBaseClassOf(member)) {
|
|
map = membersByName;
|
|
key = member.name;
|
|
} else {
|
|
map = membersByKind;
|
|
key = member.kind;
|
|
}
|
|
|
|
let list: ApiItem[] | undefined = map.get(key);
|
|
if (list === undefined) {
|
|
list = [];
|
|
map.set(key, list);
|
|
}
|
|
|
|
list.push(member);
|
|
}
|
|
|
|
this[_membersByName] = membersByName;
|
|
this[_membersByKind] = membersByKind;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
public override serializeInto(jsonObject: Partial<IApiItemContainerJson>): void {
|
|
super.serializeInto(jsonObject);
|
|
|
|
const memberObjects: IApiItemJson[] = [];
|
|
|
|
for (const member of this.members) {
|
|
const memberJsonObject: Partial<IApiItemJson> = {};
|
|
member.serializeInto(memberJsonObject);
|
|
memberObjects.push(memberJsonObject as IApiItemJson);
|
|
}
|
|
|
|
jsonObject.preserveMemberOrder = this.preserveMemberOrder;
|
|
jsonObject.members = memberObjects;
|
|
}
|
|
}
|
|
|
|
return MixedClass;
|
|
}
|
|
|
|
/**
|
|
* Static members for {@link (ApiItemContainerMixin:interface)}.
|
|
*
|
|
* @public
|
|
*/
|
|
|
|
export namespace ApiItemContainerMixin {
|
|
/**
|
|
* A type guard that tests whether the specified `ApiItem` subclass extends the `ApiItemContainerMixin` mixin.
|
|
*
|
|
* @remarks
|
|
*
|
|
* The JavaScript `instanceof` operator cannot be used to test for mixin inheritance, because each invocation of
|
|
* the mixin function produces a different subclass. (This could be mitigated by `Symbol.hasInstance`, however
|
|
* the TypeScript type system cannot invoke a runtime test.)
|
|
*/
|
|
export function isBaseClassOf(apiItem: ApiItem): apiItem is ApiItemContainerMixin {
|
|
return apiItem.hasOwnProperty(_members);
|
|
}
|
|
}
|