1
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

WKWebView 組み込み時のハマりごと

Posted at

次のような理由などで、モバイルアプリにウェブビューを組み込んで HTML ファイルを表示することがあるでしょう。

  • 既存サービス等の HTML ファイルを活用する
  • 最軽量かつ最小限のクロスプラットフォームを実現する

モバイルアプリエンジニアは、必ずしも HTML や JavaScript に詳しいわけではないため、問題に出会いやすいです。今回は iOS アプリで WKWebView を組み込んだときのハマりごと、その解決方法を紹介します。

問題設定

次の HTML ファイルを想定します。この HTML は JavaScript 関数 window.setText() を持ち、その関数で HTML 内のテキスト表示を制御します。なお、この HTML は ChatGPT に作ってもらいました。

埋め込まれる HTML ファイル
<!DOCTYPE html>
<html lang="ja">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>WKWebView Example</title>
    <script>
        function setText(text) {
            document.getElementById("display").innerText = text;
        }
    </script>
</head>
<body>
    <h1>WKWebView Test</h1>
    <p id="display">デフォルトテキスト</p>
</body>
</html>

次の画像例のように、JavaScript の関数 window.setText() を実行するとテキスト表示が更新されます。

スクリーンショット 2025-03-06 18.48.54.png

この HTML を iOS アプリに組み込んで、JavaScript の関数を実行してテキスト表示を制御したいです。この場合、次のような Swift 関数を実装すれば、アプリから JavaScript 関数を実行できます。なお、この関数内の webView は WKWebView のインスタンスです。

アプリから JavaScript 関数を実行する
func setWebViewText(text: String) {
    let code = """
    const text = "\(text)";
    window.setText(text);
    """
    webView.evaluateJavaScript(code) { _, error in
        if let error {
            print(error) // このエラー処理は暫定処理です
        }
    }
}

この関数 setWebViewText(text:) を利用すれば、JavaScript 関数が実行されて表示制御されます。しかしながら、いくつかの問題が潜んでいます。この関数を安心して使えるようにしましょう。

以降で説明する内容は iOS / Swift 側の実装で、主に次の3つの自作関数を対象とします。HTML ファイルの修正や編集はしません。

関数名 目的
setUpWebView() webView の初期化や addSubView(_:) などを行う
loadWebView() webView で HTML ファイル(index.html)を読み込む
setWebViewText(text: ) アプリから JavaScript 関数を実行して、文字を表示する

関数の呼び出しタイミング

webView で HTML ファイル(index.html)を読み込むには、次のような実装で実現できます。

index.html の読み込み
func loadWebView() {
    guard let url = Bundle.main.url(forResource: "index", withExtension: "html") else {
        assertionFailure("index.html not found.")
        return
    }
    let request = URLRequest(url: url)
    webView.load(request)
}

準備できたら、自作関数をそれぞれ実行していきます。HTML を読み込んだ WebView が表示されて「こんにちは世界」という文字が表示される…はずです。

HTML の読み込みと文字表示の関数を実行する
setUpWebView() // webView の初期設定
loadWebView()  // html の読み込み
setWebViewText(text: "こんにちは世界")

次のようなエラーが起こります(元のエラー文は改行無しですが、見やすくするため改行しています)。

Error Domain=WKErrorDomain Code=4 "A JavaScript exception occurred" UserInfo=
{WKJavaScriptExceptionLineNumber=2, WKJavaScriptExceptionMessage=TypeError:
window.setText is not a function. (In 'window.setText(text)', 'window.setText'
is undefined), WKJavaScriptExceptionColumnNumber=19, WKJavaScriptExceptionSourceURL=
undefined, NSLocalizedDescription=A JavaScript exception occurred}

このエラーは 'window.setText' is undefined と書かれているとおり、定義したはずの関数が未定義になっています。HTML の読み込みが完了しないと、定義された関数は実行できません。そこで、読み込み完了を待ってから、関数を実行します。

extension ViewController {
    override func viewDidLoad() {
        super.viewDidLoad()
        setUpWebView()
        loadWebView()
        setWebViewText(text: "こんにちは didFinish 前の世界") // JS で失敗する
    }
    
    // webView を初期設定する関数
    func loadWebView() {
        // ...
        webView.navigationDelegate = self // 読込みイベントを利用するためのデリゲート
        // ...
    }
}

extension ViewController: WKNavigationDelegate {

    // 読込み完了
    public func webView(_ webView: WKWebView, didFinish navigation: WKNavigation!) {
        setWebViewText(text: "こんにちは didFinish 後の世界")
    }

    // 読込み失敗
    public func webView(_ webView: WKWebView, didFail navigation: WKNavigation!, withError error: any Error) {
        print("didFail \(error)")
    }
}

HTML に組み込まれている関数を実行する場合、その HTML の読み込みが完了してから実行するようにしてください。この例では、完了通知のデリゲート内で実行していますが、任意なタイミングで実行する場合は完了状態(関数実行可能)フラグを変数に保存しましょう。

