11
votes

I have a simple app loading a site optimized for the iPhone in a UIWebView.

Problem is, caching does not seem to work:

[webView loadRequest: [NSURLRequest requestWithURL: [NSURL URLWithString: url]
                                       cachePolicy: NSURLRequestUseProtocolCachePolicy
                                   timeoutInterval: 60.0]];

Any things referenced in this remote page (css, images, external javascript files) never get cached (the requests never send a If-Modified-Since header or anything else in the way of cache control.)

Is it possible? It seems with a regular Cocoa WebView there a delegate methods that get called for each resource request and post load (-didFinishLoadingFromDataSource:) which you could use to roll your own caching.. but that does not seem applicable here.

My entire page (page and its referenced resources) is around 89K compressed.. which is slow over 3G in some spots and even worse over EDGE. Incoming requests are at least indicating that it accepts compression (accept-encoding=gzip, deflate), so that's good I suppose.

I read this yui study, which seems to indicate that the iPhone will cache 25k per item. The only thing referenced that is over 25k uncompressed is jquery (packed but uncompressed - it is 30k). Everything else should be cacheable. No request for anything referenced in the page fetched is triggering a 304 on the server side.

That yui study was from almost a year ago, and I am guessing with mobile safari only.

This is using a UIWebView in a native iPhone app.

6

6 Answers

7
votes

One workaround of this problem as I see is to

1) download HTML code

2) store it in the string

3) find all external links in it like

<img src="img.gif" width="..." height="..." />

4) download them all

5) replace them with embedded Base64-encoded version

<img src="data:image/gif;base64,R0lGODlhUAAPA...JADs= " width="..." height="..." />

6) finally store complete HTML with embedded images as you want.

5
votes

From https://github.com/phonegap/phonegap-iphone/issues/148:

NSURLCache* cache = [NSURLCache sharedURLCache];
[cache setMemoryCapacity:4 * 1024 * 1024];
[cache setDiskCapacity:512*1024];

[NSURLRequest requestWithURL:appURL
                 cachePolicy:NSURLRequestReturnCacheDataElseLoad
             timeoutInterval:10.0];
4
votes

You can now try ASIWebPageRequest by All Seeing Interactive:

ASIWebPageRequest is a new experimental addition to the ASIHTTPRequest family. It can be used to download a complete webpage, including external resources like images and stylesheets, in a single request. Once a web page is downloaded, the request will parse the content, look for external resources, download them, and insert them directly into the html source using Data URIS. You can then take the response and put it directly into a UIWebView / WebView on Mac.

I can only advise everyone to use Ben Copsey's great library for all sorts of HTTP operations anyways.

UPDATE: Ben has discontinued ASIHTTPRequest. I no longer suggest using it.

2
votes

You can always perform the requests manually, though that'll be tricky - and then you can cache things to your heart's content. Build a UIWebViewDelegate that starts the request in webView:shouldStartLoadWithRequest:navigationType:, cache the result, and use the UIWebView's loadHTMLString:baseURL: to update the view.

It'll be ugly, and things won't work as smoothly as you might want, but it may be good enough for what you need.

1
votes

The ihone has limited caching capacity compared to a normal computer. It limits uncompressed cache items to 25k.

Good info here: http://yuiblog.com/blog/2008/02/06/iphone-cacheability/

0
votes

You should be able to subclass NSURLCache and substitute it for the shared cache used by the UIWebView as described in this Cocoa with Love article: Substituting local data for remote UIWebView requests

For another approach have a look at Drop-in offline caching for UIWebView (and NSURLProtocol).