Search code examples
wkwebviewaddeventlistenercalendly

WKWebView addEventListener not receiving Calendly events


I'm trying to use Calendly within a WKWebView and receive an event when the user has created an appointment. The app is successfully receiving message events, however Calendly events are not appearing.

Here's the code:

import UIKit
import WebKit

class ViewController: UIViewController, WKScriptMessageHandler {
    
    var webView: WKWebView!
    var count = 0
    
    var html = """
    <html>
    <head>
        <meta name='viewport' content='width=device-width' />
    </head>
    
    <!-- Calendly inline widget begin -->
    
    <div class="calendly-inline-widget" data-auto-load="false">
        <script type="text/javascript" src="https://assets.calendly.com/assets/external/widget.js"></script>
        <script>
        Calendly.initInlineWidget({
            url: 'https://calendly.com/XXX',
            prefill: {
                name: "John Doe",
                email: "[email protected]",
                customAnswers: {
                    a1: "yes"
                }
            }
        });
        </script>
    </div>
    <!-- Calendly inline widget end -->
    </html>
    """

    override func viewDidLoad() {
        let config = WKWebViewConfiguration()
        let js =

        """
        function isCalendlyEvent(e) {
          return e.data.event &&
                 e.data.event.indexOf('calendly') === 0;
        };
         
        window.addEventListener(
          'message',
          function(e) {
            if (isCalendlyEvent(e)) {
              console.log("!!!!!!!!!!!!!!!!!!!!!!!!!!!");
              console.log(e.data);
            }
          }
        );
        
        function isCalendlyEvent(e) {
                  console.log('testing' + e.name);
                  return e.data.event &&
                         e.data.event.indexOf('calendly') === 0;
                };
        window.addEventListener('message', function(e){
            console.log('In listener. Event.type: ' + event.type +
                ' e.data.event: ' + e.data.event + ' event: ' + JSON.stringify(e.data));
            if (isCalendlyEvent(e)) {
                console.log('calendly event!!!!');
                window.webkit.messageHandlers.clickListener.postMessage('Calendly:' + e.data);
            } else {
                window.webkit.messageHandlers.clickListener.postMessage('Other:' + e.data);
            }
        });
        """

        let script = WKUserScript(source: js, injectionTime: .atDocumentEnd, forMainFrameOnly: false)

        config.userContentController.addUserScript(script)
        config.userContentController.add(self, name: "clickListener")

        webView = WKWebView(frame: view.bounds, configuration: config)

        view.addSubview(webView!)
        self.webView.loadHTMLString(self.html, baseURL: nil)
    }
    
    func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) {
        count = count + 1
        print("Msg \(count): \(message.body) ")
    }
}


Solution

  • The calendly message is only sent when the url includes the embed_domain query parameter. When Calendly.initInlineWidget is called inside of WKWebView the embed_domain query parameter is set to undefined.

    To resolve this issue, you can update the url to include an embed_domain parameter:

    Calendly.initInlineWidget({
                url: 'https://calendly.com/XXX?embed_domain=example',
                prefill: {
                    name: "John Doe",
                    email: "[email protected]",
                    customAnswers: {
                        a1: "yes"
                    }
                }
            });