Craig Shoemaker
HTML Offline Applications allow you to create applications that work without an Internet connection using technologies native to the Web browser. Pages included in an offline application are listed in the application manifest and thus are served from the application cache whether a connection to the Internet is present or not. Sometimes, however, if an Internet connection is available, you may want to display some data from the server without requiring the user to change pages.Scott Guthrie introduced the concept of “donut hole” caching in ASP.NET, where a cached page may include small windows of content that are updated independently of the cached page. This tutorial demonstrates how to implement an offline page that makes an AJAX call when connected to the Web in order to display live data to the user. When offline, the page simply renders default data in the page.
There are a number of practical reasons to implement an offline application. While most developers first think of the mobile context when considering who might use an offline application, almost any Web site can benefit from having some availability regardless of connection status. A site’s Home and Contact Us pages are excellent candidates for offline availability so users can get some basic information about the organization, even when disconnected.
Building the Application
The example in this tutorial demonstrates how to cache a “Contact Us” page that displays notifications of upcoming events to users. When a user is connected to the Web, live event listings are displayed; otherwise, a telephone number prompts the user to call for event information. This approach keeps the user informed and connected with or without access to the public Web.Figure 1 depicts how the page is rendered for offline viewing, while Figure 2 shows how the page looks when served from the application cache but the computer is connected to the Web.
Figure 1 When the User Is Working Offline, the Application Prompts Him to Call for Event Information
Figure 2 When the User Is Connected to the Internet, the Application Shows Event Information from the Server
The Manifest
The application manifest acts as the master list of resources included in the offline application. Figure 3 shows the full manifest file for this example.Figure 3 Manifest File (manifest.aspx)
CACHE MANIFEST
# version 1
CACHE:
contact.htm
style.css
jquery-1.6.3.min.js
map.png
NETWORK:
events.htm
<%@Page
ContentType="text/cache-manifest"
ResponseEncoding = "utf-8"
Language="C#" AutoEventWireup="true"
CodeBehind="manifest.aspx.cs"
Inherits="ConditionalCaching.manifest" %>
<script language="CS" runat="server">
void Page_Load(object sender, System.EventArgs e)
{
Response.Cache.SetCacheability(HttpCacheability.NoCache);
}
</script>
Next, the CACHE section references the contact page intended to be available to users regardless of connection status. The contact page references a style sheet, jQuery and a map image indicating the physical office location.
Finally, the NETWORK section is needed in this instance to ensure access to the events.htm page (shown in Figure 4). The reason this page isn’t included in the CACHE section is because in the real world the events page would be built dynamically as a server-generated page. Caching a page like this would defeat the purpose of making live event data available to the user when connected to the Web. With the events page listed in the NETWORK section rather than the CACHE section, the Application Cache API won’t attempt to cancel the request to the page. Ultimately, the page’s inclusion in the NETWORK section tells the browser to always attempt to fetch the page from the Web regardless of connection state.
Figure 4 Events Page (events.htm)
<table border="1" cellpadding="2" cellspacing="0">
<tr>
<td>Aug 15</td>
<td><a href="/events/anaheim">Anahiem Convention Center</a></td>
</tr>
<tr>
<td>Sept 5</td>
<td><a href="/events/los-angeles">Los Angeles Convention Center</a></td>
</tr>
<tr>
<td>Oct 3</td>
<td><a href="/events/las-vegas">Las Vegas Convention Center</a></td>
</tr>
<tr>
<td>Nov 14</td>
<td><a href="/events/denver">Colorado Convention Center</a></td>
</tr>
</table>
The Contact Us Page
The contact page HTML, shown in Figure 5, is implemented as you might expect regardless of whether the page is being optimized for offline access. The most important detail to note in the HTML for the page is the contents of the paragraph with the ID of localMessage. This container holds the content that’s displayed when the user is working offline; that content is replaced on-the-fly when a connection to the Internet is available.Figure 5 Contact Page (contact.htm)
<!DOCTYPE html>
<html manifest="manifest.aspx">
<head>
<title></title>
<link rel="Stylesheet" href="style.css" type="text/css" />
<script src="jquery-1.6.3.min.js" type="text/javascript"></script>
<script>
$(function () {
window.applicationCache.onupdateready = function (e) {
applicationCache.swapCache();
window.location.reload();
}
function isOnLine() {
//return false;
return navigator.onLine;
}
function showEventInfo() {
if (isOnLine()) {
$("#localMessage").hide();
$.get("/events.htm", function (data) {
$("#eventList").html(data);
$("#eventList table tr:odd").addClass("alt");
});
}
else {
$("#localMessage").show();
}
}
showEventInfo();
});
</script>
</head>
<body>
<div id="container">
<h1>Awesome Company</h1>
<h2>Contact Us</h2>
<p>
Awesome Company<br />
1800 Main Street<br />
Anytown, CA 90210<br />
(800) 555-5555<br />
<a href="mailto:awesome@company.com">awesome@company.com</a>
</p>
<img src="map.png" />
<div id="events">
<h2>Events</h2>
<p>We are coming to a city near you!</p>
<p id="localMessage">Give us a call at (800) 555-5555
to hear about our upcoming conference dates.</p>
<div id="eventList"></div>
</div>
<div id="version">Version 1</div>
</div>
</body>
</html>
$(function () {
...
});
window.applicationCache.onupdateready = function (e) {
applicationCache.swapCache();
window.location.reload();
}
function isOnLine() {
//return false;
return navigator.onLine;
}
The showEvent function is responsible for implementing the “donut hole caching” functionality for the HTML offline application. The function first detects the connection status of the computer and then either fetches and renders live event data or just shows the event information that already exists in the page.
If the isOnLine function returns true, the local message is hidden from the user and an AJAX call to the events page is initiated. When a response from the asynchronous call is recognized, the resulting HTML is fed to the eventList container and the table is styled with zebra striping.
If, on the other hand, the computer is working offline, the local message is displayed to the user like so:
function showEventInfo() {
if (isOnLine()) {
$("#localMessage").hide();
$.get("/events.htm", function (data) {
$("#eventList").html(data);
$("#eventList table tr:odd").addClass("alt");
});
}
else {
$("#localMessage").show();
}
}
<html manifest="manifest.aspx">
Figure 6 Style Sheet (style.css)
body
{
font-family:Segoe UI, Arial, Helvetica, Sans-Serif;
background-color:#eee;
}
h1
{
text-transform:uppercase;
letter-spacing:-2px;
width:400px;
margin-top:0px;
}
#container
{
position:relative;
margin-left:auto;
margin-right:auto;
padding:20px;
width:700px;
-webkit-box-shadow: 3px 3px 7px #999;
box-shadow: 6px 6px 24px #999;
-moz-border-radius: 10px;
-webkit-border-radius: 10px;
border-radius: 10px;
margin-top:20px;
background-color:#fff;
}
#events
{
position:absolute;
width:210px;
right:20px;
top:255px;
border:1px solid #ccc;
padding:10px;
font-size:.7em;
background-color:#eee;
}
#events h2
{
margin:0px;
}
#version
{
font-size:.7em;
color:#ccc;
}
table
{
border-collapse:collapse;
}
table, tr, td
{
border:1px solid #999;
}
.alt
{
background-color:#ccc;
}
Wrapping Up
Even though pages loaded into the application cache are served from the cache whether the computer is connected to the Internet or not, you can construct your pages to take advantage of online resources when they’re available. “Donut hole caching” works by making an AJAX call to the server when a connection is available and then rendering the results to the user. If the page is in a disconnected state, the application quietly renders data already available on the page—it’s the best of both worlds!Resources
- HTML Offline Web Applications Specification
- Using Application Cache
- Go Offline with Application Cache
- Tip/Trick: Implement “Donut Caching” with the ASP.NET 2.0 Output Cache Substitution Feature
- A Beginner’s Guide to Using the Application Cache
- Working Off the Grid with HTML5 Offline
- HTML5 Rocks: Offline Tutorials