HTML5 and jQuery filter images Portfolio


we will be making a beautiful HTML5 portfolio powered by jQuery and the Quicksand plugin. You can use it to showcase your latest work and it is fully customizable, so potentially you could expand it to do much more.

The HTML

The first step is to write down the markup of a new HTML5 document. In the head section, we will include the stylesheet for the page. The jQuery library, the Quicksand plugin and our script.js will go right before the closing body tag:

index.html


<!DOCTYPE html>
<html>
    <head>
        <meta charset="utf-8" />

        <title>Making a Beautiful HTML5 Portfolio | Tutorialzine Demo</title>

        <!-- Our CSS stylesheet file -->
        <link rel="stylesheet" href="assets/css/styles.css" />

  <!-- Enabling HTML5 tags for older IE browsers -->
        <!--[if lt IE 9]>
          <script src="http://html5shiv.googlecode.com/svn/trunk/html5.js"></script>
        <![endif]-->
    </head>

    <body>

        <header>
            <h1>My Portfolio</h1>
        </header>

        <nav id="filter">
   <!-- The menu items will go here (generated by jQuery) -->
  </nav>

        <section id="container">
         <ul id="stage">
    <!-- Your portfolio items go here -->
   </ul>
        </section>

        <footer>
        </footer>

  <!-- Including jQuery, the Quicksand plugin, and our own script.js -->

        <script src="http://ajax.googleapis.com/ajax/libs/jquery/1.6.1/jquery.min.js"></script>
        <script src="assets/js/jquery.quicksand.js"></script>
        <script src="assets/js/script.js"></script>

    </body>
</html>

In the body, there are a number of the new HTML5 elements. The header holds our h1 heading (which is styled as the logo), the sectionelement holds the unordered list with the portfolio items (more lists are added by jQuery, as you will see later), and the nav element, styled as a green bar, acts as a content filter.
The #stage unordered list holds our portfolio items. You can see what these items should look like below. Each of them has a HTML5 dataattribute, which defines a series of comma-separated tags. Later, when we use jQuery to loop through this list, we will record the tags and create categories that can be selected from the green bar.
<li data-tags="Print Design">
 <img src="assets/img/shots/1.jpg" />
</li>

<li data-tags="Logo Design,Print Design">
 <img src="assets/img/shots/2.jpg" />
</li>

<li data-tags="Web Design,Logo Design">
 <img src="assets/img/shots/3.jpg" />
</li>
You can put whatever you want in these li items and customize the portfolio further. The Quicksand plugin will handle the animated transitions of this list, so you are free to experiment.

The jQuery

What the Quicksand plugin does, is compare two unordered lists of items, find the matching LIs inside them, and animate them to their new positions. The jQuery script we will be writing in this section, will loop through the portfolio items in the #stage list, and create a new (hidden) unordered list for each of the tags it finds. It will also create a new menu option, which will trigger the quicksand transition between the two lists.
First we need to listen for the ready event (the earliest point in the loading of the page where we can access the DOM), and loop through all the li items detecting the associated tags.

script.js – Part 1

$(document).ready(function(){

 var items = $('#stage li'),
  itemsByTags = {};

 // Looping though all the li items:

 items.each(function(i){
  var elem = $(this),
   tags = elem.data('tags').split(',');

  // Adding a data-id attribute. Required by the Quicksand plugin:
  elem.attr('data-id',i);

  $.each(tags,function(key,value){

   // Removing extra whitespace:
   value = $.trim(value);

   if(!(value in itemsByTags)){
    // Create an empty array to hold this item:
    itemsByTags[value] = [];
   }

   // Each item is added to one array per tag:
   itemsByTags[value].push(elem);
  });

 });
Each tag is added to the itemsByTags object as an array. This would mean that itemsByTags['Web Design'] would hold an array with all the items that have Web Design as one of their tags. We will use this object to create hidden unordered lists on the page for quicksand.
It would be best to create a helper function that will handle it for us:

script.js – Part 2

function createList(text,items){

 // This is a helper function that takes the
 // text of a menu button and array of li items

 // Creating an empty unordered list:
 var ul = $('<ul>',{'class':'hidden'});

 $.each(items,function(){
  // Creating a copy of each li item
  // and adding it to the list:

  $(this).clone().appendTo(ul);
 });

 ul.appendTo('#container');

 // Creating a menu item. The unordered list is added
 // as a data parameter (available via .data('list')):

 var a = $('<a>',{
  html: text,
  href:'#',
  data: {list:ul}
 }).appendTo('#filter');
}
This function takes the name of the group and an array with LI items as parameters. It then clones these items into a new UL and adds a link in the green bar.
Now we have to loop through all the groups and call the above function, and also listen for clicks on the menu items.

script.js – Part 3

// Creating the "Everything" option in the menu:
createList('Everything',items);

// Looping though the arrays in itemsByTags:
$.each(itemsByTags,function(k,v){
 createList(k,v);
});

$('#filter a').live('click',function(e){
 var link = $(this);

 link.addClass('active').siblings().removeClass('active');

 // Using the Quicksand plugin to animate the li items.
 // It uses data('list') defined by our createList function:

 $('#stage').quicksand(link.data('list').find('li'));
 e.preventDefault();
});

// Selecting the first menu item by default:
$('#filter a:first').click();
Great! Now that we have everything in place we can move on to styling the page.

The CSS

Styling the page itself is not that interesting (you can see the CSS for this in assets/css/styles.css). However what is more interesting is the green bar (or the #filter bar), which uses the :before / :after pseudo elements to add attractive corners on the sides of the bar. As these are positioned absolutely, they also grow together with the bar.

styles.css

#filter {
 background: url("../img/bar.png") repeat-x 0 -94px;
 display: block;
 height: 39px;
 margin: 55px auto;
 position: relative;
 width: 600px;
 text-align:center;

 -moz-box-shadow:0 4px 4px #000;
 -webkit-box-shadow:0 4px 4px #000;
 box-shadow:0 4px 4px #000;
}

#filter:before, #filter:after {
 background: url("../img/bar.png") no-repeat;
 height: 43px;
 position: absolute;
 top: 0;
 width: 78px;
 content: '';

 -moz-box-shadow:0 2px 0 rgba(0,0,0,0.4);
 -webkit-box-shadow:0 2px 0 rgba(0,0,0,0.4);
 box-shadow:0 2px 0 rgba(0,0,0,0.4);
}

#filter:before {
 background-position: 0 -47px;
 left: -78px;
}

#filter:after {
 background-position: 0 0;
 right: -78px;
}

#filter a{
 color: #FFFFFF;
 display: inline-block;
 height: 39px;
 line-height: 37px;
 padding: 0 15px;
 text-shadow:1px 1px 1px #315218;
}

#filter a:hover{
 text-decoration:none;
}

#filter a.active{
 background: url("../img/bar.png") repeat-x 0 -138px;
 box-shadow: 1px 0 0 rgba(255, 255, 255, 0.2),
    -1px 0 0 rgba(255, 255, 255, 0.2),
    1px 0 1px rgba(0,0,0,0.2) inset,
    -1px 0 1px rgba(0,0,0,0.2) inset;
}
With this our beautiful HTML5 portfolio is complete!