Avoid UIWebView load iTunes App

2019-07-09 23:25发布

How to avoid an UIWebView from opening the iTunes App when loading an iTunes URL?

Example URL: http://itunes.apple.com/nl/app/bluppr-postcards/id348147229?mt=8

Above URL loads the iTunes App directly. UIWebViewDelegate doesn't seem to be able to control this to just load the page in the UIWebView.

It seems it's controlled by the JavaScript function detectAndOpenItunes(); in the body tag of the page.

Any ideas?

1条回答
戒情不戒烟
2楼-- · 2019-07-10 00:06

You're right that the relevant function is detectAndOpenItunes(), which is contained in this file and requires 'iPhone' or 'iPod' to be in the user agent string. I wrote a quick little test app with a web view that just does the following upon viewDidLoad:

[webView loadRequest:
      [NSURLRequest requestWithURL:
        [NSURL URLWithString:
          @"http://itunes.apple.com/nl/app/bluppr-postcards/id348147229?mt=8"]]];

As you say, that opens the App Store app. So I modified it to:

// set some user agent that doesn't have 'iPod' or 'iPhone' in the name
[[NSUserDefaults standardUserDefaults] 
             registerDefaults:[NSDictionary 
                        dictionaryWithObject:@"some old phone or other" 
                        forKey:@"UserAgent"]];

[webView loadRequest:
      [NSURLRequest requestWithURL:
        [NSURL URLWithString:
          @"http://itunes.apple.com/nl/app/bluppr-postcards/id348147229?mt=8"]]];

That displays the page as a web page without opening the app, but there's a problem with the formatting, the displayed page being far too wide. Do a quick search for var deviceDetect= and you'll see that the user agent is also used to determine formatting.

The only solutions I can come up with for that essentially involve screen-scraping-level behaviour. You may subclass NSURLProtocol and add any protocol handler you like via +registerClass:. If you design your protocol to perform HTTP requests then it'll replace the built-in methods for handling HTTP requests. By being selective in which requests you accept or decline, you can do real loading by allowing the HTTP requests you don't want to fall down to the real protocol handler. Hence, you can selectively catch and alter any fetched file before allowing it to be passed to the web view.

You could use that to catch and edit whichever bits of .js and .html you like, but then you're pretty much bound to encounter problems whenever Apple adjust their page.

Similarly, you could use webview's stringByEvaluatingJavaScriptFromString: to execute suitable Javascript to reformat the page after loading, but I'm unable to determine exactly what you'd run and your solution would be just as fragile.

查看更多
登录 后发表回答