LoginSignup
24
17

More than 1 year has passed since last update.

【macOS】SwiftUIだけでメニューバー常駐アプリを作ろう

Last updated at Posted at 2022-12-02

この記事はSwiftUI Advent Calendar 2022の3日目になります。
つよつよエンジニアの皆さんに紛れて参加するのは緊張しますが、気軽に読んでいただければと思います!

はじめに

10月にリリースされたmacOS Venturaから、SwiftUIのMenuBarExtra APIが使えるようになりました。

MenuBarExtraというのは、GoogleドライブやKarabiner-Elementsなどの常駐アプリで見かける、メニューバーの右上に並んだアイコンのことです。

今までは、SwiftUIを使ったアプリでもMenuBarExtraだけはAppKitで実装しなければならず、別れを告げたはずのAppDelegateを使用する必要がありました。

SwiftUIでのMenuBarExtraの作成は、細かい部分ではAppKitほどのカスタマイズ性はないものの、割と自由度があり、コードの記述量も大幅に短縮できます。

それでは、サンプルアプリを作りながら説明していきます。

ドキュメント

↓これはメニューバーが人格を持っているページ(?)です。(かわいい)

AppKitとの比較

こちらの記事に以前までの実装方法との比較が載っていますが、かなり簡潔に書けるようになっています。

プロジェクトの作成

Xcodeを開き、SwiftUIでプロジェクトを作成します。

試しにビルドすると、デフォルトではこのようなウィンドウが表示されます。

MenuBarExtraを表示させる

<アプリ名>App.swiftを編集していきます。

<アプリ名>App.swift
import SwiftUI

@main
struct MenuBarExtraSampleApp: App {
    var body: some Scene {
        WindowGroup {
            ContentView()
        }
+       MenuBarExtra("Sample", systemImage: "star.fill") {
+           MenuView()
+       }
    }
}

そして、メニューバーに表示するMenuView.swiftを作成します。

MenuView.swift
import SwiftUI

struct MenuView: View {
    var body: some View {
        Text("Hello, Swift!")
    }
}

struct MenuView_Previews: PreviewProvider {
    static var previews: some View {
        MenuView()
    }
}

これらを追加してビルドしてみると、先ほどのウィンドウに加えて、メニューバーにアイコンが表示されます。

このアイコンは従来のImage同様、SF Symbolsの中から指定できます。
(assetsが使えるかどうかは未確認です)

おまけ:ウィンドウとして表示

<アプリ名>App.swift
import SwiftUI

@main
struct MenuBarExtraSampleApp: App {
    var body: some Scene {
        WindowGroup {
            ContentView()
        }
        MenuBarExtra("Sample", systemImage: "star.fill") {
            MenuView()
        }
        .menuBarExtraStyle(.window)
    }
}

.menuBarExtraStyle(.window)モディファイアを設定すると、メニューではなくウィンドウが表示されるようになります。

テキストしか表示していないので違いが分かりづらいかもしれませんが、こんな感じです。

常駐アプリとして設定する

<アプリ名>App.swiftから、WindowGroupを削除します。
ContentView.swiftは使わないので消してしまって構いません。

<アプリ名>App.swift
import SwiftUI

@main
struct MenuBarExtraSampleApp: App {
    var body: some Scene {
-       WindowGroup {
-           ContentView()
-       }
        MenuBarExtra("Sample", systemImage: "star.fill") {
            MenuView()
        }
    }
}

しかしこのままではDockにアイコンが表示されてしまうので、以下のように編集し、Application is agent (UIElement)YESにします。

メニューバーに表示する内容をカスタムする

先ほどの書き方ではSF Symbolsのアイコンを表示することしかできませんでしたが、以下のように書くことでカスタムできます。

import SwiftUI

@main
struct MenuBarExtraSampleApp: App {
    var body: some Scene {
        MenuBarExtra {
            MenuView()
        } label: {
            Image(systemName: "message.fill")
            Text("10件")
        }
    }
}

ImageText以外に何が使えるかは確認できていません。

メニューの内容を作る

MenuView.swiftを編集していきます。

ちなみに、メニュー部分は従来のCommandMenuの場合と同じように表示されるようです。

MenuView.swift
struct MenuView: View {
    @State private var friendNames = ["太郎", "次郎", "花子", "Swift太郎"]
    
    var body: some View {
        ForEach(friendNames, id: \.self) { friendName in
            Button {
                // 処理
            } label: {
                Image(systemName: "person.fill")
                Text(friendName)
            }
        }
    }
}

ForEachでループを回すことで、ボタンを並べています。

