Skip to content

Commit 67bd40e

Browse files
committed
Merge remote-tracking branch 'origin/master' into beta
2 parents d3fc82a + c4b6545 commit 67bd40e

File tree

25 files changed

+430
-197
lines changed

25 files changed

+430
-197
lines changed

CHANGELOG.md

+41
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,47 @@ TODO:
5252

5353
# Unreleased
5454

55+
## v0.26.11 (2024-11-01)
56+
57+
### Features
58+
59+
- If `hostedBaseUrl` is set to the root page on a website, TypeDoc will now include `WebSite` structured data, #2760.
60+
61+
### Bug Fixes
62+
63+
- Fix support for ESM config files with Node 23, #2752.
64+
- Fix type errors when using `"module": "ESNext"` and importing TypeDoc, #2747.
65+
- Inherited comments on overloaded methods now consider the overload position when inheriting a comment, #2755.
66+
67+
## v0.26.10 (2024-10-16)
68+
69+
### Bug Fixes
70+
71+
- Fixed missing space on page headers, #2748.
72+
73+
## v0.26.9 (2024-10-11)
74+
75+
### Features
76+
77+
- Added `headings` option to control optional headings, #2729.
78+
- Updated Chinese translations, #2739.
79+
- Added a folder icon to page navigation elements which are not links, #2741.
80+
81+
### Bug Fixes
82+
83+
- `externalSymbolLinkMappings` now uses the TypeScript reported link target if available, #2725.
84+
- TypeDoc will no longer omit the modules page if a project contains only modules/documents, #2730.
85+
- Fixed missing breadcrumbs on project page, #2728.
86+
- TypeDoc will no longer render an empty readme page if no readme was found.
87+
88+
### Thanks!
89+
90+
- @lriggle-strib
91+
- @mrfigg
92+
- @XeroAlpha
93+
94+
## v0.26.8 (2024-10-04)
95+
5596
### Features
5697

5798
- Updated Chinese translations, #2706.

package-lock.json

