Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Introduce BridgeJS, a declarative JS interop system #330

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ jobs:
make regenerate_swiftpm_resources
git diff --exit-code Sources/JavaScriptKit/Runtime
- run: swift test --package-path ./Plugins/PackageToJS
- run: swift test --package-path ./Plugins/BridgeJS

native-build:
# Check native build to make it easy to develop applications by Xcode
Expand Down
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -10,3 +10,4 @@ xcuserdata/
Examples/*/Bundle
Examples/*/package-lock.json
Package.resolved
Plugins/BridgeJS/Sources/JavaScript/package-lock.json
25 changes: 25 additions & 0 deletions Examples/ExportSwift/Package.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
// swift-tools-version:6.0

import PackageDescription

let package = Package(
name: "MyApp",
platforms: [
.macOS(.v14)
],
dependencies: [.package(name: "JavaScriptKit", path: "../../")],
targets: [
.executableTarget(
name: "MyApp",
dependencies: [
"JavaScriptKit"
],
swiftSettings: [
.enableExperimentalFeature("Extern")
],
plugins: [
.plugin(name: "BridgeJS", package: "JavaScriptKit")
]
)
]
)
34 changes: 34 additions & 0 deletions Examples/ExportSwift/Sources/main.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
import JavaScriptKit

// Mark functions you want to export to JavaScript with the @JS attribute
// This function will be available as `renderCircleSVG(size)` in JavaScript
@JS public func renderCircleSVG(size: Int) -> String {
let strokeWidth = 3
let strokeColor = "black"
let fillColor = "red"
let cx = size / 2
let cy = size / 2
let r = (size / 2) - strokeWidth
var svg = "<svg width=\"\(size)px\" height=\"\(size)px\">"
svg +=
"<circle cx=\"\(cx)\" cy=\"\(cy)\" r=\"\(r)\" stroke=\"\(strokeColor)\" stroke-width=\"\(strokeWidth)\" fill=\"\(fillColor)\" />"
svg += "</svg>"
return svg
}

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

// Use @JS for initializers you want to expose
@JS init(name: String) {
self.name = name
}

// Methods need the @JS attribute to be accessible from JavaScript
// This method will be available as greeter.greet() in JavaScript
@JS public func greet() -> String {
"Hello, \(name)!"
}
}
12 changes: 12 additions & 0 deletions Examples/ExportSwift/index.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
<!DOCTYPE html>
<html>

<head>
<title>Getting Started</title>
</head>

<body>
<script type="module" src="index.js"></script>
</body>

</html>
14 changes: 14 additions & 0 deletions Examples/ExportSwift/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import { init } from "./.build/plugins/PackageToJS/outputs/Package/index.js";
const { exports } = await init({});

const Greeter = exports.Greeter;
const greeter = new Greeter("World");
const circle = exports.renderCircleSVG(100);

// Display the results
const textOutput = document.createElement("div");
textOutput.innerText = greeter.greet()
document.body.appendChild(textOutput);
const circleOutput = document.createElement("div");
circleOutput.innerHTML = circle;
document.body.appendChild(circleOutput);
29 changes: 29 additions & 0 deletions Examples/ImportTS/Package.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
// swift-tools-version:6.0

import PackageDescription

let package = Package(
name: "MyApp",
platforms: [
.macOS(.v10_15),
.iOS(.v13),
.tvOS(.v13),
.watchOS(.v6),
.macCatalyst(.v13),
],
dependencies: [.package(name: "JavaScriptKit", path: "../../")],
targets: [
.executableTarget(
name: "MyApp",
dependencies: [
"JavaScriptKit"
],
swiftSettings: [
.enableExperimentalFeature("Extern")
],
plugins: [
.plugin(name: "BridgeJS", package: "JavaScriptKit")
]
)
]
)
24 changes: 24 additions & 0 deletions Examples/ImportTS/Sources/bridge.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
// Function definition to expose console.log to Swift
// Will be imported as a Swift function: consoleLog(message: String)
export function consoleLog(message: string): void

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

// TypeScript object type with read-only properties
// Properties will become Swift properties with appropriate access level
type Document = {
// Regular property - will be read/write in Swift
title: string
// Read-only property - will be read-only in Swift
readonly body: HTMLElement
// Method returning an object - will become a Swift method returning an HTMLElement
createElement(tagName: string): HTMLElement
}
// Function returning a complex object
// Will be imported as a Swift function: getDocument() -> Document
export function getDocument(): Document
26 changes: 26 additions & 0 deletions Examples/ImportTS/Sources/main.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import JavaScriptKit

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

// Get the document object - this comes from the imported getDocument() function
let document = getDocument()

// Access and modify properties - the title property is read/write
document.title = "Hello, World!"

// Access read-only properties - body is defined as readonly in TypeScript
let body = document.body

// Create a new element using the document.createElement method
let h1 = document.createElement("h1")

// Set properties on the created element
h1.innerText = "Hello, World!"

// Call methods on objects - appendChild is defined in the HTMLElement interface
body.appendChild(h1)
}
16 changes: 16 additions & 0 deletions Examples/ImportTS/index.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
<!DOCTYPE html>
<html>

<head>
<title>Getting Started</title>
</head>

<body>
<script type="module" src="index.js"></script>

