-
Notifications
You must be signed in to change notification settings - Fork 137
/
Copy pathSnippet.swift
79 lines (65 loc) · 3.53 KB
/
Snippet.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/*
This source file is part of the Swift.org open source project
Copyright (c) 2021-2023 Apple Inc. and the Swift project authors
Licensed under Apache License v2.0 with Runtime Library Exception
See https://swift.org/LICENSE.txt for license information
See https://swift.org/CONTRIBUTORS.txt for Swift project authors
*/
import Foundation
import Markdown
import SymbolKit
public final class Snippet: Semantic, AutomaticDirectiveConvertible {
public static let introducedVersion = "5.6"
public let originalMarkup: BlockDirective
/// The path components of a symbol link that would be used to resolve a reference to a snippet,
/// only occurring as a block directive argument.
@DirectiveArgumentWrapped
public var path: String
/// An optional named range to limit the lines shown.
@DirectiveArgumentWrapped
public var slice: String? = nil
static var keyPaths: [String : AnyKeyPath] = [
"path" : \Snippet._path,
"slice" : \Snippet._slice,
]
static var hiddenFromDocumentation = true
@available(*, deprecated, message: "Do not call directly. Required for 'AutomaticDirectiveConvertible'.")
init(originalMarkup: BlockDirective) {
self.originalMarkup = originalMarkup
super.init()
}
func validate(problems: inout [Problem], source: URL?) -> Bool {
if path.isEmpty {
problems.append(Problem(diagnostic: Diagnostic(source: source, severity: .warning, range: originalMarkup.range, identifier: "org.swift.docc.EmptySnippetLink", summary: "No path provided to snippet; use a symbol link path to a known snippet"), possibleSolutions: []))
return false
}
return true
}
}
extension Snippet: RenderableDirectiveConvertible {
func render(with contentCompiler: inout RenderContentCompiler) -> [RenderContent] {
guard let snippet = Snippet(from: originalMarkup, for: contentCompiler.bundle, in: contentCompiler.context) else {
return []
}
guard let snippetReference = contentCompiler.resolveSymbolReference(destination: snippet.path),
let snippetEntity = try? contentCompiler.context.entity(with: snippetReference),
let snippetSymbol = snippetEntity.symbol,
let snippetMixin = snippetSymbol.mixins[SymbolGraph.Symbol.Snippet.mixinKey] as? SymbolGraph.Symbol.Snippet else {
return []
}
if let requestedSlice = snippet.slice,
let requestedLineRange = snippetMixin.slices[requestedSlice] {
// Render only the slice.
let lineRange = requestedLineRange.lowerBound..<min(requestedLineRange.upperBound, snippetMixin.lines.count)
let lines = snippetMixin.lines[lineRange]
let minimumIndentation = lines.map { $0.prefix { $0.isWhitespace }.count }.min() ?? 0
let trimmedLines = lines.map { String($0.dropFirst(minimumIndentation)) }
return [RenderBlockContent.codeListing(.init(syntax: snippetMixin.language, code: trimmedLines, metadata: nil))]
} else {
// Render the whole snippet with its explanation content.
let docCommentContent = snippetEntity.markup.children.flatMap { contentCompiler.visit($0) }
let code = RenderBlockContent.codeListing(.init(syntax: snippetMixin.language, code: snippetMixin.lines, metadata: nil))
return docCommentContent + [code]
}
}
}