@@ -15,13 +15,16 @@ import { unified } from "unified";
15
15
import rehypeParse from "rehype-parse" ;
16
16
import rehypeRemark from "rehype-remark" ;
17
17
import remarkStringify from "remark-stringify" ;
18
+ import { visit } from "unist-util-visit" ;
19
+ import { toText } from "hast-util-to-text" ;
18
20
19
21
import { ApiType } from "./Metadata.js" ;
20
22
import {
21
23
getLastPartFromFullIdentifier ,
22
24
removeSuffix ,
23
25
APOSTROPHE_HEX_CODE ,
24
26
} from "../stringUtils.js" ;
27
+ import { Root } from "mdast" ;
25
28
26
29
export type ComponentProps = {
27
30
id ?: string ;
@@ -52,6 +55,7 @@ export async function processMdxComponent(
52
55
priorApiType : ApiType | undefined ,
53
56
apiType : Exclude < ApiType , "module" > ,
54
57
id : string ,
58
+ determineSignatureUrl : ( rawLink : string ) => string ,
55
59
) : Promise < [ string , string ] > {
56
60
const tagName = APITYPE_TO_TAG [ apiType ] ;
57
61
@@ -71,7 +75,10 @@ export async function processMdxComponent(
71
75
) ;
72
76
addExtraSignatures ( componentProps , extraProps ) ;
73
77
74
- return [ await createOpeningTag ( tagName , componentProps ) , `</${ tagName } >` ] ;
78
+ return [
79
+ await createOpeningTag ( tagName , componentProps , determineSignatureUrl ) ,
80
+ `</${ tagName } >` ,
81
+ ] ;
75
82
}
76
83
77
84
// ------------------------------------------------------------------
@@ -310,6 +317,7 @@ function prepareFunctionProps(
310
317
export async function createOpeningTag (
311
318
tagName : string ,
312
319
props : ComponentProps ,
320
+ determineSignatureUrl : ( rawLink : string ) => string ,
313
321
) : Promise < string > {
314
322
const attributeTypeHint = props . attributeTypeHint ?. replaceAll (
315
323
"'" ,
@@ -319,10 +327,15 @@ export async function createOpeningTag(
319
327
"'" ,
320
328
APOSTROPHE_HEX_CODE ,
321
329
) ;
322
- const signature = await htmlSignatureToMd ( props . rawSignature ! ) ;
330
+ const signature = await htmlSignatureToMd (
331
+ props . rawSignature ! ,
332
+ determineSignatureUrl ,
333
+ ) ;
323
334
const extraSignatures : string [ ] = [ ] ;
324
335
for ( const sig of props . extraRawSignatures ?? [ ] ) {
325
- extraSignatures . push ( `"${ await htmlSignatureToMd ( sig ! ) } "` ) ;
336
+ extraSignatures . push (
337
+ `"${ await htmlSignatureToMd ( sig ! , determineSignatureUrl ) } "` ,
338
+ ) ;
326
339
}
327
340
328
341
return `<${ tagName }
@@ -398,14 +411,43 @@ export function addExtraSignatures(
398
411
*/
399
412
export async function htmlSignatureToMd (
400
413
signatureHtml : string ,
414
+ determineSignatureUrl : ( rawLink : string ) => string ,
401
415
) : Promise < string > {
402
416
if ( ! signatureHtml ) {
403
417
return "" ;
404
418
}
405
419
420
+ // The `code` tag helps us remove some undesired elements like asterisks surrounding
421
+ // the parameters.
406
422
const html = `<code>${ signatureHtml } </code>` ;
407
423
const file = await unified ( )
408
424
. use ( rehypeParse )
425
+ . use ( ( ) => ( tree : Root ) => {
426
+ visit ( tree , { tagName : "a" } , ( node : any ) => {
427
+ // We transform the links into markdown as `text` nodes to avoid losing them once
428
+ // transforming the signature from HTML to markdown. This could happen because we
429
+ // have the signatures inside a `code` element. Moreover, we have more freedom by
430
+ // manually creating the links.
431
+ if ( ! node . properties ?. href || ! node . children ?. length ) {
432
+ return ;
433
+ }
434
+
435
+ // We encode some conflicting characters that could make the markdown link break
436
+ // by using URL-encoding form.
437
+ const href = determineSignatureUrl ( node . properties . href )
438
+ . replaceAll ( '"' , "%22" )
439
+ . replaceAll ( "(" , "%28" )
440
+ . replaceAll ( ")" , "%29" ) ;
441
+ const isExternal = href . startsWith ( "http" ) ;
442
+ const title = node . properties . title ;
443
+
444
+ // We only show the title if it's from an external link
445
+ const link = title && isExternal ? `${ href } "${ title } "` : `${ href } ` ;
446
+
447
+ node . type = "text" ;
448
+ node . value = `[${ toText ( node . children [ 0 ] ) } ](${ link } )` ;
449
+ } ) ;
450
+ } )
409
451
. use ( rehypeRemark )
410
452
. use ( remarkStringify )
411
453
. process ( html ) ;
0 commit comments