Skip to content

Commit e772e9a

Browse files
BridgeJS: Enhance documentation
1 parent 7573d71 commit e772e9a

File tree

17 files changed

+1061
-43
lines changed

17 files changed

+1061
-43
lines changed

Examples/ExportSwift/Sources/main.swift

+7
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
import JavaScriptKit
22

3+
// Mark functions you want to export to JavaScript with the @JS attribute
4+
// This function will be available as `renderCircleSVG(size)` in JavaScript
35
@JS public func renderCircleSVG(size: Int) -> String {
46
let strokeWidth = 3
57
let strokeColor = "black"
@@ -14,13 +16,18 @@ import JavaScriptKit
1416
return svg
1517
}
1618

19+
// Classes can also be exported using the @JS attribute
20+
// This class will be available as a constructor in JavaScript: new Greeter("name")
1721
@JS class Greeter {
1822
var name: String
1923

24+
// Use @JS for initializers you want to expose
2025
@JS init(name: String) {
2126
self.name = name
2227
}
2328

29+
// Methods need the @JS attribute to be accessible from JavaScript
30+
// This method will be available as greeter.greet() in JavaScript
2431
@JS public func greet() -> String {
2532
"Hello, \(name)!"
2633
}

Examples/ImportTS/Package.swift

+7
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,13 @@ import PackageDescription
44

55
let package = Package(
66
name: "MyApp",
7+
platforms: [
8+
.macOS(.v10_15),
9+
.iOS(.v13),
10+
.tvOS(.v13),
11+
.watchOS(.v6),
12+
.macCatalyst(.v13),
13+
],
714
dependencies: [.package(name: "JavaScriptKit", path: "../../")],
815
targets: [
916
.executableTarget(

Examples/ImportTS/Sources/bridge.d.ts

+12
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,24 @@
1+
// Function definition to expose console.log to Swift
2+
// Will be imported as a Swift function: consoleLog(message: String)
13
export function consoleLog(message: string): void
24

5+
// TypeScript interface types are converted to Swift structs
6+
// This defines a subset of the browser's HTMLElement interface
37
type HTMLElement = Pick<globalThis.HTMLElement, "innerText"> & {
8+
// Methods with object parameters are properly handled
49
appendChild(child: HTMLElement): void
510
}
611

12+
// TypeScript object type with read-only properties
13+
// Properties will become Swift properties with appropriate access level
714
type Document = {
15+
// Regular property - will be read/write in Swift
816
title: string
17+
// Read-only property - will be read-only in Swift
918
readonly body: HTMLElement
19+
// Method returning an object - will become a Swift method returning an HTMLElement
1020
createElement(tagName: string): HTMLElement
1121
}
22+
// Function returning a complex object
23+
// Will be imported as a Swift function: getDocument() -> Document
1224
export function getDocument(): Document

Examples/ImportTS/Sources/main.swift

+13
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,26 @@
11
import JavaScriptKit
22

3+
// This function is automatically generated by the @JS plugin
4+
// It demonstrates how to use TypeScript functions and types imported from bridge.d.ts
35
@JS public func run() {
6+
// Call the imported consoleLog function defined in bridge.d.ts
47
consoleLog("Hello, World!")
58

9+
// Get the document object - this comes from the imported getDocument() function
610
let document = getDocument()
11+
12+
// Access and modify properties - the title property is read/write
713
document.title = "Hello, World!"
14+
15+
// Access read-only properties - body is defined as readonly in TypeScript
816
let body = document.body
917

18+
// Create a new element using the document.createElement method
1019
let h1 = document.createElement("h1")
20+
21+
// Set properties on the created element
1122
h1.innerText = "Hello, World!"
23+
24+
// Call methods on objects - appendChild is defined in the HTMLElement interface
1225
body.appendChild(h1)
1326
}

Plugins/BridgeJS/JS_ATTRIBUTE.md

+181
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,181 @@
1+
# The `@JS` Attribute
2+
3+
The `@JS` attribute is a key feature of the BridgeJS system that enables seamless interoperability between Swift and JavaScript. This document explains how to use the attribute effectively.
4+
5+
## Purpose
6+
7+
The `@JS` attribute serves two main purposes:
8+
9+
1. **Exporting Swift APIs to JavaScript** - Mark Swift functions, classes, and methods to make them callable from JavaScript
10+
2. **Importing JavaScript/TypeScript APIs to Swift** - Mark Swift functions that are implemented in JavaScript
11+
12+
## Syntax
13+
14+
The `@JS` attribute is placed before declarations:
15+
16+
```swift
17+
@JS public func myFunction() -> String { ... }
18+
19+
@JS class MyClass {
20+
@JS init(...) { ... }
21+
@JS func myMethod() { ... }
22+
}
23+
```
24+
25+
## Exporting Swift APIs to JavaScript
26+
27+
### Functions
28+
29+
To export a Swift function to JavaScript:
30+
31+
```swift
32+
@JS public func calculateArea(width: Int, height: Int) -> Int {
33+
return width * height
34+
}
35+
```
36+
37+
This function will be available in JavaScript as:
38+
39+
```javascript
40+
const area = exports.calculateArea(10, 20);
41+
```
42+
43+
Requirements:
44+
- Functions must be `public` to be exported
45+
- Only supported types can be used in parameters and return values
46+
- Functions are exported with their Swift names
47+
48+
### Classes
49+
50+
To export a Swift class to JavaScript:
51+
52+
```swift
53+
@JS class Person {
54+
var name: String
55+
var age: Int
56+
57+
@JS init(name: String, age: Int) {
58+
self.name = name
59+
self.age = age
60+
}
61+
62+
@JS public func greet() -> String {
63+
return "Hello, I'm \(name) and I'm \(age) years old."
64+
}
65+
66+
// This method won't be accessible from JavaScript (no @JS)
67+
func privateMethod() {
68+
// ...
69+
}
70+
}
71+
```
72+
73+
This class will be available in JavaScript as:
74+
75+
```javascript
76+
const person = new exports.Person("John", 30);
77+
console.log(person.greet()); // "Hello, I'm John and I'm 30 years old."
78+
```
79+
80+
Requirements:
81+
- The class itself needs the `@JS` attribute
82+
- Any initializers you want to expose need the `@JS` attribute
83+
- Any methods you want to expose need the `@JS` attribute and must be `public`
84+
- Properties are not directly exported (use getter/setter methods if needed)
85+
86+
## Importing TypeScript APIs to Swift
87+
88+
When using the BridgeJS import feature with a TypeScript definition file, the generated Swift code will include functions that can be called directly:
89+
90+
```typescript
91+
// In TypeScript definition
92+
export function consoleLog(message: string): void
93+
```
94+
95+
```swift
96+
// In Swift code
97+
consoleLog("Hello from Swift!")
98+
```
99+
100+
## Supported Types
101+
102+
The following Swift types can be used with `@JS`:
103+
104+
| Swift Type | JavaScript Type |
105+
|------------|----------------|
106+
| `Int` | `number` |
107+
| `Float` | `number` |
108+
| `Double` | `number` |
109+
| `Bool` | `boolean` |
110+
| `String` | `string` |
111+
| Classes | Objects |
112+
113+
## Best Practices
114+
115+
1. **Keep interfaces simple** - Use only supported types in exported functions and methods
116+
2. **Use consistent naming** - Follow Swift/JavaScript naming conventions
117+
3. **Document your APIs** - Add comments to explain usage for both languages
118+
4. **Test both sides** - Ensure your bridge works correctly in both Swift and JavaScript
119+
5. **Handle errors gracefully** - Check for errors when crossing language boundaries
120+
121+
## Examples
122+
123+
### Basic Function Export
124+
125+
```swift
126+
@JS public func sayHello(name: String) -> String {
127+
return "Hello, \(name)!"
128+
}
129+
```
130+
131+
### Class with Methods
132+
133+
```swift
134+
@JS class Counter {
135+
private var count: Int = 0
136+
137+
@JS init() {}
138+
139+
@JS public func increment() -> Int {
140+
count += 1
141+
return count
142+
}
143+
144+
@JS public func reset() {
145+
count = 0
146+
}
147+
148+
@JS public func getCount() -> Int {
149+
return count
150+
}
151+
}
152+
```
153+
154+
### Working with DOM Elements (Import)
155+
156+
```typescript
157+
// In bridge.d.ts
158+
export function getDocument(): Document
159+
160+
type Document = {
161+
getElementById(id: string): HTMLElement
162+
}
163+
164+
type HTMLElement = {
165+
innerText: string
166+
}
167+
```
168+
169+
```swift
170+
// In Swift
171+
let document = getDocument()
172+
let element = document.getElementById("my-element")
173+
element.innerText = "Updated from Swift!"
174+
```
175+
176+
## Limitations
177+
178+
- Asynchronous functions are not yet supported
179+
- Complex objects require manual conversion
180+
- Closures/callbacks have limited support
181+
- `throws` functions are not yet supported

0 commit comments

Comments
 (0)