Offline iOS web app: loads my manifest, but doesn't work offline
Asked Answered
P

13

74

I'm writing a web app to be used offline on iOS. I've created a manifest, am serving it up as text/cache-manifest, and it usually works fine, when running inside Safari.

If I add it as an app to my home screen, then turn on Airplane mode, it can't open the app at all -- I get an error and it offers to close the app. (I thought this was the entire purpose of an offline app!)

  • When I load the app a first time when online, I can see in my logs that it's requesting every page listed in the manifest.

  • If I turn off Airplane mode, and load the app, I can see the first file it's requesting is my main.html file (which is both listed in the manifest, and has the manifest=... attribute). It then requests the manifest, and all my other files, getting 200's for all (and 304's for anything requested a second time during this load).

  • When I load the page in Chrome, and click around, the logs show the only thing it's trying to reach on the server is "/favicon.ico" (which is a 404, and which I don't think iOS Safari tries to load, anyway). All of the files listed in the manifest are valid and served without error.

  • The Chrome inspector lists, under "APPLICATION CACHE", all the cached files I've listed which I expect. The entire set of files is about 50 KB, way under any limit on offline resources that I've found.

Is this supposed to work, i.e., am I supposed to be able to create an offline iOS app using only HTML/CSS/JS? And where do I go about figuring out why it's failing to work offline?

(Related but doesn't sound quite the same to me, since it's about Safari and not a standalone app: "Can't get a web app to work offline on iPod")

Part answered 5/12, 2010 at 23:22 Comment(0)
B
22

I confirm that name 'cache.manifest' solved the offline caching problem in IOS 4.3. Other name simply did not work.

Busyness answered 27/7, 2011 at 19:32 Comment(1)
+1 this worked for me - even though I could see that the iPad was fetching and using my previous manifest called offline.manifestArguelles
N
7

I found debugging HTML5 offline apps to be a pain. I found the code from this article helped me figure out what was wrong with my app:

http://jonathanstark.com/blog/2009/09/27/debugging-html-5-offline-application-cache/

Debugging HTML 5 Offline Application Cache by Jonathan Stark

If you are looking to provide offline access to your web app, the Offline Application Cache available in HTML5 is killer. However, it’s a giant PITA to debug, especially if you’re still trying to get your head around it.

If you are struggling with the cache manifest, add the following JavaScript to your main HTML page and view the output in the console using Firebug in Firefox or Debug > Show Error Console in Safari.

If you have any questions, PLMK in the comments.

HTH,
j

var cacheStatusValues = [];
cacheStatusValues[0] = 'uncached';
cacheStatusValues[1] = 'idle';
cacheStatusValues[2] = 'checking';
cacheStatusValues[3] = 'downloading';
cacheStatusValues[4] = 'updateready';
cacheStatusValues[5] = 'obsolete';

var cache = window.applicationCache;
cache.addEventListener('cached', logEvent, false);
cache.addEventListener('checking', logEvent, false);
cache.addEventListener('downloading', logEvent, false);
cache.addEventListener('error', logEvent, false);
cache.addEventListener('noupdate', logEvent, false);
cache.addEventListener('obsolete', logEvent, false);
cache.addEventListener('progress', logEvent, false);
cache.addEventListener('updateready', logEvent, false);

function logEvent(e) {
    var online, status, type, message;
    online = (navigator.onLine) ? 'yes' : 'no';
    status = cacheStatusValues[cache.status];
    type = e.type;
    message = 'online: ' + online;
    message+= ', event: ' + type;
    message+= ', status: ' + status;
    if (type == 'error' && navigator.onLine) {
        message+= ' (prolly a syntax error in manifest)';
    }
    console.log(message);
}

window.applicationCache.addEventListener(
    'updateready',
    function(){
        window.applicationCache.swapCache();
        console.log('swap cache has been called');
    },
    false
);

setInterval(function(){cache.update()}, 10000);
Node answered 5/12, 2010 at 23:39 Comment(3)
This is a neat little bit of JS, but so far it hasn't helped me learn anything new. All the downloading/checking/idle in the log looks exactly as I'd expect.Part
@Part - sorry to hear that. it definitely helped me narrow down my errors (although it was still a pain to figure the exact cause of it). It also gave me some good feedback as to whether cache loaded/reloaded successfully (before knowing this, I would test offline after each and every change - after knowing this, I knew I needed to get the updateready event successfully before it was even worth trying it offline).Node
Hmm. Do I need to worry about updateready and swapCache() and such? I was under the impression that, since I'm making an app which will live entirely offline (no server interaction at all, after the initial load), I could just put my filenames in the cache manifest, and that's it. Is it more complex than that in my case?Part
F
5