<div id="exports-result"></div>
<div id="imports-result"></div>
<pre id="code"></pre>
</body>

</html>
13 changes: 13 additions & 0 deletions Examples/ImportTS/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import { init } from "./.build/plugins/PackageToJS/outputs/Package/index.js";
const { exports } = await init({
imports: {
consoleLog: (message) => {
console.log(message);
},
getDocument: () => {
return document;
},
}
});

exports.run()
11 changes: 10 additions & 1 deletion Examples/Multithreading/Package.resolved
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
{
"originHash" : "e66f4c272838a860049b7e3528f1db03ee6ae99c2b21c3b6ea58a293be4db41b",
"originHash" : "072d03a6e24e01bd372682a6090adb80cf29dea39421e065de6ff8853de704c9",
"pins" : [
{
"identity" : "chibi-ray",
Expand All @@ -8,6 +8,15 @@
"state" : {
"revision" : "c8cab621a3338dd2f8e817d3785362409d3b8cf1"
}
},
{
"identity" : "swift-syntax",
"kind" : "remoteSourceControl",
"location" : "https://github.com/swiftlang/swift-syntax",
"state" : {
"revision" : "0687f71944021d616d34d922343dcef086855920",
"version" : "600.0.1"
}
}
],
"version" : 3
Expand Down
48 changes: 47 additions & 1 deletion Package.swift
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// swift-tools-version:6.0

import CompilerPluginSupport
import PackageDescription

// NOTE: needed for embedded customizations, ideally this will not be necessary at all in the future, or can be replaced with traits
Expand All @@ -9,12 +10,24 @@ let useLegacyResourceBundling =

let package = Package(
name: "JavaScriptKit",
platforms: [
.macOS(.v10_15),
.iOS(.v13),
.tvOS(.v13),
.watchOS(.v6),
.macCatalyst(.v13),
],
products: [
.library(name: "JavaScriptKit", targets: ["JavaScriptKit"]),
.library(name: "JavaScriptEventLoop", targets: ["JavaScriptEventLoop"]),
.library(name: "JavaScriptBigIntSupport", targets: ["JavaScriptBigIntSupport"]),
.library(name: "JavaScriptEventLoopTestSupport", targets: ["JavaScriptEventLoopTestSupport"]),
.plugin(name: "PackageToJS", targets: ["PackageToJS"]),
.plugin(name: "BridgeJS", targets: ["BridgeJS"]),
.plugin(name: "BridgeJSCommandPlugin", targets: ["BridgeJSCommandPlugin"]),
],
dependencies: [
.package(url: "https://github.com/swiftlang/swift-syntax", "600.0.0"..<"601.0.0")
],
targets: [
.target(
Expand Down Expand Up @@ -98,7 +111,40 @@ let package = Package(
capability: .command(
intent: .custom(verb: "js", description: "Convert a Swift package to a JavaScript package")
),
sources: ["Sources"]
path: "Plugins/PackageToJS/Sources"
),
.plugin(
name: "BridgeJS",
capability: .buildTool(),
dependencies: ["BridgeJSTool"],
path: "Plugins/BridgeJS/Sources/BridgeJSBuildPlugin"
),
.plugin(
name: "BridgeJSCommandPlugin",
capability: .command(
intent: .custom(verb: "bridge-js", description: "Generate bridging code"),
permissions: [.writeToPackageDirectory(reason: "Generate bridging code")]
),
dependencies: ["BridgeJSTool"],
path: "Plugins/BridgeJS/Sources/BridgeJSCommandPlugin"
),
.executableTarget(
name: "BridgeJSTool",
dependencies: [
.product(name: "SwiftParser", package: "swift-syntax"),
.product(name: "SwiftSyntax", package: "swift-syntax"),
.product(name: "SwiftBasicFormat", package: "swift-syntax"),
.product(name: "SwiftSyntaxBuilder", package: "swift-syntax"),
],
path: "Plugins/BridgeJS/Sources/BridgeJSTool"
),
.testTarget(
name: "BridgeJSRuntimeTests",
dependencies: ["JavaScriptKit"],
exclude: ["Generated/JavaScript"],
swiftSettings: [
.enableExperimentalFeature("Extern")
]
),
]
)
29 changes: 29 additions & 0 deletions Plugins/BridgeJS/Package.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
// swift-tools-version: 6.0

import PackageDescription

let package = Package(
name: "BridgeJS",
platforms: [.macOS(.v13)],
dependencies: [
.package(url: "https://github.com/swiftlang/swift-syntax", from: "600.0.1")
],
targets: [
.target(name: "BridgeJSBuildPlugin"),
.target(name: "BridgeJSLink"),
.executableTarget(
name: "BridgeJSTool",
dependencies: [
.product(name: "SwiftParser", package: "swift-syntax"),
.product(name: "SwiftSyntax", package: "swift-syntax"),
.product(name: "SwiftBasicFormat", package: "swift-syntax"),
.product(name: "SwiftSyntaxBuilder", package: "swift-syntax"),
]
),
.testTarget(
name: "BridgeJSToolTests",
dependencies: ["BridgeJSTool", "BridgeJSLink"],
exclude: ["__Snapshots__", "Inputs"]
),
]
)
Loading