+6-6
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
{
22
"name": "typedoc",
33
"description": "Create api documentation for TypeScript projects.",
4-
"version": "0.26.8",
4+
"version": "0.26.11",
55
"homepage": "https://typedoc.org",
66
"type": "module",
77
"exports": {
@@ -50,8 +50,8 @@
5050
"semver": "^7.6.3",
5151
"ts-node": "^10.9.2",
5252
"tsx": "^4.15.7",
53-
"typescript-eslint": "^8.4.0",
54-
"typescript": "5.6.1-rc"
53+
"typescript": "5.6.3",
54+
"typescript-eslint": "^8.4.0"
5555
},
5656
"files": [
5757
"/bin",

scripts/build_themes.js

+41-14
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,48 @@
11
// @ts-check
22
import esbuild from "esbuild";
3+
import fs from "node:fs";
34

4-
const context = await esbuild.context({
5-
entryPoints: ["src/lib/output/themes/default/assets/bootstrap.ts"],
6-
bundle: true,
7-
minify: true,
8-
outfile: "static/main.js",
9-
banner: {
10-
js: '"use strict";',
5+
const watch = process.argv.slice(2).some((t) => t == "--watch" || t == "-w");
6+
7+
// It's convenient to be able to build the themes in watch mode without rebuilding the whole docs
8+
// to test some change to the frontend JS.
9+
/** @type {esbuild.Plugin} */
10+
const copyToDocsPlugin = {
11+
name: "copyToDocs",
12+
setup(build) {
13+
if (watch) {
14+
build.onEnd((result) => {
15+
if (
16+
!result.errors.length &&
17+
fs.existsSync("docs/assets/main.js")
18+
) {
19+
fs.copyFileSync("static/main.js", "docs/assets/main.js");
20+
}
21+
});
22+
}
1123
},
12-
logLevel: "info",
13-
});
24+
};
25+
26+
async function main() {
27+
const context = await esbuild.context({
28+
entryPoints: ["src/lib/output/themes/default/assets/bootstrap.ts"],
29+
bundle: true,
30+
minify: true,
31+
outfile: "static/main.js",
32+
logLevel: "info",
33+
plugins: [copyToDocsPlugin],
34+
});
1435

15-
await context.rebuild();
36+
await context.rebuild();
1637

17-
if (process.argv.slice(2).includes("--watch")) {
18-
await context.watch();
19-
} else {
20-
await context.dispose();
38+
if (watch) {
39+
await context.watch();
40+
} else {
41+
await context.dispose();
42+
}
2143
}
44+
45+
main().catch((err) => {
46+
console.error(err);
47+
process.exitCode = 1;
48+
});

src/lib/converter/comments/discovery.ts

+23-3
Original file line numberDiff line numberDiff line change
@@ -502,13 +502,33 @@ function declarationToCommentNodes(
502502
},
503503
];
504504

505+
let overloadIndex: number | undefined = undefined;
506+
if (ts.isMethodDeclaration(node)) {
507+
const symbol = checker.getSymbolAtLocation(node.name || node);
508+
if (symbol) {
509+
overloadIndex = symbol.declarations
510+
?.filter((d) => d.kind === node.kind)
511+
.indexOf(node);
512+
ok(overloadIndex !== -1, "Should always find declaration");
513+
}
514+
}
515+
505516
const seenSymbols = new Set<ts.Symbol>();
506517
const bases = findBaseOfDeclaration(checker, node, (symbol) => {
507518
if (!seenSymbols.has(symbol)) {
508519
seenSymbols.add(symbol);
509-
return symbol.declarations?.map(
510-
(node) => declarationToCommentNodeIgnoringParents(node) || node,
511-
);
520+
if (overloadIndex === undefined) {
521+
return symbol.declarations?.map(
522+
(node) =>
523+
declarationToCommentNodeIgnoringParents(node) || node,
524+
);
525+
} else if (symbol.declarations?.[overloadIndex]) {
526+
const parentSigNode = symbol.declarations[overloadIndex];
527+
return [
528+
declarationToCommentNodeIgnoringParents(parentSigNode) ||
529+
parentSigNode,
530+
];
531+
}
512532
}
513533
});
514534

src/lib/converter/converter.ts

+17-2
Original file line numberDiff line numberDiff line change
@@ -234,7 +234,21 @@ export class Converter extends AbstractComponent<Application, ConverterEvents> {
234234
constructor(owner: Application) {
235235
super(owner);
236236

237-
this.addUnknownSymbolResolver((ref) => {
237+
const userConfiguredSymbolResolver: ExternalSymbolResolver = (
238+
ref,
239+
refl,
240+
_part,
241+
symbolId,
242+
) => {
243+
if (symbolId) {
244+
return userConfiguredSymbolResolver(
245+
symbolId.toDeclarationReference(),
246+
refl,
247+
undefined,
248+
undefined,
249+
);
250+
}
251+
238252
// Require global links, matching local ones will likely hide mistakes where the
239253
// user meant to link to a local type.
240254
if (ref.resolutionStart !== "global" || !ref.symbolReference) {
@@ -261,7 +275,8 @@ export class Converter extends AbstractComponent<Application, ConverterEvents> {
261275
if (typeof modLinks["*"] === "string") {
262276
return modLinks["*"];
263277
}
264-
});
278+
};
279+
this.addUnknownSymbolResolver(userConfiguredSymbolResolver);
265280

266281
new CategoryPlugin(this);
267282
new CommentPlugin(this);

src/lib/internationalization/locales/en.cts

+1
Original file line numberDiff line numberDiff line change
@@ -291,6 +291,7 @@ export = {
291291
help_sidebarLinks: "Defines links to be included in the sidebar",
292292
help_navigationLeaves:
293293
"Branches of the navigation tree which should not be expanded",
294+
help_headings: "Determines which optional headings are rendered",
294295
help_navigation: "Determines how the navigation sidebar is organized",
295296
help_visibilityFilters:
296297
"Specify the default visibility for builtin filters and additional filters according to modifier tags",

src/lib/internationalization/locales/zh.cts

+1
Original file line numberDiff line numberDiff line change
@@ -287,6 +287,7 @@ export = localeUtils.buildIncompleteTranslation({
287287
help_sidebarLinks: "定义要包含在侧边栏中的链接",
288288
help_navigationLeaves: "导航树中不应扩展的分支",
289289
help_navigation: "确定导航侧边栏的组织方式",
290+
help_headings: "确定标题是否需要被渲染",
290291
help_visibilityFilters:
291292
"根据修饰符标签指定内置过滤器和附加过滤器的默认可见性",
292293
help_searchCategoryBoosts: "配置搜索以提高所选类别的相关性",

src/lib/output/plugins/IconsPlugin.tsx

+6-2
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ const ICONS_JS = `
1212
function addIcons() {
1313
if (document.readyState === "loading") return document.addEventListener("DOMContentLoaded", addIcons);
1414
const svg = document.body.appendChild(document.createElementNS("http://www.w3.org/2000/svg", "svg"));
15-
svg.innerHTML = \`"SVG_HTML"\`;
15+
svg.innerHTML = \`SVG_HTML\`;
1616
svg.style.display = "none";
1717
if (location.protocol === "file:") updateUseElements();
1818
}
@@ -50,7 +50,11 @@ export class IconsPlugin extends RendererComponent {
5050
const icons = (this.owner.theme as DefaultTheme).icons;
5151

5252
for (const [name, icon] of Object.entries(icons)) {
53-
children.push(<g id={`icon-${name}`}>{icon.call(icons).children}</g>);
53+
children.push(
54+
<g id={`icon-${name}`} class="tsd-no-select">
55+
{icon.call(icons).children}
56+
</g>,
57+
);
5458
}
5559

5660
const svg = renderElement(<svg xmlns="http://www.w3.org/2000/svg">{children}</svg>);

src/lib/output/themes/default/DefaultTheme.tsx

+1-10
Original file line numberDiff line numberDiff line change
@@ -169,14 +169,9 @@ export class DefaultTheme extends Theme {
169169
const urls: UrlMapping[] = [];
170170
this.sluggers.set(project, new Slugger());
171171

172-
if (!hasReadme(this.application.options.getValue("readme"))) {
172+
if (!project.readme?.length) {
173173
project.url = "index.html";
174174
urls.push(new UrlMapping<ContainerReflection>("index.html", project, this.reflectionTemplate));
175-
} else if (project.children?.every((child) => child.kindOf(ReflectionKind.Module))) {
176-
// If there are no non-module children, then there's no point in having a modules page since there
177-
// will be nothing on it besides the navigation, so redirect the module page to the readme page
178-
project.url = "index.html";
179-
urls.push(new UrlMapping("index.html", project, this.indexTemplate));
180175
} else {
181176
project.url = "modules.html";
182177
urls.push(new UrlMapping("modules.html", project, this.reflectionTemplate));
@@ -476,10 +471,6 @@ export class DefaultTheme extends Theme {
476471
}
477472
}
478473

479-
function hasReadme(readme: string) {
480-
return !readme.endsWith("none");
481-
}
482-
483474
function getReflectionClasses(
484475
reflection: DeclarationReflection | DocumentReflection,
485476
filters: Record<string, boolean>,

src/lib/output/themes/default/assets/typedoc/Navigation.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -103,7 +103,8 @@ function addNavText(
103103
}
104104
a.appendChild(document.createElement("span")).textContent = el.text;
105105
} else {
106-
parent.appendChild(document.createElement("span")).textContent =
107-
el.text;
106+
const span = parent.appendChild(document.createElement("span"));
107+
span.innerHTML = `<svg width="20" height="20" viewBox="0 0 24 24" fill="none" class="tsd-kind-icon"><use href="#icon-folder"></use></svg>`;
108+
span.appendChild(document.createElement("span")).textContent = el.text;
108109
}
109110
}

src/lib/output/themes/default/layouts/default.tsx

+29
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,34 @@ function favicon(context: DefaultThemeRenderContext) {
2020
}
2121
}
2222

23+
// See #2760
24+
function buildSiteMetadata(context: DefaultThemeRenderContext) {
25+
try {
26+
// We have to know where we are hosted in order to generate this block
27+
const url = new URL(context.options.getValue("hostedBaseUrl"));
28+
29+
// No point in generating this if we aren't the root page on the site
30+
if (url.pathname !== "/") {
31+
return null;
32+
}
33+
34+
return (
35+
<script type="application/ld+json">
36+
<Raw
37+
html={JSON.stringify({
38+
"@context": "https://schema.org",
39+
"@type": "WebSite",
40+
name: context.page.project.name,
41+
url: url.toString(),
42+
})}
43+
/>
44+
</script>
45+
);
46+
} catch {
47+
return null;
48+
}
49+
}
50+
2351
export const defaultLayout = (
2452
context: DefaultThemeRenderContext,
2553
template: RenderTemplate<PageEvent<Reflection>>,
@@ -36,6 +64,7 @@ export const defaultLayout = (
3664
: `${getDisplayName(props.model)} | ${getDisplayName(props.project)}`}
3765
</title>
3866
{favicon(context)}
67+
{props.url === "index.html" && buildSiteMetadata(context)}
3968
<meta name="description" content={"Documentation for " + props.project.name} />
4069
<meta name="viewport" content="width=device-width, initial-scale=1" />
4170

0 commit comments

Comments
 (0)