Friday, 5 April 2013

jQuery: Slideshow pagination Part1


Paginating content is a standard choice when dealing with large chunks of data. The implementation usually involves passing the page number to the back-end, where the appropriate data is fetched from the database and returned in some form. A cumbersome process, but it is a necessary evil. Or is it?
When dealing with small data sets, wouldn’t it be better to have the content readily available, but still neatly organized and easy to access?
Today we are making a jQuery plugin that will enable you to convert a regular unordered list of items into a SEO friendly set of easily navigatable pages. It can be used for comment threads, slideshows, or any kind of structured content.

The Idea

When called, the jQuery plugin splits the LI elements contained in the unordered list into a configurable number of groups. These groups (or pages) are floated to the left and hidden from view, as they overflow the UL which is given overflow:hidden. A number of control links are generated, which slide the appropriate page of LIs into view.
You can also take a look at the illustration below.
How it works
How it works

Step 1 – XHTML

The first step of the tutorial is to set up the XHTML markup. The plugin only needs an unordered list, UL, with some li elements inside it. Here is the code from demo.html, which you can find in the download archive:

demo.html

1<div id="main">
2    <ul id="holder">
3        <li>Lorem ipsum dolor sit amet...</li>
4        <li>Lorem ipsum dolor sit amet...</li>
5        <li>Lorem ipsum dolor sit amet...</li>
6        <li>Lorem ipsum dolor sit amet...</li>
7    </ul>
8</div>
The main div acts as a container for the paginated UL, and is styled with a nice light-gray background. The unordered list holds the list elements (hence the id).
In most practical situations, the markup above would probably be generated by a back-end script, freeing you from having to do it manually. You could have all sorts of content inside those LIs, as the height and size is dynamically calculated by jQuery (just a reminder – if you plan on using images, specify the width and the height).
A Pure jQuery & CSS Pagination
A Pure jQuery & CSS Pagination

Step 2 – CSS

After creating the XHTML markup, we can move on to styling it. If is a good idea to style your pages as if there were no navigation, as the plug-in is JavaScript dependent. This means that it is possible that some users will not be able to see nor use the pagination.

styles.css – Part 1

01#main{
02    /* The main container div */
03    position:relative;
04    margin:50px auto;
05    width:410px;
06    background:url('img/main_bg.jpg'repeat-x #aeadad;
07    border:1px solid #CCCCCC;
08    padding:70px 25px 60px;
09 
10    /* CSS3 rounded cornenrs */
11 
12    -moz-border-radius:12px;
13    -webkit-border-radius:12px;
14    border-radius:12px;
15}
16 
17#holder{
18    /* The unordered list that is to be split into pages */
19 
20    width:400px;
21    overflow:hidden;
22    position:relative;
23    background:url('img/dark_bg.jpg'repeat #4e5355;
24    padding-bottom:10px;
25 
26    /*  CSS3 inner shadow (the webkit one is commeted, because Google Chrome
27        does not like rounded corners combined with inset shadows): */
28 
29    -moz-box-shadow:0 0 10px #222 inset;
30    /*-webkit-box-shadow:0 0 10px #222 inset;*/
31    box-shadow:0 0 10px #222 inset;
32}
33 
34.swControls{
35    position:absolute;
36    margin-top:10px;
37}
First we style the main div and the unordered list (the latter is assigned the id of holder).
Notice how we use the CSS3 box shadow property with the inset attribute, to mimic an inner shadow. As with most CSS3 rules, we still have to provide vendor-specific prefixes for Mozilla (Firefox) and Webkit browsers (Safri and Chrome).
You can see that the webkit version of the property is commented out. This is because there is a bug in the rendering of box shadows in Chrome, when combined with the border-radius property (the shadows are rendered as if the div is square, ignoring the rounded corners and thus ruining the effect).

styles.css – Part 2

