EasyNodeEditorはSwift製のノードエディタ作成ライブラリです
ノードエディタとしての複雑なロジック部分を開発者ができるだけ意識せずに済むようにして、ノードの作成やその他の開発に集中できるようにする
というコンセプトで開発を進めています。
好きな名前でクラスを作成して、NodeModelBase
クラスを継承してください。
class YourOutputNode: NodeModelBase {
}
好きな名前で変数を作って、@objc
と@Output
をつけてください。
変数はいくらでも作ってOKで、作った分だけノードに出力が作成されます。
ちなみにノードに表示される出力名はこの変数名になるので、命名の際はそれだけ気をつけてください。
class YourOutputNode: NodeModelBase {
@objc @Output var output: Int = 3
}
EasyNodeEditorを使うときに引数としてクラスの型を渡してあげてください。
struct ContentView: View {
var body: some View {
EasyNodeEditor(nodeTypes: [YourOutputNode.self])
}
}
これだけ! 下の画像のようなノードが自動的に登録されます。 本当に簡単に出力ノードが作れたと思います。
class YourIONode: NodeModelBase {
}
出力の際には@objc
と@Output
をつけましたが、入力を作る際には@objc
と@Input
をつけます。
今回はinput
とoutput
という名前の変数を作りましたが、本当にここの名前はなんでもいいです。
class YourIONode: NodeModelBase {
@objc @Input var input: Int = 0
@objc @Output var output: Int = 0
}
processOnChange()
関数をoverrideして、中に好きな処理を書きましょう。
今回は入力値を5倍にするノードを作ってみます。
ちなみにprocessOnChange()
の中でinputの値を変えると無限ループが始まるので気をつけてください。
class YourIONode: NodeModelBase {
@objc @Input var input: Int = 0
@objc @Output var output: Int = 0
override func processOnChange() {
output = input * 5
}
}
出力ノードの時と同じように登録しましょう。
struct ContentView: View {
var body: some View {
EasyNodeEditor(nodeTypes: [YourOutputNode.self, YoutIONode.self])
}
}
class YourDisplayNode: NodeModelBase {
}
class YourDisplayNode: NodeModelBase {
@objc @Input var input: Int = 0
}
middleContent()
関数をoverrideして、好きにViewを定義しましょう。
SwiftUIの一般的な文法で書けるのでまじで簡単に書けるはずです。
実装の都合上AnyView
で囲ってあげるのだけお願いします。
class YourDisplayNode: NodeModelBase {
@objc @Input var input: Int = 0
override func middleContent() -> any View {
Text("number is now -> \(input)")
}
}
struct ContentView: View {
var body: some View {
EasyNodeEditor(nodeTypes: [YourOutputNode.self, YoutIONode.self, YourDisplayNode.self])
}
}
使い方1~3は読んだものとして進めますね。
インタラクションが必要なノード用には@Middle
を用意しています。
@Input
と@Middle
がついた変数の値が更新されるたびに、processOnChange()
メソッドが呼ばれるようになっています。
この例であるようなSlider
などバインディングが必要なUIパーツを使いたい場合は、@ObservableObject
を継承した新しいクラスを作成して、@Published
をつけた変数をその中で定義してください。
実装の都合上申し訳ないのですがNodeModelBase
クラスを継承したクラスの中ではUIパーツ用のバインディングができない仕様になっています。
ノードができたら今までと同じように登録しましょう。
class YourInteractiveNode: NodeModelBase {
@objc @Input var input: Int = 0
@objc @Middle var count: Int = 0
@objc @Output var output: Int = 0
override func processOnChange() {
output = input * count
}
override func middleContent() -> any View {
Group {
Slider(value: binding(\YourInteractiveNode.count), in: 0...100)
}
.frame(minWidth: 200, maxWidth: 200)
.fixedSize()
}
}
自分でもびっくりするくらい簡単です。こんなノードが登録されます。
import SwiftUI
import EasyNodeEditor
struct ContentView: View {
var body: some View {
EasyNodeEditor(nodeTypes: [YourOutputNode.self, YourIONode.self, YourDisplayNode.self, YourInteractiveNode.self])
}
}
class YourOutputNode: NodeModelBase {
@objc @Output var output: Int = 3
}
class YourIONode: NodeModelBase {
@objc @Input var input: Int = 0
@objc @Output var output: Int = 0
override func processOnChange() {
output = input * 5
}
}
class YourDisplayNode: NodeModelBase {
@objc @Input var input: Int = 0
override func middleContent() -> any View {
Text("number is now -> \(input)")
}
}
class YourInteractiveNode: NodeModelBase {
@objc @Input var input: Int = 0
@objc @Middle var count: Int = 0
@objc @Output var output: Int = 0
override func processOnChange() {
output = input * count
}
override func middleContent() -> any View {
Group {
Slider(value: binding(\YourInteractiveNode.count), in: 0...100)
}
.frame(minWidth: 200, maxWidth: 200)
.fixedSize()
}
}