Skip to content

Commit a9ecebc

Browse files
authored
Merge pull request #790 from davedawkins/issue-789
Documentation and tests for <seealso> when cref omits arguments
2 parents bd0ba65 + 8706dc8 commit a9ecebc

File tree

6 files changed

+107
-11
lines changed

6 files changed

+107
-11
lines changed

RELEASE_NOTES.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
## 17.2.2
2+
3+
* Improvement for `<seealso/>` [#789](https://github.com/fsprojects/FSharp.Formatting/issues/789)
4+
15
## 17.2.1
26

37
* Fix support for `<exclude/>` [#786](https://github.com/fsprojects/FSharp.Formatting/issues/786)

docs/apidocs.fsx

+38
Original file line numberDiff line numberDiff line change
@@ -169,6 +169,44 @@ type GenericClass2<'T>() =
169169
/// and <see cref="M:TheNamespace.GenericClass2`1.GenericMethod``1(`0,``0)" />
170170
let referringFunction2 () = "result"
171171

172+
(**
173+
### Cross-referencing with &lt;seealso&gt;
174+
175+
Use `<seealso cref="..."/>` within `<summary>` to create cross-references.
176+
177+
For example:
178+
*)
179+
180+
module Forest =
181+
182+
/// <summary>
183+
/// Find at most <c>limit</c> foxes in current forest
184+
///
185+
/// See also: <seealso cref="M:App.Forest.findSquirrels(System.Int32)"/>
186+
/// </summary>
187+
let findFoxes (limit : int) = []
188+
189+
/// <summary>
190+
/// Find at most <c>limit</c> squirrels in current forest
191+
///
192+
/// See also: <seealso cref="M:App.Forest.findFoxes(System.Int32)"/>
193+
/// </summary>
194+
let findSquirrels (limit : int) = []
195+
196+
197+
(** You can find the correct value for `cref` in the generated `.xml` documentation file (this will be generated alongside the assembly's `.dll``).
198+
199+
You can also omit the `cref`'s arguments, and `fsdocs` will make an attempt to find the first member that matches.
200+
201+
For example:
202+
```
203+
/// See also: <seealso cref="M:App.Forest.findSquirrels"/>
204+
```
205+
206+
If the member cannot be found, a link to the containing module/type will be used instead.
207+
*)
208+
209+
172210
(**
173211
### Classic XMl Doc Comments: Excluding APIs from the docs
174212

src/FSharp.Formatting.ApiDocs/GenerateModel.fs

+30-11
Original file line numberDiff line numberDiff line change
@@ -858,6 +858,15 @@ type internal CrossReferenceResolver(root, collectionName, qualify, extensions)
858858
else
859859
None
860860

861+
let tryGetShortMemberNameFromMemberName (memberName: string) =
862+
let sub = removeParen memberName
863+
let lastPeriod = sub.LastIndexOf(".")
864+
865+
if lastPeriod > 0 then
866+
Some(memberName.Substring(lastPeriod + 1))
867+
else
868+
None
869+
861870
let getMemberName keepParts hasModuleSuffix (memberNameNoParen: string) =
862871
let splits = memberNameNoParen.Split('.') |> Array.toList
863872

@@ -992,6 +1001,13 @@ type internal CrossReferenceResolver(root, collectionName, qualify, extensions)
9921001
let simple = getMemberName 1 false typeName
9931002
externalDocsLink false simple typeName typeName
9941003

1004+
let mfvToCref (mfv: FSharpMemberOrFunctionOrValue) =
1005+
let entityUrlBaseName = getUrlBaseNameForRegisteredEntity mfv.DeclaringEntity.Value
1006+
1007+
{ IsInternal = true
1008+
ReferenceLink = internalCrossReferenceForMember entityUrlBaseName mfv
1009+
NiceName = mfv.DeclaringEntity.Value.DisplayName + "." + mfv.DisplayName }
1010+
9951011
let tryResolveCrossReferenceForMemberByXmlSig (memberXmlSig: string) =
9961012
assert
9971013
(memberXmlSig.StartsWith("M:")
@@ -1000,13 +1016,7 @@ type internal CrossReferenceResolver(root, collectionName, qualify, extensions)
10001016
|| memberXmlSig.StartsWith("E:"))
10011017

10021018
match xmlDocNameToSymbol.TryGetValue(memberXmlSig) with
1003-
| true, (:? FSharpMemberOrFunctionOrValue as memb) when memb.DeclaringEntity.IsSome ->
1004-
let entityUrlBaseName = getUrlBaseNameForRegisteredEntity memb.DeclaringEntity.Value
1005-
1006-
{ IsInternal = true
1007-
ReferenceLink = internalCrossReferenceForMember entityUrlBaseName memb
1008-
NiceName = memb.DeclaringEntity.Value.DisplayName + "." + memb.DisplayName }
1009-
|> Some
1019+
| true, (:? FSharpMemberOrFunctionOrValue as memb) when memb.DeclaringEntity.IsSome -> memb |> mfvToCref |> Some
10101020
| _ ->
10111021
// If we can't find the exact symbol for the member, don't despair, look for the type
10121022
let memberName = memberXmlSig.Substring(2) |> removeParen
@@ -1019,10 +1029,19 @@ type internal CrossReferenceResolver(root, collectionName, qualify, extensions)
10191029
| true, (:? FSharpEntity as entity) ->
10201030
let urlBaseName = getUrlBaseNameForRegisteredEntity entity
10211031

1022-
Some
1023-
{ IsInternal = true
1024-
ReferenceLink = internalCrossReference urlBaseName
1025-
NiceName = getMemberName 2 entity.HasFSharpModuleSuffix memberName }
1032+
// See if we find the member that was intended, otherwise default to containing entity
1033+
tryGetShortMemberNameFromMemberName memberName
1034+
|> Option.bind (fun shortName ->
1035+
entity.MembersFunctionsAndValues
1036+
|> Seq.tryFind (fun mfv -> mfv.DisplayName = shortName))
1037+
|> function
1038+
| Some mb -> Some(mfvToCref mb)
1039+
| None ->
1040+
Some
1041+
{ IsInternal = true
1042+
ReferenceLink = internalCrossReference urlBaseName
1043+
NiceName = getMemberName 2 entity.HasFSharpModuleSuffix memberName }
1044+
10261045
| _ ->
10271046
// A reference to something external, currently assumed to be in .NET
10281047
let simple = getMemberName 2 false memberName

tests/FSharp.ApiDocs.Tests/ApiDocsTests.fs

+16
Original file line numberDiff line numberDiff line change
@@ -127,6 +127,22 @@ let generateApiDocs (libraries: string list) (format: OutputFormat) useMdComment
127127

128128
do FSharp.Formatting.TestHelpers.enableLogging ()
129129

130+
[<Test>]
131+
[<TestCaseSource("formats")>]
132+
let ``ApiDocs seealso can find members`` (format: OutputFormat) =
133+
let library = testBin </> "TestLib3.dll" |> fullpath
134+
135+
let files = generateApiDocs [ library ] format false "TestLib3"
136+
137+
let (textA, textB) =
138+
if format = OutputFormat.Html then
139+
"seealso.html#disposeOnUnmount", "seealso.html#unsubscribeOnUnmount"
140+
else
141+
"seealso#disposeOnUnmount", "seealso#unsubscribeOnUnmount"
142+
143+
files.[(sprintf "test-seealso.%s" format.Extension)] |> shouldContainText textA
144+
145+
files.[(sprintf "test-seealso.%s" format.Extension)] |> shouldContainText textB
130146

131147
[<Test>]
132148
[<TestCaseSource("formats")>]
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
/// <summary>
2+
/// DOM low-level helper functions
3+
/// </summary>
4+
module Test.SeeAlso
5+
6+
/// <summary>
7+
/// Dispose all given items when the parent <c>SutilElement</c> is unmounted. Each item should implement <c>System.IDisposable</c>.
8+
///
9+
/// See also: <seealso cref="M:Test.SeeAlso.unsubscribeOnUnmount"/>
10+
/// </summary>
11+
let disposeOnUnmount (ds: System.IDisposable list) = ignore ds
12+
13+
/// <summary>
14+
/// Call each function of type `(unit -> unit)` when the element is unmounted
15+
///
16+
/// See also: <seealso cref="M:Test.SeeAlso.disposeOnUnmount"/>
17+
/// </summary>
18+
let unsubscribeOnUnmount (ds: (unit -> unit) list) = ignore ds

tests/FSharp.ApiDocs.Tests/files/TestLib3/TestLib3.fsproj

+1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
<OutputPath>..\bin\$(Configuration)</OutputPath>
66
</PropertyGroup>
77
<ItemGroup>
8+
<Compile Include="SeeAlso.fs" />
89
<Compile Include="DOM.fs" />
910
<Compile Include="Library3.fs" />
1011
<Compile Include="UndocumentedModule.fs" />

0 commit comments

Comments
 (0)