01a.swShowPage{
02 
03    /* The links that initiate the page slide */
04 
05    background-color:#444444;
06    float:left;
07    height:15px;
08    margin:4px 3px;
09    text-indent:-9999px;
10    width:15px;
11    /*border:1px solid #ccc;*/
12 
13    /* CSS3 rounded corners */
14 
15    -moz-border-radius:7px;
16    -webkit-border-radius:7px;
17    border-radius:7px;
18}
19 
20a.swShowPage:hover,
21a.swShowPage.active{
22    background-color:#2993dd;
23 
24    /*  CSS3 inner shadow */
25 
26    -moz-box-shadow:0 0 7px #1e435d inset;
27    /*-webkit-box-shadow:0 0 7px #1e435d inset;*/
28    box-shadow:0 0 7px #1e435d inset;
29}
30 
31#holder li{
32    background-color:#F4F4F4;
33    list-style:none outside none;
34    margin:10px 10px 0;
35    padding:20px;
36    float:left;
37 
38    /* Regular CSS3 box shadows (not inset): */
39 
40    -moz-box-shadow:0 0 6px #111111;
41    -webkit-box-shadow:0 0 6px #111111;
42    box-shadow:0 0 6px #111111;
43}
44 
45#holder,
46#holder li{
47    /* Applying rouded corners to both the holder and the holder lis */
48    -moz-border-radius:8px;
49    -webkit-border-radius:8px;
50    border-radius:8px;
51}
52 
53.clear{
54    /* This class clears the floated elements */
55    clear:both;
56}
In the second part of the code, we style the page control links and the li elements. As you can see on line 46, we are applying rounded corners to both the unordered list and the li elements in one declaration, which saves us a from duplicating code.
Lastly is the clear class, which is used to clear the floats of the elements, also known as the clearfix technique.
The Pagination Explained
The Pagination Explained

Step 3 – jQuery

Moving to the last part of the tutorial, we need to include the latest version of the jQuery library in the page. Performance-wise, it is best to include all external JavaScript files just before the closing body tag, as scripts block he rendering of the page.

script.js – Part 1

01(function($){
02 
03// Creating the sweetPages jQuery plugin:
04$.fn.sweetPages = function(opts){
05 
06    // If no options were passed, create an empty opts object
07    if(!opts) opts = {};
08 
09    var resultsPerPage = opts.perPage || 3;
10 
11    // The plugin works best for unordered lists,
12    // although OLs would do just as well:
13    var ul = this;
14    var li = ul.find('li');
15 
16    li.each(function(){
17        // Calculating the height of each li element,
18        // and storing it with the data method:
19 
20        var el = $(this);
21        el.data('height',el.outerHeight(true));
22    });
23 
24    // Calculating the total number of pages:
25    var pagesNumber = Math.ceil(li.length/resultsPerPage);
26 
27    // If the pages are less than two, do nothing:
28    if(pagesNumber<2) return this;
29 
30    // Creating the controls div:
31    var swControls = $('<div class="swControls">');
32 
33    for(var i=0;i<pagesNumber;i++)
34    {
35        // Slice a portion of the li elements, and wrap it in a swPage div:
36        li.slice(i*resultsPerPage,(i+1)*resultsPerPage).wrapAll('<div class="swPage" />');
37 
38        // Adding a link to the swControls div:
39        swControls.append('<a href="" class="swShowPage">'+(i+1)+'</a>');
40    }
41 
42    ul.append(swControls);
Creating a jQuery plug-in is not as hard as you might think. We just need to create a new function as a property of jQuery.fn (or $.fn, as given here). The this of the function points to the original jQuery object that it was called on.
Moving from there, we check for the existence of the opts object and set resultsPerPage accordingly. This is the number of li elements that are going to be grouped as a page.
After this, we calculate the total number of pages with the Math.ceil() function. It rounds the result to the nearest greater integer, which gives the correct number of pages.
Now that we have the number of pages obtained, we can enter a for loop in which we split the li elements into portions and wrap them in a swPage div, forming a page. Keep in mind that calling the jQueryslice() method on line 36 creates a new set of elements and leaves the original set intact (thus in every iteration of the for loop we start with the original set of li elements).


No comments:

Post a Comment