SwiftUI macOSでmakeKeyAndOrderFrontが機能しない場合

SwiftUI macOSでmakeKeyAndOrderFrontが機能しない場合

Xcode 12.3macOSアプリのプロジェクトをSwiftUIで作成した場合、windowをClose()してしまうとmakeKeyAndOrderFront()が機能しませんでした。makeKeyAndOrderFrontは、ウィンドウを表示させるメソッドでNSWindowに入っています。

macOSアプリ内でウィンドウを閉じた場合再表示できない原因は、NSWindowControllerで所有していないwindowは、ウィンドウが閉じられると解放してしまうためでした。

これらを回避してmakeKeyAndOrderFrontを機能させる方法の一つは、windowを閉じない処理を入れた[NSWindowをラップしたクラス]を作ってclose()をoverrideする。

class Window: NSWindow {
    override func close() {
        self.orderOut(NSApp)
    }
}

別の方法としては、SwiftUIでアプリ作成の場合ObservableObject継承したクラスを作成すると思うので、そのクラス内にでもNSWindowControllerを入れてしまう。

windowController = NSWindowController(window: window)

NSWindowControllerinitpublic init(window: NSWindow?)があるのでそこにAppDelegateなどで作ったwindowを入れます。

この2つの方法どちらかでmakeKeyAndOrderFrontが機能し閉じてもウィンドウが再表示されました。

SwiftUI JSONキーにハイフン(-)が付いている場合の対処 hyphen

hyphen json JSONEncoder

前回、SwiftJSONDecoderを使った処理方法を書きました。今回JSONキーにハイフンが混ざった処理をご紹介します。ハイフン(hyphen)が入った変数を作成することができません。ビルドエラーになってしまいます。しかしWeb上のAPIにはハイフンが混ざったキーもよく見かけます。例えば下記のようなJSONです。

{ "name-1": "Lenina Crowne" }

この場合”name1“なので下記のようにしたいのですがこれではビルドできません。

ビルドエラー
ビルドエラー

JSONキーにハイフンが含まれる場合CodingKeyが含まれたenum宣言を構造体に埋めてキー名の変換を行います。

struct TJson : Codable {
    var name1: String
    enum CodingKeys : String, CodingKey {
        case name1 = "name-1"
    }
}

enumの名称は「CodingKeys」とする必要があるようです。他の名前ではビルドは成功するものの実行するとエラーになりました。

struct TCodingKeyJSONView: View {
    @State var json1 = """
        { "name-1": "Lenina Crowne" }
    """
    var body: some View {
        Text(jsonConvert(json1).name1)
    }
    func jsonConvert(_ s1: String) -> TJson {
        do {
            return try JSONDecoder().decode(TJson.self, from: s1.data(using: .utf8)! )
        } catch {
            return TJson(name1: "")
        }
    }
}

ハイフンキーが入ったJSONの処理ができました。
ハイフンキーが入ったJSONの処理ができました。

JSONEncoderを使ってTJsonからJSON文字列に変換も試してみました。

struct TCodingKeyJSONView: View {
    var body: some View {
        Text(tJsonToJsonStr(TJson(name1: "Bernard Marx")))
    }
    func tJsonToJsonStr(_ j: TJson) -> String{
        let enc = JSONEncoder()
        do {
            let d1 = try enc.encode(j)
            return String(data: d1, encoding: .utf8)!
        } catch {
            return "{}"
        }
    }
}

JSONEncoderもうまく動きました。

ハイフンキー付きJSON作成する場合のJSONEncoder
ハイフンキー付きJSON作成する場合のJSONEncoder
2022 MJELD TECHNOLOGIES. ALL RIGHTS RESERVED