Better HTML & CSS Tutorial
Better HTML & CSS Tutorial
Better HTML & CSS Tutorial
BETTER HTML
& CSS
[email protected]
@chriscoyier
CHRIS
COYIER
BETTER?
1 Less of it
2 More Semantic
3 More Accessible
4 Futureproof
// Speed
// Maintainability
// SEO
// Happy People
23 1
HOW WUFOO
DOES CSS
1
No inline styles or <style> blocks
Only 2 CSS files per page (Global + Site Section)
These 2 are made from combined smaller files
(like design patterns or site sub sections)
Versioned in production (dynamic.13432.css)
Timestamped in development (dynamic.TIMESTAMP.css)
Reuse everything (e.g. table.css)
Work in chunks (e.g. print.css)
RULES
SIMPLIFIED SITE STRUCTURE
GLOBAL CSS
GLOBAL CSS
SITE SECTION CSS
SITE SECTION CSS
<!DOCTYPE html>
<head>
{autoVersion}/css/global/dynamic.css{/autoVersion}
{autoVersion}/css/landing/dynamic.css{/autoVersion}
AddHandler application/x-httpd-php .php .html .xml .css .js
.htaccess
<?php
require_once($GLOBALS['root'].'/library/services/
AutoVersion.php');
$fileArray = array(
'/css/global/structure.css',
'/css/global/buttons.css',
'/css/global/lightbox.css',
'/css/global/form.css'
);
$av = new AutoVersion();
$av->fly($dynamicURL,$fileArray);
?>
/css/global/dynamic.css
<?php
require_once($GLOBALS['root'].'/library/services/
AutoVersion.php');
$fileArray = array(
'/css/landing/structure.css',
'/css/landing/table.css',
'/css/landing/else.css',
'/css/landing/signup.css',
'/css/landing/tour.css'
);
$av = new AutoVersion();
$av->fly($dynamicURL,$fileArray);
?>
/css/landing/dynamic.css
1) Fetches all files
2) Minifies*
3) Combines Together*
4) Adds version number
AutoVersion function:
<!DOCTYPE html>
<head>
<link rel="stylesheet" href="/css/global/dynamic.1234.css">
<link rel="stylesheet" href="/css/landing/dynamic.1234.css">
global/dynamic.css
Loaded on every page of site
Put as much as practical in here. User only loads this
file once, so maximizes use of browser cache.
Common design patterns are in here
(buttons.css, lightbox.css, forms.css)
area/dynamic.css
Loaded in specific area of site
Less common design patterns in here
(graph.css, calendar.css, table.css)
global
/css/global/structure.css
/css/global/buttons.css
/css/global/lightbox.css
/css/global/form.css
area
/css/admin/manager.css
area
/css/widgets/datagrid.css
/css/global/filter.css
/css/global/calendar.css
/css/global/quicksearch.css
/css/entries/structure.css
/css/entries/print.css
global
/css/global/structure.css
/css/global/buttons.css
/css/global/lightbox.css
/css/global/form.css
area
/css/docs/docs.css
/css/global/table.css
global
/css/global/structure.css
/css/global/buttons.css
/css/global/lightbox.css
/css/global/form.css
ALL
CSS
is in /css/
organized by
site section
DONT OVER
THINK IT
2
Primary color #BADA55 / Secondary color #F00
Headers should be 20px from navigation and
15px from following content
Logo should have 30px of padding around it
Links should have 1px dotted bottom borders
BIG FANCY STYLE GUIDE
thats what
GLOBAL.CSS
is for
NEED TO DEVIATE?
Really? Do you?
BY
SECTION
STYLE
ONLY
NEED TO DEVIATE?
Really? Do you?
TOTALLY
UNIQUE
NEED TO DEVIATE?
Really? Do you?
DONT
OVER
THINK
IT
/* Good thing we specified this link is a
descendant of html, so none of those
crazy outsider links get this style */
html body #page-wrap .inner ul li ul li a {
font-size: 0.8em;
}
li li li li {
font-size: 1.0em;
/* Erm, use REM */
}
a {
text-decoration: none;
}
/* Killing a fly with a jackhammer */
#page-wrap a {
text-decoration: underline;
}
/* Thus begins a nasty cycle */
aside a {
text-decoration: none !important;
}
CHALLENGE:
Dont use IDs in your
CSS whatsoever for
one small project.
Just try it
PSEUDO
ELEMENTS
3
:visited :hover :active :link
:rst-child :last-child :nth-child() :nth-of-type()
:enabled :disabled :checked :indeterminate
:focus :target :root :lang()
http://css-tricks.com/pseudo-class-selectors/
:before
:after
<div>In</div>
div:before {
content: "Robots ";
}
HTML
CSS
In
<div>In</div>
div:before {
content: "Robots ";
}
In
div:after {
content: " Disguise";
}
Robots Disguise
HTML
CSS
So whats with
the diferent name?
Pseudo selectors select elements that
already exist (perhaps in diferent states).
Pseudo elements create new content that
doesnt exist (yet).
::before
::after
::rst-line ::rst-letter
:before
:after
:rst-line :rst-letter
<div>In</div>
div:before {
content: "Robots ";
}
In
div:after {
content: " Disguise";
}
Robots Disguise
HTML
CSS
<div>
In
</div>
Resulting
HTML
(sorta)
Robots
<div>
In
</div>
Disguise
Not before/after the element...
Resulting
HTML
(sorta)
<div>
Robots
In
Disguise
</div>
Its before/after the content inside.
Resulting
HTML
(sorta)
<div>
<h1>Blah blah blah</h1>
<p>More stuff</p>
Nothing to see here.
</div>
Resulting
HTML
(sorta)
<div>
Robots
<h1>Blah blah blah</h1>
<p>More stuff</p>
Nothing to see here.
Disguise
</div>
Resulting
HTML
(sorta)
Its only a model...
(Not really in DOM)
C
A
M
E
L
O
T
!
C
A
M
E
L
O
T
!
C
A
M
E
L
O
T
!
<img src=robot.jpg alt=Robot>
<input type=email name=email />
<br>
Not for no content elements
Allows but shouldnt
Styles as if was inside
Checkboxes
Radio Buttons
BUTTONS
WITH ICONS
CSS
.button {
/* Awesome gradients and stuff */
}
.button img {
/* Probably some margin and stuff */
}
HTML
<a class=button href=http://wufoo.com/gallery/>
<img src=/images/icon_gallery.png alt=>
Visit Our Form Gallery
</a>
alt=""
equals
Thats not important.
Screen readers dont need to see that.
alt=""
equals
Then get that mothersucker
out of your HTML
HTML
CSS
.button {
/* Awesome gradients and stuff */
}
.button-gallery:before {
content: url(/images/icon_gallery.png);
}
<a class=button button-gallery href=http://wufoo.com/gallery/>
Visit Our Form Gallery
</a>
x200
<a class=button href=http://wufoo.com/gallery/>
<img src=/images/icon_gallery.png alt=>
Visit Our Form Gallery
</a>
200 extra lines of HTML
200 places you arent being semantic
200 places you need to change one-by-one
200 opportunities to be more efcient
<html style=background: yellow;>
CSS
CSS
html {
background: red;
}
EFFICIENCY!
HTML
CSS
.button {
/* Awesome gradients and stuff */
}
.button-gallery:before {
content: url(/images/icon_gallery.png);
}
<a class=button button-gallery href=http://wufoo.com/gallery/>
Visit Our Form Gallery
</a>
<a class=button button-gallery href=http://wufoo.com/gallery/>
Visit Our Form Gallery
</a>
x200
.button-gallery:before {
content: url(/images/icon_gallery.png);
content: ;
display: inline-block;
width: 16px;
height: 16px;
background-image: url(/images/sprite.png);
background-position: -32px -32px;
}
spritecow.com
spriteme.org
UNFORTUNATE
NAME
<h1></h1>
<h2></h2>
h1:before {
content: Wufoo;
}
h2:before {
content: Making forms easy + fast + fun;
}
HTML
CSS
RABBLE RABBLE RABBLE!
SCREEN READERS
NVDA doesnt read
Jaws doesnt read
Window Eyes doesnt read
VoiceOver (OS X) does read
Testing (mostly) by Lucica Ibanescu
http://cssgallery.info/testing-the-accessibility-of-the-css-generated-content/
Bad for accessibility
Bad semantically
Bad for SEO
.thing:before {
content: ?
}
What can content be?
TEXT / STRING
content: $;
content: \0022;
IMAGE
content: url(i/icon-smile.png);
content: -webkit-linear-gradient(...);
ATTRIBUTE
content: attr(href);
content: attr(data-city);
COUNTER
content: counter(li);
counter-increment: li;
counter-reset: li;
NOTHING
content: ;
HTML
content: <h1>Nope</h1>;
.price:before {
content: $;
}
[lang=cn] .price:before,
.price[lang=cn]:before {
content: \00a5;
}
<div class="price">30</div>
<div class="price" lang="cn">100</div>
TEXT / STRING
http://www.456bereastreet.com/lab/styling-ordered-list-numbers/
COUNTER
ol {
counter-reset: li;
list-style: none;
}
ol > li:before {
content: counter(li);
counter-increment: li;
background: #666;
color: white;
padding: 10px;
}
COMBINING WITH MEDIA QUERIES
mobile portrait
mobile landscape
CSS
@media (min-width: 1001px) {
aside li a:after {
content: " (" attr(data-email) ")";
font-size: 11px;
font-style: italic;
color: #666; }
}
@media (max-width: 1000px) and (min-width: 700px) {
aside li a:before {
content: "Email: ";
font-style: italic;
color: #666; }
}
@media (max-width: 699px) and (min-width: 520px),
(min-width: 1151px) {
aside li a {
padding-left: 21px;
background: url(i/email.png) left center no-repeat; }
}
Self!
You know what
would be neat?
You fade in
pseudo elements
on hover.
TOTAL
EPIC
FRICKING
DISASTER
You cant animate
or transition
pseudo elements.
But WAIT!
You totally can
in Firefox 4+
ATTRIBUTE
a {
position: relative;
}
a:after {
content: attr(data-tooltip);
position: absolute;
bottom: 100%;
left: 20px;
background: yellow;
opacity: 0;
}
a:hover:after {
opacity: 1;
}
a:before {
/* triangle action */
}
Nicolas
Dr. Pseudo Element
Gallagher
http://nicolasgallagher.com/
@necolas
You cant talk about Pseudo Elements
without talking about...
Multiple Backgrounds
Multiple Borders
Background Opacity
Clear Floats
Responsive Data Tables
http://css-tricks.com/9516-pseudo-element-roundup/
Shapes!
T
h
ese are easy.
T
h
ese are less easy.
.star {
width: 0;
height: 0;
border-left: 50px solid transparent;
border-right: 50px solid transparent;
border-bottom: 100px solid red;
position: relative;
}
.star:after {
width: 0;
height: 0;
border-left: 50px solid transparent;
border-right: 50px solid transparent;
border-top: 100px solid red;
position: absolute;
content: "";
top: 30px;
left: -50px;
}
http://css-tricks.com/examples/ShapesOfCSS/
http://nicolasgallagher.com/pure-css-gui-icons/demo/
Browser Support
3.5+
3.0- positioning issues
6+
9+
8 :: / :hover / z-index
7-
1+
1.3+
http://css-tricks.com/browser-support-pseudo-elements/
Remember, CSS TWO not THREE
85%
CSS-Tricks
97%
Other tech
92%
23 1
Links
http://necolas.github.com/normalize.css/
http://snook.ca/archives/html_and_css/font-size-with-rem
http://particletree.com/notebook/automatically-version-your-css-and-javascript-les/
http://css-tricks.com/855-specics-on-css-specicity/
Photos
http://www.ickr.com/photos/webel/347801397/
Type
Gotham Condensed
Gotham Rounded
TUNGSTEN
Whitney
Thanks!
bit.ly/fowa-better-css