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.
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 inwebView:shouldStartLoadWithRequest:navigationType:
, cache the result, and use the UIWebView'sloadHTMLString: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.
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/
From https://github.com/phonegap/phonegap-iphone/issues/148:
You can now try
ASIWebPageRequest
by All Seeing Interactive: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.
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
4) download them all
5) replace them with embedded Base64-encoded version
6) finally store complete HTML with embedded images as you want.
You should be able to subclass
NSURLCache
and substitute it for the shared cache used by theUIWebView
as described in this Cocoa with Love article: Substituting local data for remote UIWebView requestsFor another approach have a look at Drop-in offline caching for UIWebView (and NSURLProtocol).