Sometimes an application cache group gets into a bad state in MobileSafari — it downloads every item in the cache and then fires a generic cache error event at the end. An application cache group, as per the spec, is based on the absolute URL of the manifest. I've found that when this error occurs, changing the path to the manifest (eg, cache2.manifest, etc) gives you a fresh cache group and circumvents the problem. I can vouch that all of our web apps work offline in full-screen with 4.2 and 4.3.

Frawley answered 15/3, 2011 at 21:17 Comment(0)
P
3

No offline web app (as of iOS 4.2) can run without an internet connection (which means Airplane mode, too) when using <meta name="apple-mobile-web-app-capable" content="yes" /> in the html head section. I have verified this with every example I've seen and the ones that use Safari to render the site work fine, but when you throw in that meta tag, it won't work. Try your app without it and you'll see what I mean.

Passepartout answered 17/1, 2011 at 20:1 Comment(7)
@pattern86 Interestingly I have a site here (still developing it, so can't share I'm afraid) which appears to work fine using <meta name="apple-mobile-web-app-capable" content="yes" /> . I can close the browser and the offline web app, switch on airplane mode, and still load the offline web app without an issue. Is there any other behaviour required to trigger this bug? I want to make sure we don't run into it in the wild...Pipkin
Curiously PieGuy works for me too. I'm on 4.2 (8C134) which is the 4.2GM. I'll try an update to 4.2.1 and see what happens :(Pipkin
@Pipkin I am referring to offline web apps as ones that are added to the home screen. If you bookmark them in Safari and use them there, it should still work fine. Haven't tested this in iOS 4.3 yet.Passepartout
@pattern86 Aye, I also meant homescreen web apps. I've had a chance to test my site with 4.2.1 and 4.3 - in both cases, it's worked fine. So there seems to be some other effect at play here (the manifest is so sensitive to many things), but the manifest as a whole does appear to be able to function as intended.Pipkin
@Pipkin I'm glad you got it to work! I'll give this a try sometime this weekend.Passepartout
+1 I have just tested this on a home-screen app on IOS 5.1.1. The main image was failing to cache, but removing this tag solved the problem.Lash
Removing that line fixed the problem for me too, but I'd really like to have both features...Kippar
F
3

I have found that clearing the Safari cache after enabling Airplane mode to be an effective way of testing whether the app is really functioning offline.

I have sometimes been fooled into thinking that the application cache was working when it wasn't.

Fairleigh answered 15/3, 2011 at 21:21 Comment(0)
L
2

I had struggled with this iOS 4.3 "no offline cache" problem since I updated my iPad to 4.3.1 from 4.2. I saw in another post in this site that it was working again in 4.3.2. So I updated by iPad again, now to iOS 4.3.3. But still couldn't get the offline caching to work until I renamed my manifest file to "cache.manifest". Then the caching started working again and I could run my HTML5 offline app from the Home Screen. I did not need to put the favicon.ico in to the cache manifest. And I also had full screen going (setting the "apple-mobile-web-app-capable" to "yes").

Lixivium answered 10/5, 2011 at 8:29 Comment(0)
M
1

I have several working offline and on/offline web apps.

When I turn off airport mode, I get a request for the manifest and some other files.

I don't get requests for images, JavaScript, CSS or cached AJAX files.

If you see requests for your resources, IOS doesn't have them cached.

Safari in general is more picky with manifests.

I suggest you try Safari on your computer.

Messidor answered 5/1, 2011 at 19:13 Comment(2)
Nope, as noted above (Safari is the same as Chrome, here, AFAICT) it never tries to request any resource (except /favicon.ico) from a desktop browser. The only time it does this is from iOS Safari, when in Airplane Mode (or when otherwise offline, presumably).Part
I'd like to see the page. I run chrome/safari/firefox on my mac at home and iOS on iPhone and iPad.Messidor
I
1

I have one potential workaround for this - it seems a bit crazy, but here goes... I work with the cache.manifest and full screen apps a lot (here's a test if you need: http://www.mrspeaker.net/2010/07/12/argy-bargy/ - add to home screen then turn on flight mode and it launches - at least, as of iOS 4.2.1)

One weird thing I found is that sometimes it seems that some kind of "meta" information in files can mess them up from being cached - Have you ever noticed that in bash that if you do a "ls" some files (depending on your colour settings) are highlighted for no apparent reason? Files can have meta data that the operating system (I think) adds automagically - and there are ways to remove it... I can't remember why but here's some more details: Strip metadata from files in Snow Leopard

After tearing my hair out one day - and refusing to give up because I knew it SHOULD have worked... Chrome was saying it loaded all the files, but ended with a generic error. In the end I recreated the project structure with blank files and copy/pasted the contents over. It worked - started caching as it was supposed to!

When I looked at the files I noticed there was some meta info. I tried scrubbing this info and the original project worked again. I'm not sure this was the reason it worked again - perhaps it was just a coincidence.

Because it worked, I didn't think too much about it. The same problem happened again some months later and the copy/paste trick worked again. I was busy, so I didn't investigate further - but vowed I would get to the bottom of it the next time it happened.... but I haven't had to yet.

Phew. Anyway, glad I got to write that down somewhere...

[UPDATE: months and months later - I've not been able to reproduce this, so I don't think it's the metadata]

Intention answered 15/3, 2011 at 19:43 Comment(0)
N
1

I have run into the same problem today on iOS 4.3. I was able to fix the problem by adding a favicon.ico file and also adding it to the manifest.

Nidorf answered 16/3, 2011 at 11:32 Comment(0)
D
1

I've written an offline app that still seems to work in 4.2 and 4.2.1; the post is a little dusty, but the code still runs:

http://kentbrewster.com/backchannel/

Remy Sharp has a newer post with code that also works, here:

http://remysharp.com/2011/01/31/simple-offline-application/

His app:

http://rem.im/boggle/

Dalesman answered 16/3, 2011 at 13:42 Comment(0)
D
1

After days of messing with getting offline web apps to work on an iPhone/iPod Touch using the Webserver's HTTP authentication, I discovered these useful nuggets:

  1. Make sure Safari is at the URL root of the web app when tapping "Add to Home Screen". I used jQuery Mobile and was sometimes adding the link with"/#pageId". Caused trouble.

  2. Run your Ajax calls in serial. This might only be important if your web app is using HTTP authentication, but my app was firing a whole slew of Ajax calls on page load in parallel and it caused the app to hang on the "apple-touch-startup-image".

  3. Ajax calls are "successful" when offline (at least using Prototype.js). Test for an actual piece of data in the Ajax response, not just on the HTTP status. I used this to test for displaying cached (SQL) or live data.

  4. In the manifest use "NETWORK:\n*\n". From what I could muster, this is a catch-all statement for anything not explicit in the "CACHE:" section. Use Chrome to make sure your manifest is correct. Look at Chrome's console for errors.

  5. Not directly related, but tripped me up for a bit, openDatabase.transaction() calls are ASYNCHRONOUS! Meaning, the line of JS code after transaction (execute(), error(), success()) will execute BEFORE the success() function.

Good luck!

Decimate answered 31/3, 2011 at 23:2 Comment(0)
M
1

I found this solution that seemed to work for me, since I also ran into this problem during my development. This fix has worked fine for me so far and also for other people that I've asked to test it with, and I'm able to get it running offline (in airplane mode) and off the home screen after caching and whatnot. I've written a post about it on my site:

http://www.offlinewebapp.com/solved-apple-mobile-web-app-capable-manifest-error/

  1. Delete your current web app icon on the home screen.
  2. Go to settings and clear your Safari browser cache.
  3. Double tap your home button to open the multitasking bar. Find the Safari one, hold your finger down on it, and exit it.

Please let me know if this works for you also! Good luck!

Matthewmatthews answered 27/7, 2011 at 23:21 Comment(0)
A
1

I've written an app and it works fine through the mobile browser, but when adding the desktop... Doesnt work. I guess apple have given up on IOS4 and all efforts are now on OS5. Shame :(

Ahmednagar answered 9/9, 2011 at 14:47 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.