-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathIDLBuilder.swift
135 lines (126 loc) · 6.12 KB
/
IDLBuilder.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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
import Foundation
import WebIDL
enum IDLBuilder {
static let preamble = """
// This file was auto-generated by WebIDLToSwift. DO NOT EDIT!
import JavaScriptKit
import JavaScriptEventLoop
\n
"""
static let ignoredNames: Set = [
// dictionaries that depend on types not exposed to Window environments
"BreakTokenOptions", "TrustedTypePolicyOptions", "FragmentResultOptions",
"Client_or_MessagePort_or_ServiceWorker", "ExtendableMessageEventInit",
// redundant unions
"CSSColorValue_or_CSSStyleValue",
// need types from specs not yet included
"ShadowAnimation", "MediaProvider", "Blob_or_MediaSource",
"OffscreenRenderingContext", "RenderingContext", "CanvasImageSource",
"HTMLOrSVGImageElement", "HTMLOrSVGScriptElement", "BodyInit",
// implemented manually
// ArrayBufferView
"BigInt64Array_or_BigUint64Array_or_DataView_or_Float32Array_or_Float64Array_or_Int16Array_or_Int32Array_or_Int8Array_or_Uint16Array_or_Uint32Array_or_Uint8Array_or_Uint8ClampedArray",
// RotationMatrixType
"DOMMatrix_or_Float32Array_or_Float64Array",
]
static func writeFile(path: String, content: String) throws {
if FileManager.default.fileExists(atPath: path) {
try FileManager.default.removeItem(atPath: path)
}
try (preamble + content).write(toFile: path, atomically: true, encoding: .utf8)
}
static func generateIDLBindings(idl: [GenericCollection<IDLNode>]) throws -> SwiftSource {
let declarations = idl.flatMap(\.array)
let merged = DeclarationMerger.merge(declarations: declarations)
var contents: [SwiftSource] = []
for node in merged.declarations.sorted(by: { $0.name < $1.name }) {
if ignoredNames.contains(node.name) {
continue
}
let nodeContent = Context.withState(.root(
interfaces: merged.interfaces,
ignored: [
// functions as parameters are unsupported
"AnimationFrameProvider": ["requestAnimationFrame"],
"AnimationWorkletGlobalScope": ["registerAnimator"],
"AudioWorkletGlobalScope": ["registerProcessor"],
"BaseAudioContext": ["decodeAudioData"],
"ComputePressureObserver": ["<constructor>"],
"DataTransferItem": ["getAsString"],
"FileSystemDirectoryEntry": ["getFile", "getDirectory"],
"FileSystemDirectoryReader": ["readEntries"],
"FileSystemEntry": ["getParent"],
"FileSystemFileEntry": ["file"],
"Geolocation": ["getCurrentPosition", "watchPosition"],
"HTMLCanvasElement": ["toBlob"],
"HTMLVideoElement": ["requestVideoFrameCallback"],
"IntersectionObserver": ["<constructor>"],
"LayoutWorkletGlobalScope": ["registerLayout"],
"LockManager": ["request"],
"MediaSession": ["setActionHandler"],
"MutationObserver": ["<constructor>"],
"Navigator": ["getUserMedia"],
"Notification": ["requestPermission"],
"PaintWorkletGlobalScope": ["registerPaint"],
"PerformanceObserver": ["<constructor>"],
"RemotePlayback": ["watchAvailability"],
"ReportingObserver": ["<constructor>"],
"ResizeObserver": ["<constructor>"],
"RTCPeerConnection": ["createOffer", "setLocalDescription", "createAnswer", "setRemoteDescription", "addIceCandidate"],
"Scheduler": ["postTask"],
"Window": ["requestIdleCallback"],
"WindowOrWorkerGlobalScope": ["queueMicrotask"],
"XRSession": ["requestAnimationFrame"],
// variadic callbacks are unsupported
"TrustedTypePolicyFactory": ["createPolicy"],
// NodeFilter
"Document": ["createNodeIterator", "createTreeWalker"],
"NodeIterator": ["filter"],
"TreeWalker": ["filter"],
// EventListener
"EventTarget": ["addEventListener", "removeEventListener"],
"MediaQueryList": ["addListener", "removeListener"],
// invalid override in Swift
"BeforeUnloadEvent": ["returnValue"],
"CSSColor": ["colorSpace"],
"SVGElement": ["className"],
"AudioBufferSourceNode": ["start"],
// XPathNSResolver
"XPathEvaluatorBase": ["createExpression", "createNSResolver", "evaluate"],
// disabled pending addition of more specs
"HTMLMediaElement": ["srcObject"],
"Blob": ["stream"],
"Body": ["body"],
],
types: merged.types
)) {
toSwift(node)
}
contents.append(nodeContent)
}
return "\(lines: contents)"
}
static func generateClosureTypes() throws -> SwiftSource {
return """
/* variadic generics please */
\(lines: Context.closurePatterns.sorted().map(\.swiftRepresentation))
"""
}
static func generateStrings() throws -> SwiftSource {
let strings = Context.strings.sorted()
return """
@usableFromInline enum Strings {
static let _self: JSString = "self"
\(lines: strings.map { "@usableFromInline static let `\(raw: $0)`: JSString = \(quoted: $0)" })
}
"""
}
static func generateUnions() throws -> SwiftSource {
var contents: [SwiftSource] = []
for union in Context.unions.sorted(by: { $0.name < $1.name }) {
guard !ignoredNames.contains(union.name) else { continue }
contents.append(union.swiftRepresentation)
}
return "\(lines: contents)"
}
}