Skip to content

davidmarkclements/Respondu

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

49 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Respondu 0.0.5

A cross browser platform for implementing and creating gracefully degrading Responsive techniques

##Features

  • Simple straightforward syntax
  • Supports all modern browsers and IE7+
  • Entirely client side solution, no unneccessary HTTP requests
  • No Double Downloads - Defers loading of src's until code has been parsed
  • Write code exactly as specified by the proposed standards (e.g. use actual picture tags, instead of divs and data attributes)
  • Gracefully Degrades for non-JS clients (e.g. search bots)
  • Easy plugin system that allows you to write custom imlpementations (e.g. roll your own responsive video code)
  • Forward looking - simple feature detection renders seamless compatible with future browsers
  • Custom builds with grunt allows you to take only what you need

###Included Implementations

##Examples

Examples can be found at http://respondu.davidmarkclements.com

##Basic Use

<!DOCTYPE HTML>
<html>
<head>
<script src=R.js></script>
</head>
<body>

<script>Respondu();</script>
<noscript>

<img src="/images/responsive.jpg">

</noscript-->
</body>
</html>

The end dashes in </noscript--> are essential for Respondu to operate in certain browsers (Safari, IE8) and to block the img src's from loading prior to responsive processing.

The positioning of the Respondu call in the body, just before the noscript tag is also essential. See How It Works.

Basic usage will apply the default breakpoints

breakpoints: {
        typical: 500, //typical will leave src unchanged
        medium : 1000, //configurable properties, will be reflected in requested image e.g. name.medium.png
        large : Infinity //largest size set to infinity
      }

##Setting Breakpoints

Break points can be set thusly

<script>Respondu({breakpoints: {
  typical:300, 
  small: 500, 
  medium: 1200, 
  large: 1500,
  x-large: Infinity}});</script>

Using Implementations

Respondu also provides an implementations system, it currently includes the picture and srcset implementations as proposed on w3.org

picture

For picture we would do something like

<!DOCTYPE HTML>
<html>
<head>
<link href='css/style.css' rel='stylesheet' type='text/css'>
<script src=R.js></script>
</head>
<body>

<script>Respondu('picture');</script>
<noscript>
<picture id=thepic> 
    <source srcset="images/photo.small.jpg 1x, images/photo.small.jpg 2x">
    <source media="(min-width:320px)" srcset="images/photo.medium.jpg 1x, images/photo.medium@2x.jpg 2x">
    <source media="(min-width:640px)" srcset="images/photo.large.jpg 1x, images/photo.large@2x.jpg 2x">
    <source media="(min-width:1280px)" srcset="images/photo.xlarge.jpg 1x, images/photo.xlarge@2x.jpg 2x">
    <img src="images/photo.jpg" alt="Alt tag describing the image represented">
</picture>
</noscript-->
</body>
</html>

srcset

For srcset we could do

<!DOCTYPE HTML>
<html>
<head>
<link href='css/style.css' rel='stylesheet' type='text/css'>
<script src=js/R.js></script>
</head>
<body>

<script>Respondu('srcset');</script>
<noscript>

<img src="images/photo.jpg" srcset="images/photo.small.jpg 320w, images/photo.small@2x.jpg 320w 2x, images/photo.medium.jpg 640w, images/photo.medium@2x.jpg 640w 2x, images/photo.large.jpg 1280w, images/photo.large@2x.jpg 1280w 2x, images/photo.xlarge.jpg 20000w, images/photo.xlarge@2x.jpg 20000w x2">

</noscript-->

</body>
</html>

Creating an Implementation

We create an implementation by adding it to the Respondu prototype, the format for creating an implementation is:

  Respondu.plugin('MyNewImplementation', function (doc, done) {
    // all your code here
    done();
  });

The doc parameter is a DOM object (but not the actual DOM) The done paremeter is called as a function when processing is complete (using the done callback function allows for any async stuff)

We could then use it with

<script>Respondu('MyNewImplementation');</script>

Here's how we could implement picture (already included):

Respondu.plugin('picture', function (doc, done) {
  if ('srcset' in document.createElement('picture')) { //if picture is implemented then just pass through
    if (done) done();
    return; 
  }

  var pictures = (doc.getElementsByTagName('picture')), pic, attrs, sources, src, i, c,
    media, sourceImg, img, sW = window.screen.width,  pixelRatio, 
    pr = window.devicePixelRatio || 1;//set devices pixel ratio;
        
  for(i = 0; i < pictures.length; i++) {
    pic = pictures[i];

    attrs = pic.attributes;
    sources = pic.getElementsByTagName('source');
    
    sourceCandidates = [];
    
    for(c = 0; c < sources.length; c++) {
      srcel = sources[c];
      media = srcel.getAttribute('media'); 
      //!media means no media attribute, so will be used if no other source elements qualify                        
      if (!media || matchMedia(media).matches) { sourceCandidates.push(srcel); }                                                                                          
    }
    
    var x = sourceCandidates.length, closest = 0;
    
    if (x > 1) {
      while(x--) {
        media = sourceCandidates[x].getAttribute('media');
        if (!media && sourceCandidates.length > 1) { 
          delete sourceCandidates[x];
          break;
        }
        
        media = media.match(/\((min|max)-width:([0-9]+)(px|em)\)/);
        
        
        if (Math.abs(media[2] - sW) <= Math.abs(closest - sW)) {
          closest = media[2];
          sourceImg = sourceCandidates[x];
        }
      
      }
    } else {
      sourceImg = sourceCandidates[0];
    }
   
    var srcset, srcsetCounter, srcsetToken, imageSrc;
    if (sourceImg) {
      srcset = sourceImg.getAttribute('srcset').split(',');
      
      srcsetCounter = srcset.length;
      
      while (srcsetCounter--) {
        srcsetToken = srcset[srcsetCounter].replace(/^\s\s*/, '').replace(/\s\s*$/, '').split(' ');
        
        if (srcsetToken[1] === pr + 'x') {
          imageSrc = srcsetToken[0];
        }
        
        if (!srcsetToken[1]) {
          //no density supplied, assume default
          imageSrc = srcsetToken[0];            
        }
      
      
      }
    
    
      img = doc.createElement('img'); //create a new image element on the ghost DOM

      img.setAttribute('src', imageSrc);

              
      for(c = 0; c < attrs.length; c++) {
       img.setAttribute(attrs[c].nodeName, attrs[c].nodeValue);        
      }
      
     
      pic.parentNode.replaceChild(img, pic); //replace picture element with created img element
    }
  }
  


  done(); //finished.
});    

