page-dynamic.html 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1">
  6. <title>jQuery Mobile Docs - Dynamically Injecting Pages</title>
  7. <link rel="stylesheet" href="../../jquery.mobile-1.0.1.min.css" />
  8. <link rel="stylesheet" href="../_assets/css/jqm-docs.css"/>
  9. <script src="../../jquery.js"></script>
  10. <script src="../../experiments/themeswitcher/jquery.mobile.themeswitcher.js"></script>
  11. <script src="../_assets/js/jqm-docs.js"></script>
  12. <script src="../../jquery.mobile-1.0.1.min.js"></script>
  13. </head>
  14. <body>
  15. <div data-role="page" class="type-interior">
  16. <div data-role="header" data-theme="f">
  17. <h1>Dynamically Injecting Pages</h1>
  18. <a href="../../" data-icon="home" data-iconpos="notext" data-direction="reverse" class="ui-btn-right jqm-home">Home</a>
  19. </div><!-- /header -->
  20. <div data-role="content">
  21. <div class="content-primary">
  22. <h2>jQuery Mobile and Dynamic Page Generation</h2>
  23. <p>jQuery Mobile allows pages to be pulled into the DOM dynamically via its default click hijacking behavior, or through manual calls to <code>$.mobile.changePage()</code>. This is great for applications that generate HTML pages/fragments on the server-side, but there are sometimes cases where an application needs to dynamically generate page content on the client-side from JSON or some other format. This may be necessary for bandwidth/performance reasons, or because it is the data format of choice for the server they are interacting with.</p>
  24. <p>For applications that need to generate page markup on the client-side, it's important to know about the notifications that are triggered during a <code>$.mobile.changePage()</code> call because they can be used as hooks into the navigation system that will allow you to generate your content at the appropriate time.</p>
  25. <p>A call to <code>changePage()</code> will usually trigger the following event notifications:</p>
  26. <ul>
  27. <li><code>pagebeforechange</code>
  28. <ul>
  29. <li>Fired off before any page loading or transition.</li>
  30. <li>NOTE: This event was formerly known as &quot;beforechangepage&quot;.</li>
  31. </ul>
  32. </li>
  33. <li><code>pagechange</code>
  34. <ul>
  35. <li>Fired off after all page loading and transitions.</li>
  36. <li>NOTE: this event was formerly known as &quot;changepage&quot;.</li>
  37. </ul>
  38. </li>
  39. <li><code>pagechangefailed</code>
  40. <ul>
  41. <li>Fired off if an error has occurred while attempting to dynamically load a new page. </li>
  42. </ul>
  43. </li>
  44. </ul>
  45. <p>These notifications are triggered on the parent container element (<code>$.mobile.pageContainer</code>) of pages, and will bubble all the way up to the document element and window.</p>
  46. <p>For applications wishing to inject pages, or radically modify the content of an existing page, based on some non-HTML data, such as JSON or in-memory JS object, the <code>pagebeforechange</code> event is very useful since it gives you a hook for analyzing the URL or page element the application is being asked to load or switch to, and short-circuit the default <code>changePage()</code> behavior by simply calling <code>preventDefault()</code> on the <code>pagebeforechange</code> event.</p>
  47. <p>To illustrate this technique, take a look at this <a href="dynamic-samples/sample-reuse-page.html" rel="external">working sample</a>. In this sample, the main page starts off with a list of categories that the user can navigate into. The actual items in each category are stored in a JavaScript object in memory, for illustrative purposes, but the data can really come from anywhere.</p>
  48. <pre><code>
  49. var categoryData = {
  50. animals: {
  51. name: &quot;Animals&quot;,
  52. description: &quot;All your favorites from aardvarks to zebras.&quot;,
  53. items: [
  54. {
  55. name: &quot;Pets&quot;
  56. },
  57. {
  58. name: &quot;Farm Animals&quot;
  59. },
  60. {
  61. name: &quot;Wild Animals&quot;
  62. }
  63. ]
  64. },
  65. colors: {
  66. name: &quot;Colors&quot;,
  67. description: &quot;Fresh colors from the magic rainbow.&quot;,
  68. items: [
  69. {
  70. name: &quot;Blue&quot;
  71. },
  72. {
  73. name: &quot;Green&quot;
  74. },
  75. {
  76. name: &quot;Orange&quot;
  77. },
  78. {
  79. name: &quot;Purple&quot;
  80. },
  81. {
  82. name: &quot;Red&quot;
  83. },
  84. {
  85. name: &quot;Yellow&quot;
  86. },
  87. {
  88. name: &quot;Violet&quot;
  89. }
  90. ]
  91. },
  92. vehicles: {
  93. name: &quot;Vehicles&quot;,
  94. description: &quot;Everything from cars to planes.&quot;,
  95. items: [
  96. {
  97. name: &quot;Cars&quot;
  98. },
  99. {
  100. name: &quot;Planes&quot;
  101. },
  102. {
  103. name: &quot;Construction&quot;
  104. }
  105. ]
  106. }
  107. };
  108. </code></pre>
  109. <p>The application uses links with urls that contain a hash that tells the application what category items to display:</p>
  110. <pre>
  111. <code>
  112. &lt;h2&gt;Select a Category Below:&lt;/h2&gt;
  113. &lt;ul data-role=&quot;listview&quot; data-inset=&quot;true&quot;&gt;
  114. &lt;li&gt;&lt;a href=&quot;#category-items?category=animals&quot;&gt;Animals&lt;/a&gt;&lt;/li&gt;
  115. &lt;li&gt;&lt;a href=&quot;#category-items?category=colors&quot;&gt;Colors&lt;/a&gt;&lt;/li&gt;
  116. &lt;li&gt;&lt;a href=&quot;#category-items?category=vehicles&quot;&gt;Vehicles&lt;/a&gt;&lt;/li&gt;
  117. &lt;/ul&gt;
  118. </code>
  119. </pre>
  120. <p>Internally, when the user clicks on one of these links, the application intercepts the internal <code>$.mobile.changePage()</code> call that is invoked by the frameworks' default link hijacking behavior. It then analyzes the URL for the page about to be loaded, and then decides whether or not it should handle the loading itself, or to let the normal <code>changePage()</code> code handle things.</p>
  121. <p>The application was able to insert itself into the <code>changePage()</code> flow by binding to the <code>pagebeforechange</code> event at the document level:</p>
  122. <pre>
  123. <code>
  124. // Listen for any attempts to call changePage().
  125. $(document).bind( &quot;pagebeforechange&quot;, function( e, data ) {
  126. // We only want to handle changePage() calls where the caller is
  127. // asking us to load a page by URL.
  128. if ( typeof data.toPage === &quot;string&quot; ) {
  129. // We are being asked to load a page by URL, but we only
  130. // want to handle URLs that request the data for a specific
  131. // category.
  132. var u = $.mobile.path.parseUrl( data.toPage ),
  133. re = /^#category-item/;
  134. if ( u.hash.search(re) !== -1 ) {
  135. // We're being asked to display the items for a specific category.
  136. // Call our internal method that builds the content for the category
  137. // on the fly based on our in-memory category data structure.
  138. showCategory( u, data.options );
  139. // Make sure to tell changePage() we've handled this call so it doesn't
  140. // have to do anything.
  141. e.preventDefault();
  142. }
  143. }
  144. });
  145. </code>
  146. </pre>
  147. <p>So why listen at the document level? In short, because of deep-linking. We need our binding to be active before the jQuery Mobile framework initializes and decides how to process the initial URL that invoked the application.</p>
  148. <p>When the callback for the <code>pagebeforechange</code> binding is invoked, the 2nd argument to the callback will be a data object that contains the arguments that were passed to the initial <code>$.mobile.changePage()</code> call. The properties of this object are as follows:</p>
  149. <ul>
  150. <li><code>toPage</code>
  151. <ul>
  152. <li>Can be either a jQuery collection object containing the page to be transitioned to, <b>OR</b> a URL reference for a page to be loaded/transitioned to.</li>
  153. </ul>
  154. </li>
  155. <li>options
  156. <ul>
  157. <li>Object containing the options that were passed in by the caller of the <code>$.mobile.changePage()</code> function.</li>
  158. <li>A list of the options can be found <a href="../api/methods.html">here</a>.</li>
  159. </ul>
  160. </li>
  161. </ul>
  162. <p>For our sample application, we are only interested in <code>changePage()</code> calls where URLs are initially passed in, so the first thing our callback does is check the type for the <code>toPage</code>. Next, with the help of some URL parsing utilities, it checks to make sure if the URL contains a hash that we are interested in handling ourselves. If so, it then calls an application function called <code>showCategory()</code> which will dynamically create the content for the category specified by the URL hash, and then it calls <code>preventDefault()</code> on the event.</p>
  163. <p>Calling <code>preventDefault()</code> on a <code>pagebeforechange</code> event causes the originating <code>$.mobile.changePage()</code> call to exit without performing any work. Calling the <code>preventDefault()</code> method on the event is the equivalent of telling jQuery Mobile that you have handled the <code>changePage()</code> request yourself.</p>
  164. <p>If <code>preventDefault()</code> is not called, <code>changePage()</code> will continue on processing as it normally does. One thing to point out about the data object that is passed into our callback, is that any changes you make to the <code>toPage</code> property, or options properties, will affect <code>changePage()</code> processing if <code>preventDefault()</code> is not called. So for example, if we wanted to redirect or map a specific URL to another internal/external page, our callback could simply set the <code>data.toPage</code> property in the callback to the URL or DOM element of the page to redirect to. Likewise, we could set, or un-set any option from within our callback, and <code>changePage()</code> would use the new settings.</p>
  165. <p>So now that we know how to intercept <code>changePage()</code> calls, let's take a closer look at how this sample actually generates the markup for a page. Our example actually uses, or we should say, re-uses the same page to display each of the categories. Each time one of our special links is clicked, the function <code>showCategory()</code> gets invoked:</p>
  166. <pre><code>
  167. // Load the data for a specific category, based on
  168. // the URL passed in. Generate markup for the items in the
  169. // category, inject it into an embedded page, and then make
  170. // that page the current active page.
  171. function showCategory( urlObj, options )
  172. {
  173. var categoryName = urlObj.hash.replace( /.*category=/, &quot;&quot; ),
  174. // Get the object that represents the category we
  175. // are interested in. Note, that at this point we could
  176. // instead fire off an ajax request to fetch the data, but
  177. // for the purposes of this sample, it's already in memory.
  178. category = categoryData[ categoryName ],
  179. // The pages we use to display our content are already in
  180. // the DOM. The id of the page we are going to write our
  181. // content into is specified in the hash before the '?'.
  182. pageSelector = urlObj.hash.replace( /\?.*$/, &quot;&quot; );
  183. if ( category ) {
  184. // Get the page we are going to dump our content into.
  185. var $page = $( pageSelector ),
  186. // Get the header for the page.
  187. $header = $page.children( &quot;:jqmData(role=header)&quot; ),
  188. // Get the content area element for the page.
  189. $content = $page.children( &quot;:jqmData(role=content)&quot; ),
  190. // The markup we are going to inject into the content
  191. // area of the page.
  192. markup = &quot;&lt;p&gt;&quot; + category.description + &quot;&lt;/p&gt;&lt;ul data-role='listview' data-inset='true'&gt;&quot;,
  193. // The array of items for this category.
  194. cItems = category.items,
  195. // The number of items in the category.
  196. numItems = cItems.length;
  197. // Generate a list item for each item in the category
  198. // and add it to our markup.
  199. for ( var i = 0; i &lt; numItems; i++ ) {
  200. markup += &quot;&lt;li&gt;&quot; + cItems[i].name + &quot;&lt;/li&gt;&quot;;
  201. }
  202. markup += &quot;&lt;/ul&gt;&quot;;
  203. // Find the h1 element in our header and inject the name of
  204. // the category into it.
  205. $header.find( &quot;h1&quot; ).html( category.name );
  206. // Inject the category items markup into the content element.
  207. $content.html( markup );
  208. // Pages are lazily enhanced. We call page() on the page
  209. // element to make sure it is always enhanced before we
  210. // attempt to enhance the listview markup we just injected.
  211. // Subsequent calls to page() are ignored since a page/widget
  212. // can only be enhanced once.
  213. $page.page();
  214. // Enhance the listview we just injected.
  215. $content.find( &quot;:jqmData(role=listview)&quot; ).listview();
  216. // We don't want the data-url of the page we just modified
  217. // to be the url that shows up in the browser's location field,
  218. // so set the dataUrl option to the URL for the category
  219. // we just loaded.
  220. options.dataUrl = urlObj.href;
  221. // Now call changePage() and tell it to switch to
  222. // the page we just modified.
  223. $.mobile.changePage( $page, options );
  224. }
  225. }
  226. </code></pre>
  227. <p>In our sample app, the hash of the URL we handle contains 2 parts:</p>
  228. <pre><code>
  229. #category-items?category=vehicles
  230. </code></pre>
  231. <p>The first part, before the '?' is actually the id of the page to write content into, the part after the '?' is info the app uses to figure out what data it should use when generating the markup for the page. The first thing <code>showCategory()</code> does is deconstruct this hash to extract out the id of the page to write content into, and the name of the category it should use to get the correct set of data from our in-memory JavaScript category object. After it figures out what category data to use, it then generates the markup for the category, and then injects it into the header and content area of the page, wiping out any other markup that previously existed in those elements.</p>
  232. <p>After it injects the markup, it then calls the appropriate jQuery Mobile widget calls to enhance the list markup it just injected. This is what turns the normal list markup into a fully styled listview with all its behaviors.</p>
  233. <p>Once that's done, it then calls <code>$.mobile.changePage()</code>, passing it the DOM element of the page we just modified, to tell the framework that it wants to show that page.</p>
  234. <p>Now an interesting problem here is that jQuery Mobile typically updates the browser's location hash with the URL associated with the page it is showing. Because we are re-using the same page for each category, this wouldn't be ideal, because the URL for that page has no specific category info associated with it. To get around this problem, <code>showCategory()</code> simply sets the <code>dataUrl</code> property on the options object it passes into <code>changePage()</code> to tell it to display our original URL instead.</p>
  235. <p>That's the sample in a nutshell. It should be noted that this particular sample and its usage is not a very good example of an app that degrades gracefully when JavaScript is turned off. That means it probably won't work very well on C-Grade browsers. We will be posting other examples that demonstrate how to degrade gracefully in the future. Check this <a href="http://jquerymobile.com/test/docs/pages/dynamic-samples/">page</a> for updates.</p>
  236. </div>
  237. <!--/content-primary -->
  238. <div class="content-secondary">
  239. <div data-role="collapsible" data-collapsed="true" data-theme="b" data-content-theme="d">
  240. <h3>More in this section</h3>
  241. <ul data-role="listview" data-theme="c" data-dividertheme="d">
  242. <li data-role="list-divider">Pages &amp; Dialogs</li>
  243. <li><a href="page-anatomy.html">Anatomy of a page</a></li>
  244. <li><a href="page-template.html" data-ajax="false">Single page template</a></li>
  245. <li><a href="multipage-template.html" data-ajax="false">Multi-page template</a></li>
  246. <li><a href="page-titles.html">Page titles</a></li>
  247. <li><a href="page-links.html">Linking pages</a></li>
  248. <li><a href="page-transitions.html" data-ajax="false">Page transitions</a></li>
  249. <li><a href="page-dialogs.html">Dialogs</a></li>
  250. <li><a href="page-cache.html">Prefetching &amp; caching pages</a></li>
  251. <li><a href="page-navmodel.html">Ajax, hashes &amp; history</a></li>
  252. <li data-theme="a"><a href="page-dynamic.html">Dynamically injecting pages</a></li>
  253. <li><a href="page-scripting.html">Scripting pages</a></li>
  254. <li><a href="phonegap.html">PhoneGap apps</a></li>
  255. <li><a href="touchoverflow.html">touchOverflow feature</a></li>
  256. <li><a href="pages-themes.html">Theming pages</a></li>
  257. </ul>
  258. </div>
  259. </div>
  260. </div><!-- /content -->
  261. <div data-role="footer" class="footer-docs" data-theme="c">
  262. <p>&copy; 2011-2012 The jQuery Project</p>
  263. </div>
  264. </div><!-- /page -->
  265. </body>
  266. </html>