ちなみに、メニューが表示されている間にプロパティが更新されても、メニューを再度開くまでUIは再描画されないようです(要出典)。

AppKitでは表示中にメニューを書き換えることもできましたが、たまに表示がバグるのと、操作中にメニューが変化する挙動もあまり良くない気がするので、特に問題はないと思います。

メニューを入れ子にする

MenuView.swift
struct MenuView: View {
    @State private var friendNames = ["太郎", "次郎", "花子", "Swift太郎"]
    
    var body: some View {
        ForEach(friendNames, id: \.self) { friendName in
            Menu {
                Button("電話する") {
                    // 処理
                }
                Button("メッセージを送る") {
                    // 処理
                }
            } label: {
                Image(systemName: "person.fill")
                Text(friendName)
            }
        }
    }
}

Menuを使うことで、入れ子にできます。

設定画面を作る

<アプリ名>App.swiftSettingsを追加します。

<アプリ名>App.swift
import SwiftUI

@main
struct MenuBarExtraSampleApp: App {
    var body: some Scene {
        MenuBarExtra {
            MenuView()
        } label: {
            Image(systemName: "message.fill")
            Text("10件")
        }
+       Settings {
+           SettingsView()
+       }
    }
}

続いて、SettingsView.swiftを作成します。

SettingsView.swift
import SwiftUI

struct SettingsView: View {
    @AppStorage("yourName") var yourName = "User"
    
    var body: some View {
        TabView {
            Form {
                TextField("あなたの名前", text: $yourName)
            }
            .padding(20)
            .tabItem {
                Label("一般", systemImage: "slider.horizontal.3")
            }
        }
        .frame(width: 500, height: .none)
    }
}

struct SettingsView_Previews: PreviewProvider {
    static var previews: some View {
        SettingsView()
    }
}

AppStorageを使って、ユーザー名が保存されるようにしてみました。

常駐アプリは左上のメニューが表示されないのでまだ設定を開く方法がありませんが、これからボタンを作っていきます。

よくあるボタンを作る

メニューバー常駐アプリでよく見かける、こんな感じのボタンを実装してみます。

MenuView.swiftを以下のように編集します。

MenuView.swift
struct MenuView: View {
    @State private var friendNames = ["太郎", "次郎", "花子", "Swift太郎"]
    
    var body: some View {
        ForEach(friendNames, id: \.self) { friendName in
            Menu {
                Button("電話する") {
                    // 処理
                }
                Button("メッセージを送る") {
                    // 処理
                }
            } label: {
                Image(systemName: "person.fill")
                Text(friendName)
            }
        }
        Divider()
        Button("設定...") {
            showSettingsWindow()
        }
        .keyboardShortcut(",")
        Button("MenuBarExtraSampleについて") {
            showAbout()
        }
        Button("MenuBarExtraSampleを終了...") {
            quitApp()
        }
        .keyboardShortcut("Q")
    }
    
    private func showSettingsWindow() {
        NSApp.sendAction(Selector(("showSettingsWindow:")), to: nil, from: nil)
        NSApp.activate(ignoringOtherApps: true)
    }
    
    private func showAbout() {
        NSApp.activate(ignoringOtherApps: true)
        NSApp.orderFrontStandardAboutPanel()
    }
    
    private func quitApp() {
        NSApplication.shared.terminate(nil)
    }
}

いい感じになりました。

それぞれのボタンもちゃんと動きます。

設定画面はタブが一つしかなくて寂しいですね…

おまけ:メニューにAppStorageの情報を表示してみる

MenuView.swift
struct MenuView: View {
    @State private var friendNames = ["太郎", "次郎", "花子", "Swift太郎"]
+   @AppStorage("yourName") var yourName = "User"
    
    var body: some View {
+       Text("あなたの名前:\(yourName)")
        ForEach(friendNames, id: \.self) { friendName in
            Menu {
                Button("電話する") {
                    // 処理
                }
                Button("メッセージを送る") {
                    // 処理
                }
            } label: {
                Image(systemName: "person.fill")
                Text(friendName)
            }
        }
        // 省略
    }
    // 省略
}

設定画面で変更した値がメニューに反映できるようになりました。

おわりに

AppKitではUIの再描画を考慮する必要がありましたが、SwiftUIでは基本的に勝手にやってくれるのでとても楽です。

もちろん、SwiftUI特有の罠というのも存在しますが、シンプルなメニューバーアプリを作るのであればSwiftUIはかなり使いやすいと感じました。

この記事が皆さんのアプリ開発の参考になれば幸いです。
もしよければ最後に💚を押していただけると嬉しいです!

24
17
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
24
17