How It Works

Wrapping content in <noscript> tags stops it from being added to the DOM, whilst also providing complete content to non-js clients (such as search engine bots, screen readers, text browsers). On some browsers its possible to extract and process the content of the <noscript> tags using normal methods (e.g. getElementById etc.) Other browsers (IE, Safari) however, actually remove the content of the noscript tags as soon as the Javascript engine initializes. So Respondu uses a hack to to ensure the content can be extracted from all browsers.

We do this by dynamically wrapping the <noscript> tags in another context - there are several ways: comments, textareas, script tags, style tags...

After experimentation and thought, Respondu's chosen way is to use the <noscript>#content#</noscript--> technique, once Respondu has been called we have <--<noscript>#content#<noscript-->. The dashes in the closing noscript tag enables us to close a dynamically inserted comment opener.

Dynamically wrapping the noscript tags in a comment prevents browsers such as IE and Safari from stripping the contents out of noscript. Respondu relies on being called just before the noscript tags, so that it

  1. Knows where to insert the opening <!-- and
  2. Knows where in the document to insert the final processed code

As a result, we can't put HTML comments inside our noscript tags (which means conditional comments are out). If we really want to have comments within our noscript HTML we can use <script>/*comment here*/</script>

Once the noscript content has been extracted, Respondu loads it into a "ghost DOM". (see createHTMLDocument) The ghost DOM doesn't load any src's. We can manipulate this ghost DOM as the doc parameter when creating implementations (see Creating an Implementation)

When all changes have been made to our ghost document (e.g. when we've replaced img src's according to screen width), we load it into the real document

Browsers Confirmed as Working

  • Chrome
  • IE 7
  • IE 8
  • IE 9
  • Firefox
  • iOS Safari (iOS4, 5, 6)
  • Safari (win)
  • Opera

Contributing

I warmly welcome all contributions, you can help by

  • testing
  • pull requests
  • issues

Version changes

###0.0.2

  • Support for IE8 + IE7
  • Intelligently triggers window.onload and $(window).load in IE7/8/9
  • Improved code base
  • Modularized code for great customisation
    • R.base provides basic use
    • R.srcset, R.picture, and R.hybrid can be combined with R.base to become a polyfill for the respective implementation
    • Combine with doclate for seamless jQuery compatability

###0.0.3

  • Standardisation of picture element has made hybrid solution inaccurate/reduntant so hybrid has been removed, picture will be updated in 0.0.4
  • Now includes grunt.js file to automatically build R.js
  • Customized builds available in the builds folder
  • Future version will have a builder gui for fine grain control

0.0.4

  • Full uncustomized R.js minified and gzipped weighs at 2.8kb
    • This can be reduced with custom builds - e.g. just including picture and not srcset brings us down to 2.4kb
  • Picture updated to new syntax according to [proposed spec]http://dvcs.w3.org/hg/html-proposals/raw-file/tip/responsive-images/responsive-images.html)
  • API changes
    • New plugin API for creating plugins of implementations
    • changed Respondu() to simply Respondu()
  • New hack method: instead of <noscript></noscript></style> Respondu now uses <noscript></noscript-->
    • The closing </noscript--> must have the two dashes for cross browser compatibility
    • Warning Do not use <!-- --> (comment tags) inside the <noscript> tag*
      • This will break the functionality some browsers (ie, safari)
      • If you wish to make comments within the noscript tags use <script>/*comment here*/</script>
  • Behavioural changes
    • Instead of putting all content in <noscript> tags, only relevant content (such as a picture element) goes inside a noscript tag
    • The call to Respondu isolates the following <noscript></noscript--> tag, processes the code and reinserts it in the place where Respondu is called
    • This leads to a more familiar, progressive loading experience
    • It also means R.base has been trimmed down, there's no need for loadScripts, doclate, window.open hackarounds etc
      • Infrastructure examples have been removed, since this functionality is no longer needful
    • Any scripts should go outside the noscript tag, the contents of the noscript tags should only be for assets you wish to responsively select the source for
  • Stronger compatibility for source elements
    • It turns out some browsers cough iOS Safari cough sometimes strip source tags unless they're in a video element, this is secured against
  • Picture/srcset now check for a browser implementation before processing (i.e. Respondu can now be used as a polyfill)
  • Includes matchMedia polyfill for use in plugin implementations (and could be used anywhere in a project)

About

An Extendible Deferred Asset Responsive Framework

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published