完了イベントが独自の場合

HTML の実装や関数の特性によってはライフサイクルが独自なものもあるでしょう。その場合、HTML が発行するイベントを WKWebView が受け取ることで解決する場合があります(HTML の仕様は、設計者に確認しましょう)。説明する実装例では、独自イベントを HTML に実装するのは手間なので、一般的なイベント laod、error、そして unhandledrejection を監視しました。

extension ViewController {
    private func setUpWebView() {
        // アプリで対応できるイベント名を設定する
        let eventHandlerName = "eventHandlerName"
        
        let contentController = WKUserContentController()
        
        // イベント受け取りの設定
        contentController.add(self, name: eventHandlerName)
        
        // HTML に埋め込む JavaScript のコードの設定
        let source = """
        window.addEventListener('load', (event) => {
            const message = {type: 'onLoad', message: event.message};
            const jsonString = JSON.stringify(message);
            window.webkit.messageHandlers.\(eventHandlerName).postMessage(jsonString);
        });
        window.addEventListener('error', (event) => {
            const message = {type: 'onError', message: event.error?.message ?? event.message};
            const jsonString = JSON.stringify(message);
            window.webkit.messageHandlers.\(eventHandlerName).postMessage(jsonString);
        });
        window.addEventListener('unhandledrejection', (event) => {
            const message = {type: 'onException', message: `${event.reason}`};
            const jsonString = JSON.stringify(message);
            window.webkit.messageHandlers.\(eventHandlerName).postMessage(jsonString);
        });
        """
        let userScript = WKUserScript(source: source, injectionTime: .atDocumentEnd, forMainFrameOnly: true)
        contentController.addUserScript(userScript)
        
        let config = WKWebViewConfiguration()
        config.userContentController = contentController
        
        webView = WKWebView(frame: .zero, configuration: config)
        // ...
    }
}

extension ViewController: WKScriptMessageHandler {
    func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) {
        print("didReceive \(message.name), \(message.body),")
    }
}

このように、アプリから JavaScript のコードを HTML に埋め込むことができます。イベント取得で利用しましたが、簡単な機能修正などもできます。

JavaScript に渡す文字列のエンコード

