Skip to content

Latest commit

 

History

History
149 lines (94 loc) · 13.7 KB

README.md

File metadata and controls

149 lines (94 loc) · 13.7 KB

Focal Point: Pure HTML/CSS Adaptive Images Framework

A small set of CSS classnames to help keep images cropped on the focal point for responsive designs. Using only HTML/CSS, web authors can specify an image's focal point, which stays as the image's primary focus as the image scales on responsive webpages. This puts web authors in control of art direction for responsive and high-resolution images. Crop and re-size images depending on available width and let CSS to do all of the work, and without any JavaScript. Resize your browser down to as small as it can go when viewing the demos.

Many of today's designs are migrating to the responsive web design technique coined by Ethan Marcotte. Elements should to be fluid and adjust to the available display, but a wrench is usually thrown into the system when exact pixel dimensions are set for images. The good news is that browsers only used to have poor image downsizing abilities. Today's browsers, however, do a remarkable job of resizing images on the fly, which is one of the primary reasons this small framework is now possible.

The Focal Point Framework gives web authors the flexibility of how responsive and hi-res images should be rendered depending on the image. For example, images can use common CSS classnames which allows a standard pattern to be resuabled throughout an entire site, which drastically reduces HTML markup and CSS required. Additionally, each individual image can also be given specific CSS for the general crop/size needed. All this without the use of any JavaScript!

Developers also have control to add and subtract from the framework's CSS as needed. The CSS is minimal in size, 2.98KB compressed (723 bytes gzipped), and can be grouped with existing CSS files as to not add any additional HTTP requests.

  • Author: Adam Bradley, cdnconnect.com (c) 2012
  • License: MIT/GPLv2

Demos

Be sure to adjust your browser window size to as small as it can go and take notice of the image's edges. As the image's available width changes, study how the image gets cropped as it narrows in on the image's focal point.

Also check out the DesignShack.net article Focal Point: Intelligent Cropping of Responsive Images by Joshua Johnson for a better visualization of how it works.

12x12 Grid

An image is represented in a grid of 12x12 units (no matter what its natural dimensions are), and you can locate the general focal point of each image using CSS classnames. I use the term "general" focal point because this framework is not built around exact pixels and cropping to strict dimensions. Instead it uses relative positioning to narrow in on general areas of an image in an effort to crop and resize with a simple and quick markup change. If you need to crop, resize and rotate images to exact pixel-perfect dimensions for exact breakpoints widths then this framework may not be for you.

Web authors can either use default settings to crop and resize images which centers in on the image, or can specify an image's general focal point. By controlling web authoring in markup, this puts content editors in control of how content images render as responsive webpages adjust to the display.

High-Resolution Images, But Smaller Filesizes!

While not required, you can also render high-resolution images using a technique from the blog post Retina Revolution by Daan Jobsis. Essentially you would serve high-resolution images, but the image would also be saved with high compression. Amazingly this works great for high-density devices, such as Retina displays, but as the same time, standard resolution devices are not affected by being served large file sizes. The technique is so simple its genius: "A smaller filesize AND a better quality on both screen types! This is impossible." I encourage you to read his entire post.

HTML Pattern

Not too scarey right? If you've ever used a div or img element before you should be good to go.

<div class="focal-point">
    <div><img src="http://demo.cdnconnect.com/images/hi-res/chrysler-plant.w1020.q20.jpg"></div>
</div>

Landscape and Portrait Aspect Ratios

Out of the box all images are considered to be landscape with a 4x3 aspect ratio, and portrait images are generalized to a 3x4 aspect ratio. These ratios are used to help crop images as they are resized. Because of the nature of the CSS em unit and its ability to adapt to its surroundings, using the em unit allows the crop and resize to be elastic to the image's natural dimensions. While cropping and resizing images, the em unit does not render strict 4x3 or 3x4 ratio images, but instead adjusts relative to the image's natural dimensions. Check out the demos to get a better idea of how they work.

CSS Framework Classnames

An image is represented in 12x12 units, and you can locate the general focal point of each image using CSS classnames. By not setting a left or right classname, the crop and resize will zoom in to the horizontal center. Likewise, by not setting an up or down classname, the crop and resize will zoom into the vertical center. Basically, by default it'll zoom into the center of the image. The classnames below allow you to crop and resize into a general area of the image. Note: X in the classnames represent a number between 1 and 6.

  • left-X: The images focal point is X out of 6 units left of the center. ie: left-1 would move the focal point 1 out of 6 units to the left.
  • right-X: The images focal point is X out of 6 units right of the center. ie: right-6 would move the focal point 6 out of 6 units to the right.
  • up-X: The images focal point is X out of 6 units up from the center. ie: up-3 would move the focal point 3 out of 6 units up.
  • down-X: The images focal point is X out of 6 units down from the center. ie: down-2 would move the focal point 2 out of 6 units down.
  • portrait: If an image has a portrait layout (its height is more than its width) then this classname will help to respect the image's natural aspect ratio while cropping/resizing into the focal point. The image will be cropped with a shape loosly representing an aspect ratio of 3x4 units. By default the 4x3 shape (landscape) is applied and doesn't need to be specified, but this can be overwritten for portrait images. You'll find that landscape will be able to work for most images, and portrait works for the rest. Note that images still crop/resize to the focal point even when not assigning the portrait classname on portrait images, however it doesn't respect the natural aspect ratio as much. Additionally, any shape needed can be added by the developer for uncommon image dimensions. If you do not need the portrait classname then feel free to cut out the CSS to save yourself a few bytes.

