What is responsive design?
Responsive Design might better be named "Device-Independent Design" or
"Flexible-Width Design." It is the term commonly used to describe a web layout that adjusts
itself (or "responds") to the available width of the window in which it is being viewed
— from small mobile device screens to 4k displays.
Responsive layout is an expected feature of any modern web design, and is perhaps the
biggest single factor that makes designing for screens different than designing for print or
other media. The good news is that browser default styles are (for the most part) already
responsive. That is, an HTML document with no stylesheet will generally display without
complaint on any device's browser, and the only user-interaction required to view the
content will be vertical scrolling. (Images are a notable exception this statement, but one
that is easy to remedy with a simple line of css: img { max-width: 100%; }
)
- Create a link stylesheet for this document.
- Add the rule:
img { max-width: 100%; }
.
The viewport meta-tag
In the dark, early days of the internet, web-designers paid little attention to designing
responsively, and instead built fixed-width layouts for a 'lowest common denominator' screen
size, usually 960px wide. That quickly became untenable with the advent of the iPhone, as
well as HD and higher-resolution displays. Unfortunately, millions of pages were built with
fixed widths before we got wise. Smaller devices need a way to handle these 'legacy' pages,
and a way to distinguish them from pages which use responsive layout. Enter the
viewport meta tag. This tag is the way we tell the browser that our page uses
responsive layout, and should there not apply the browser's default 'pinch
and zoom' feature intended for legacy pages.
The tag usually looks like this:
meta name="viewport" content="width=device-width, initial-scale=1.0"
. It belongs in the head of the document, and is critical to
proper rendering of your page on a mobile device. From this assignment forward,
every page you build should include this tag.
- Add the viewport meta tag to this document, in the head, where indicated.
Testing responsive behavior
While we can crudely test a page's responsiveness by simply re-sizing the browser window,
the Chrome developer tools give us a better way to emulate small devices, as well as test
how a page renders at any width. Open the inspector, and find the "Toggle Device Toolbar"
button, shown below.
You will likely now see this page re-rendered in a narrow view, controlled by the settings
in the toolbar now visible above the page.
-
Toggle the button as noted. Try setting the device-type to iPhone X, or
iPad, then try Responsive.
The Responsive device option, rather than simulating a specific device, allows you
to test a range of widths/heights by grabbing the drag-handles now found at the right and
bottom of the page.
- Drag the resize-handles and watch the page re-flow.
-
Experiement with the zoom selector in the menubar. Make sure to end on
100%
for zoom.
Manage scale by setting limits, rather than setting widths
One thing you'll have noticed about this page is that text is going all the way to the edge
of the page, with insufficient margin. Lines of text are as wide as the window, no matter
how wide that is. This is responsive... but hard to read. We should add some left and right
margin space. There's also an upper limit beyond which we really don't need (or want) the
line-length to grow. When there's excess space available it might be better to just add
left/right margin, rather than expand the content area. When width is limited, we want to
remove excess margin first, but the content itself will also need to be narrower. We cannot
achieve our goal by setting a fixed width for the content. Instead, let's use css properties
and proper unit selection to set some limits, rather than fixed sizes.
Let's start by making left and right margins that grow proportionally with the page.
- Create a ruleset for the
body
tag.
- Add
margin-left: auto; margin-right: auto;
- Add
width: 90%;
This approach is a little odd, but it's very useful. By using auto
for the
margin values, we can now trust that the page will center the
body
element in the available space, no matter big or small it is.
Setting the width of the body as a percentage means it will grow as the window width grows,
and the space left for margin will grow proportionally. You can use the 'Responsive'
device-emulation mode to confirm this.
Now let's set an upper limit for how wide we want the content to grow. This is often
dictated by what we consider to be a comfortable line-length for reading a column of text,
so it's appropriate in this case to set our upper limit in relation to the base font-size of
the document. The rem
unit is perfect for this.
- Add
max-width: 40rem;
to your body
ruleset.
The max-width
property overrules the width
property when needed,
so our page now has an upper width limit of 40rem
for content. When the
available width is larger, margin: auto;
is keeping our content centered. When
available width is smaller, our content uses 90% of the available width, leaving 5% on each
side for margin.
This example (controlling content width to prevent over-long lines of text) is a common
scenario, but the particular approach we used here is just one way to solve the problem. The
important thing is the mindset: how do I want this element to grow as the page width grows,
and what limits should it have? Managing how things scale and grow, rather than setting
fixed values, is critical to responsive design.
Responsive images
The browser does not do a great job handling img
elements
responsively without some help. By default, it shows images at their actual size. This page
already has a style rule applied to prevent img
elements from
extending beyond their parent containers: img { max-width: 100%; }
. In many
cases, this is all that's needed to make sure you're images scale appropriately with
available width. Having set that rule, you can observe using Chrome's 'responsive' mode that
the screenshots above scale down nicely to fit their parent container as we shrink the
available width.
Media queries
Setting limits only gets us so far. Most of the time, the single-column layout that works
best for a mobile device is not ideally suited for wider screens, and we want to take
advantage of the available space to build more complex grids and layouts.
Media queries allow us to selectively apply style rules based on the current
width of the page.
A simple example
Consider the following style rules:
html { font-size: 12pt; }
@media (min-width: 640px) {
html { font-size: 16pt; }
}
The first line, as we have discussed before, sets the base font-size for the document to
be 12pt
. The next three lines could be interpreted like this: "When the
window width is a minimum of 640px, the base font-size for the document should be
16pt
." The media query is just a conditional wrapper around one or more css
rulesets. If the condition is met, the rule(s) are applied. Let's add the rules above to
this page.
- Add the style rules as written above to your stylesheet.
-
Use the inspector responsive emulation mode to view the page at different widths. You
should see the base font-size increase when width is above 640px, causing all type to
get marginally larger.
Using media queries to modify layout
One of the most frequent needs for a responsive page is manage some content that is
stacked vertically for narrow views, but arranged side-by-side for wider views. There are
many, many ways to accomplish this, but one of the easiest and most powerful is to use
flex layout, like we did for grids.
In the example below, we have a div
element with two children,
each of which is also a div
. I've added a bit of styling so we
can see their boundaries easily, but they are behaving like any generic block-level
element: They fill the width of their parent container, and their height is controlled by
their content.
Let's use flex
to made these two containers site side-by-side, but only when
there's at lest 640px of available width.
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Proin non nisi porttitor,
finibus magna in, pharetra magna. Nunc a convallis ante. Vivamus aliquam rhoncus diam,
eget elementum nisi faucibus ultricies.
Donec quis arcu pharetra, imperdiet felis et, consequat ligula. Maecenas et sem lorem.
Nullam in venenatis nulla, in gravida justo. Sed sollicitudin mi sit amet scelerisque
facilisis. Maecenas vulputate mattis urna vel mollis. Vestibulum non gravida urna.
- Create a media query, with condition
(min-width: 640px)
.
-
Inside the query's curly-braces, create a css selector for
.two-columns-example
.
- Set
display: flex;
on .two-columns-example
.
-
Add another selector for
.two-columns-example > *
. This should live inside
the same media query. Its job is to select both children of
.two-columns-example
.
- For the children, set
width: 50%;
You should now be able to use Chrome 'responsive' mode to watch the behavior of this
element change at our 640px break-point: i.e.: the width at which we've decided
this element should change its visual behavior.
Guidelines for using media queries
Media Queries do not replace your default styles, they add to them.
DO NOT define all your styles for one width, then use a media query to
"start over again" for a different width. That's a recipe for confusion and massive
repetition in your stylesheet. Instead, start with the rules that apply everywhere, and
then add media queries ONLY to selectively provide additional rules for
the elements that need them. I find it easiest to keep my media queries close to the
primary elements they modify in my stylesheet. "Here are the styles for element X,
immediately followed by a couple additional styles that apply to the same element at large
widths."
For various reasons, many designers find it easiest to take a
mobile-first approach to building responsive pages. That is, style the page for
mobile, then 'grow' the width of the browser a bit, see what starts to have problems
visually, and create media queries to address those problems, until you've reached a
full-width layout. This approach usually uses @media (min-width)
conditions,
i.e. "When the window is at least this wide, apply these additional styles."
Other designers prefer desktop-first approach, where you build the full
complexity of the page first, then use @media (max-width)
conditions to
selectively simplify things as the page-width decreases. Read max-width as "When
the page is no larger than this wide, apply these rules." I prefer the first
approach personally, but either works. However, avoid doing both at the same time —
this is often a recipe for confusion and frustration!