関数 setWebViewText(text:) は Swift の文字列を JavaScript に直接渡しています。渡す文字列に特殊文字(", \など)が含まれていると、エラーになります。

Error Domain=WKErrorDomain Code=4 "A JavaScript exception occurred" UserInfo=
{WKJavaScriptExceptionLineNumber=1, WKJavaScriptExceptionMessage=SyntaxError:
Unexpected EOF, WKJavaScriptExceptionColumnNumber=0, WKJavaScriptExceptionSourceURL=
file:///Users/********/Library/Developer/CoreSimulator/Devices/BA932DFF-6DD8-
4557-8DD8-B9537DA179F5/data/Containers/Bundle/Application/A60A4F3F-ED34-4B97-
B160-846B708A7C22/SampleWKWebViewApp.app/index.html, NSLocalizedDescription=
A JavaScript exception occurred}

Swift と JavaScript で文字の意味が異なるため、解釈不一致が起こっています。これを防ぐため、文字列をエスケープしてから渡します。そして、JavaScript のコードでアンエスケープした文字列を利用します。

アプリから JavaScript 関数を実行する
func setWebViewText(text: String) {
    guard let encodedText = text.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {
        return
    }
    // ↓ このコードブロックは、もはや Swift ではなく JavaScript です!
    let code = """
    const text = decodeURIComponent("\(escapedText)");
    window.setText(text);
    """
    webView.evaluateJavaScript(code) { _, error in
        if let error {
            print(error)
        }
    }
}

今回の例は文字列を渡してますが、オブジェクトを渡したいときもあるでしょう。Swift と JavaScript でオブジェクトは異なるので、オブジェクトを渡すことはできません。その場合、オブジェクトを JSON 文字列に変換して、Swift から JavaScript に渡ます。そして、JavaScript 側でオブジェクトに戻して利用します。このときもエスケープ・アンエスケープを忘れずに行いましょう。

関数実行のスコープ

アプリでは、必要に応じて、文字更新関数 setWebViewText(text:)を呼びます。次のような例があるのかは不明ですが、関数を連続して実行してます。

更新関数を複数回実行する
setWebViewText(text: "テキスト1")
setWebViewText(text: "テキスト2")

残念ながら、エラーが起こります。

Error Domain=WKErrorDomain Code=4 "A JavaScript exception occurred" UserInfo=
{WKJavaScriptExceptionLineNumber=0, WKJavaScriptExceptionMessage=SyntaxError:
Can't create duplicate variable: 'text', WKJavaScriptExceptionColumnNumber=0, 
NSLocalizedDescription=A JavaScript exception occurred}

連続した関数の実行は、WebView では次のような JavaScript として解釈されます。同じスコープで実行されています。つまり、同名変数が再定義されたため、JavaScript でエラーが起こりました。

複数実行時の JavaScript 側のコード
// Swift で setWebViewText(text: "テキスト1") を実行した
const text = decodeURIComponent("テキスト1");
window.setText(text);

// Swift で setWebViewText(text: "テキスト2") を実行した
const text = decodeURIComponent("テキスト2"); // ← ここでエラー。同名変数の定義になった。
window.setText(text);

2つ目の定数 text を別名に変えれば実行できますが、実行ごとにユニークな命名をするのは実装不可です。そこで、関数を実行するスコープを分けることで回避します。

アプリから JavaScript 関数を実行する(スコープ対応)
func setWebViewText(text: String) {
    guard let encodedText = text.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {
        return
    }
    let code = """
    try {
        const text = decodeURIComponent("\(escapedText)");
        window.setText(text);
    } catch (error) {
        throw error
    }
    """
    webView.evaluateJavaScript(code) { _, error in
        if let error {
            print(error)
        }
    }
}

これによって、関数を何回実行しても try-catch のスコープ内で実行されるので、名前が衝突することはありません。

この問題は私も実際にハマり、大いに悩みました。evaluateJavaScript は実行ごとにスコープを分けて、よしなに実行してくれてるだろう、と思っていたのが原因でした。

関数の完了結果を待つ

これまでの関数例は実行するだけで、その実行結果は取得してきませんでした。この例では起こらないですが、実行順序が大切な場合、エラー結果を取得したい場合を考えます。

コールバック

クロージャを使って、完了コールバックを作成します。完了状態は取得できますが、順番に実行したい場合はコールバック地獄になるので、お勧めはしません。

コールバックで完了状態を取得する
func setWebViewText(text: String, completion: @escaping ((_ error: Error?) -> Void)) {
    // 略...
    webView.evaluateJavaScript(code) { _, error in
        completion(error)
    }
}

非同期関数にする

Swift Concurrency を使って、非同期関数にします。

非同期関数にする
func setWebViewText(text: String) async throws {
    // 略...
    try await withCheckedThrowingContinuation { [weak self] continuation in
        guard let webView = self?.webView else { return }
        webView.evaluateJavaScript(code) { _, error in
            if let error {
                continuation.resume(throwing: error)
            } else {
                continuation.resume()
            }
        }
    } as Void
}

次のように順番ごとに実行できるようになりました。

func setWebViewTextSequence() {
    Task { @MainActor in
        do {
            try await setWebViewText(text: "await 1")
            try await setWebViewText(text: "await 2")
            try await setWebViewText(text: "await 3")
        } catch {
            print(error)
        }
    }
}

HTML の動作検証

HTML のデバッグは iOS アプリエンジニアの責務を越えますが、動作検証はしたいです。ブラウザの開発者モードやインスペクターを利用すると、動作確認ができます。

開発者モード

HTML を Chrome や Safari などのブラウザで開き、メニューから開発者モードを選択します。HTML ファイルで定義されたコンポーネントなどが確認できます。また、コンソールから関数を実行して、動作確認ができます。

開発者モード コンソールから関数を実行した
スクリーンショット 2025-03-06 18.48.47.png スクリーンショット 2025-03-06 18.48.54.png

ブラウザの例で Chrome も挙げましたが、ブラウザごとにエンジンが異なるので、iOS アプリなら Safari で確認するのが望ましいです。なお、Safari の開発者モードは設定から有効します。

スクリーンショット 2025-03-06 18.25.01.png

インスペクター

アプリに組み込んだ WebView(HTML)は Safari のインスペクターで確認できます。コードからインスペクターできるように設定します。

private func setUpWebView() {
    // 略...
    
#if DEBUG
    // iOS 16.4 以上はこの設定が必要です
    webView.isInspectable = true
#endif
}

実機の場合は、さらに、端末の設定から「Safari」→「詳細」に進んで、「Webインスペクタ」を有効にします。

そして、Safari の「開発」からシミュレータまたは実機の WebView(HTML)を選択します。複数の WebView があると多くの候補が表示されます。同名が並んでどれが確認したいものか分かりづらいですが、マウスオーバーでコンポーネントが選択される(青くなる)ので、確認したいものを選択します。

スクリーンショット 2025-03-06 18.44.06.png

インスペクタでは、開発者モードと同様な確認ができます。

スクリーンショット 2025-03-06 18.45.39.png

まとめ

本記事は iOS アプリに WKWebView を組み込んだときに、起こりうる問題とその解決方法を紹介しました。今回の実装例は次のリポジトリにあります。

WKWebView はネイティブのコンポーネントといえど、内部は Web の世界が広がっています。今回紹介した以外にも HTML 固有の問題が起こることもあり、WKWebView を利用した開発は大変です。分からない原因で悩み、なぜかアレコレしたら治ったこともあります。WKWebView、ムズい…

みなさんも WKWebView に関する知見の記事を書いてくださいね。

1
0
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
1
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?