Below is an example of setting the image's focal point 1 out of 6 units to the right of the center, and 2 out of 6 units up from the center:

<div class="focal-point right-1 up-2">
    <div><img src="http://demo.cdnconnect.com/images/hi-res/chrysler-plant.w1020.q20.jpg"></div>
</div>

Sass mixin

The mixin gives web authors using Sass the ability to fully customize the Focal Point framework to their needs.

  • Custom grid: You're able to define your own grid by adding more units for better focal precision (more classes) or use fewer units for a simpler grid and minimal CSS output (fewer classes). $grid defaults to 12.

  • Landscape and/or portrait classes: Choose whether you prefer to generate landscape or portrait classes or both. $landscape-classes and $portrait-classes default to true.

  • Custom breakpoint: Set your own media query breakpoint. $breakpoint defaults to 767px.

  • Custom inner element: You can use a different inner element like a <figure> or any element with a .class-name. $inner-element defaults to div.

  • Custom default zoom factor: Set your own base zoom factor for more or less zooming. $zoom-factor defaults to 1.

  • Additional zoom levels: You can generate multiple zoom classes, such as .zoom-2 and .zoom-3 if you wish to use multiple zoom levels. $zoom-levels defaults to 1.

Usage:

First, copy the _focal-point.scss file into your Sass folder.

Then import the partial into your Sass document:

@import 'focal-point';

Finally include the mixin and tweak the settings if needed:

@include focal-point(
    $grid: 12,
    $landscape-classes: true,
    $portrait-classes: true,
    $breakpoint: 767px,
    $inner-element: div,
    $zoom-factor: 1,
    $zoom-levels: 1
    );

Discussion Points

Zoom Classnames: The standard CSS uses a zoom factor I considered to be general enough for most cases. But some images may want more or less "zooming" depending on the image. There could easily be additional CSS classnames, such as .zoom-2 and .zoom-3, which could set how slow and fast the crop and resize adjusts. The only reason I did not create zoom classnames was to keep the CSS minimal. However, if you have a need for various zoom factors then go for it since it's definately possible. The Sass mixin allows you to generate multiple zoom level classes and set a different base zoom factor.

Multiple or No Media Queries: Same as the zoom classnames, I chose a general breakpoint and only included one media query for common cases. This isn't to say you couldn't have two, five, or even no media queries all. Depending on your use case you could even change zoom factors between each media query. What's included with the standard CSS is general enough for most cases, or at least a good starting point for your own project.

Browser Support: It appears that IE8 and below does not crop the image, but it does resize the image according to its available width. I don't see this as a show stopper what-so-ever. If the worst case scenario is that IE8 users do not get the cropped image on responsive webpages, but instead view images no different that every other image created today, then I'm fine with that. Review the demos in IE8 to see what I mean. (Let's not forget that it's just two div's and an img element). Also, if you come across any other issues, or know of a better way to write the HTML/CSS please let me know.

See a way to improve this? Contribute / Contact Me

Is there a better/simpler/easier/smaller way to write the HTML/CSS all while keeping it reusable, compact, simple and meets the requirements below? Submit a pull request or contact me: @adamdbradley

Requirements:

  • All browsers must be able to at least view the image (ie: images do not crop and change focal points for IE8 and below, but the image still resizes according to available width. In my book that's still good to go. If a different method doesn't allow the image to load at all in IE8 then that's a no go.)
  • Only one img request per image
  • Set image breakpoints depending on available CSS width
  • Re-crop and re-size images depending on available CSS width
  • Image web authoring can be assigned to sitewide patterns
  • Each image can also have its own individual crop/resize
  • Keep the CSS file as small as possible
  • Works without JavaScript
  • Entirely static and controlled only by HTML/CSS. ie: No requirements for HTTP headers/cookies
  • No DOM manipulation
  • Crawlable by search engines so the image can be indexed and associated to page content
  • Accessible text (ie: alt tag)
  • Printable images
  • Context menu usable when "right-clicking" content images, ie: "Save image as..."

Other Resources