Having trouble converting an Obj-C observer to Swift syntax

329 Views Asked by At

The goal here is to detect whenever the URL of the webView changes. Since the didCommit and didStartProvisionalNavigation functions of the WKWebView class don't always seem to fire when working with certain elements within a WebView, I now have to add an observer to pick up when the WebView's URL value changes.

So far I have created my own Notification extension with name checkURL:

Swift Progress

extension Notification.Name {
    static let checkURL = Notification.Name("checkURL")
}

NotificationCenter.default.post(name: .checkURL, object: nil)
NotificationCenter.default.addObserver(self, selector: #selector(getter: webView.url), name: .checkURL, object: webView.url)

Objective-C Solution (KVO)

// Add an observer
[webView_ addObserver:self forKeyPath:@"URL" options:NSKeyValueObservingOptionNew context:NULL];

// Method that gets called when the URL value changes
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    // Code
}

How would I go about applying that Objective-C code to fire a function whenever the URL value changes? I am still trying to wrap my head around Notifications and how they work with object values, so any explanation would be very helpful as well!

Thank you for reading!

EDIT: Just to clear things up, I will reword my question:

Since the didCommit and didStartProvisionalNavigation don't always seem to fire (especially when working with JavaScript-based sites, as well as PHP – you'll constantly see URLs being changed with symbols such as # and whatnot); however, as stated before, the built-in WKWebView functions don't seem to catch those. So what I am trying to do here is find a workaround to catch any sort of changes made to the URL, regardless of if it's just a #, etc. And preferably keeping the code entirely Swift – as I am still learning. :)

1

There are 1 best solutions below

7
On

First, forgive the (quite a few) assumptions in this answer, but I can't comment asking for clarification.

Second, I would recommend giving a good read to the NSNotificationCenter reference. Also if I remember correctly, this writeup about notifications was quite helpful for me at first.

Now to the actual code. If I understand your predicament (other than having to use Objective-C and Swift together, which is always... fun), you are trying to fire a notification on an URL change so that your Swift code can do something with the new URL.

As your swift code is listening for the notification, you don't want to post it there. Rather, the notification should be posted in the KVO callback like this:

// I have also changed the 'URL' to 'url' as I suspect that KVO is case-sensitive
// and replaced NULL with nil as it is generally a better practice to use it.
[webView_ addObserver:self forKeyPath:@"url" options:NSKeyValueObservingOptionNew context:nil];

// Method that gets called when the URL value changes
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    if (sender == webView_ && [keyPath isEqualToString:@"url"]) {
        [[NSNotificationCenter defaultCenter] postNotificationName:@"checkURL" object:self];
    }
}

This would send the notification to any observers, like your Swift class. The Swift code would have to be changed to something like:

extension Notification.Name {
    static let checkURL = Notification.Name("checkURL")
}

// Notice that the call to post(name:object:) has been removed.

// The 'object' parameter is only used for filtering out the sender of the notification.
NotificationCenter.default.addObserver(self, selector: #selector(doSomething(notification:)), name: .checkURL, object: nil)

// We can get the notification in the first parameter.
func doSomething(notification: Notification) {
    // Do something, like reading the URL from the web view.
}

Now if you do not have a reference to the webview in your Swift code, you can pass the value in the userInfo parameter when posting the notification, like this:

[[NSNotificationCenter defaultCenter] postNotificationName:@"checkURL" object:self userInfo:@{ @"url" : webView.url }];

Then you can read the userInfo in the notification callback:

func doSomething(notification: Notification) {
    let url = notification.userInfo["url"]
    // Do something with the url.
}