Create a Retro Navigation Menu with CSS3

Preview

Topic: CSS3
Difficulty: Intermediate
Estimated completion time: 50 minutes

In this you’ll learn how to create a vintage-looking navigation menu using pure CSS3. I hope you’ll enjoy this tutorial and you’ll learn some new techniques.

Resources needed:

Step 1: HTML Markup

<ul class="nav">
	<li><a href="#">Home</a></li>
	<li>
		<a href="#">Portofolio</a>
		<ul>
			<li><a href="#">Webdesign</a></li>
			<li><a href="#">Development</a></li>
			<li><a href="#">App Design</a></li>
			<li>
				<a href="#">Identity</a>
				<ul>
					<li><a href="#">Level 2</a></li>
					<li><a href="#">Level 2</a></li>
					<li>
						<a href="#">Level 2</a>
						<ul>
							<li><a href="#">Level 3</a></li>
							<li><a href="#">Level 3</a></li>
							<li><a href="#">Level 3</a></li>
							<li><a href="#">Level 3</a></li>
						<ul>
					</li>
					</li><a href="#">Level 2</a></li>
				</ul>
			</li>
			<li><a href="#">Marketing</a></li>
		</ul>
	</li>
	<li><a href="#">About</a></li>
	<li><a href="#">Blog</a></li>
	<li><a href="#">Services</a></li>
	<li><a href="#">Contact</a></li>
</ul>

We are going to use a font from Google Web Fonts (Oswald), so you’ll also have to include this line inside the head of your HTML file:

<link href='http://fonts.googleapis.com/css?family=Oswald:700' rel='stylesheet' type='text/css'>

Step 2: Background Pattern

As you can see from the preview image, the menu has a subtle blue pattern as background, but I said that we will use only CSS3. So, how is that possible? We are going to use a technique called Data URI scheme. This allows you to encode a resource into base 64 code and include that code in the CSS file. What is the main advantage of this technique? It will reduce the amount of HTTP requests.

Let’s open up Photoshop and load the pixel patterns. Create a new file (CTRL/Command + N) with the size 10 x 10 pixels. Fill the background layer with the color #65c0bb and duplicate it (right click > Duplicate Layer). Select the background layer, go to Blending Options > Pattern Overlay and apply these settings:

pattern Overlay

Select the duplicated layer and go to Filter > Noise > Add Noise:

noise

Then, set the blending options of the layer to “Soft Light” and lower the opacity to 50%:

blending Options

Save the file as a PNG image and go to this website which has a very useful tool that will allow us to convert the image into code. Upload the image (select “Browse”) and hit the “Convert” button. Keep the page opn or save the line of code with the background-image property, as we’ll need it for the next step.

Step 3: Styling the Main Menu

We’ll start with some stylesheet reset code:

.nav, .nav ul {
	margin: 0;
	padding: 0;
	list-style: none;
	line-height: 1;
}

Next, we’ll modify the main navigation bar. We’ll make it 900 pixels wide, center the text and add some shadows. Also, we’ll set the background, so this is where we need the code generated from our pattern:

.nav {
	position: relative;
	margin: auto; /* Centering the menu */
	height: 46px;
	width: 900px;
	text-align: center;
	box-shadow: 0 1px 2px rgba(0, 0, 0, .3);
	background: #65c0bb;
	background-image: url();
}

For now, we’ll hide the submenus:

.nav ul {
	display: none;
}

Now let’s format the links a little bit:

.nav>li {
	margin: 0;
	line-height: 1;
	padding: 0;
	display: inline;
	position: relative;
	margin: 0 12px;
}

The menu should look like this:

step 3

Step 4: The Ribbons

We’ll create the ribbons using pseudo-elements and some border tricks:

.nav::after, .nav::before {
	content: "";
	display: block;
	position: absolute;
	top: 6px;
	height: 0px;
	width: 0px;
	border: 23px solid #65c0bb;
	z-index: -1;
}

/* The left ribbon */
.nav::before {
	border-left-color: transparent;
	left: -30px;
}

/* The right ribbon */
.nav::after {
	border-right-color: transparent;
	right: -30px;
}

step 4

Step 5: The Main Links

The main links are going be white, with some text shadow to enhance the retro effect. On mouse over, they’ll change their color to a muted yellow:

.nav>li>a {
	display: inline-block;
	padding: 15px 20px;
	position: relative;

	font-family: 'Oswald', sans-serif;
	font-size: 16px;
	text-transform: uppercase;
	text-decoration: none;
	color: #fff;
	text-shadow: 1px 2px rgba(0, 0, 0, .2);

	-webkit-transition: color .3s linear;
	   -moz-transition: color .3s linear;
	     -o-transition: color .3s linear;
	    -ms-transition: color .3s linear;
	        transition: color .3s linear;
}

.nav>li>a:hover, .nav>li:hover>a {
	color: #eae8a5;
}

Next, we have to create the stars that separate the links. Open Photoshop again and create a new 15 x 15 pixels file. Make sure your file has the background transparent. Open the UI Icons PSD file that you downloaded and select the star icon layer. Copy the layer into your new file. Go to Edit > Transform Path > Scale and scale down the layer to 80% (both width and height). Then, right-click the layer and select Clear Layer Style. Next, select Blending Options and apply the following settings:


Save your file as a PNG image. Go to the Image to Data URI Convertor again and convert this image as well. The code obtained will be the background image for the pseudo-elements placed to the right of every link:

.nav>li>a::after {
	content: "";
	height: 15px;
	width: 15px;
	background-image: url();
	position: absolute;
	right: -20px;
	top: 16px;
	display: block;
}

The last link doesn’t need a star icon to its right, so we won’t display its pseudo-element:

.nav>li:last-child>a::after {
	display: none;
}

Step 6: Styling the Submenus

First, search and delete these lines:

.nav ul {
	display: none;
}

Next, we are going to make some basic general styling. All the submenus will be hidden (pushed 9999 pixels to the left) and their initial opacity will be 0. That’s because we want to obtain a subtle fade effect using CSS3 transitions.

.nav ul {
	position: absolute;
	left: -9999px;
	padding-top: 10px;
	border-bottom: 1px solid #ccc;
	opacity: 0;
	-webkit-transition: opacity .3s linear;
	-moz-transition: opacity .3s linear;
	-o-transition: opacity .3s linear;
	-ms-transition: opacity .3s linear;
}

The first level submenus will appear under its parent link:

.nav>li:hover>ul {
	left: 0;
	opacity: 1;
	top: 30px;
}

The rest of the submenus must appear to the left of their parent link:

.nav ul li:hover>ul {
	left: 150px;
	opacity: 1;
	top: -11px;
	padding-left: 12px;
	border-bottom: 0;
	box-shadow: none;
}

Right now, the dropdown menu looks like this:

step 6-1

Not very nice, huh? Well, hang in there, we still have a pretty long way to go. First, let’s deal with the list elements which contain the links:

.nav ul li {
	display: block;
	position: relative;
	border-top: 1px solid #ccc;
	border-left: 1px solid #ccc;
	border-right: 1px solid #ccc;
	width: 150px;
	text-align: justify;
	z-index: 9;
	background: #eee;
	box-shadow: 3px 4px 0 rgba(0, 0, 0, .1);
	-webkit-transition: background .3s linear;
	-moz-transition: background .3s linear;
	-ms-transition: background .3s linear;
	-o-transition: background .3s linear;
}

step 6-2

I’ll explain a little bit what I’ve done. I gave the list elements a fixed width of 150 pixels, a light grey background, some borders and crisp shadows for the retro look. Also, they inherited the property for centered text from their parent, so I had to overwrite it, using text-align: justify.

Next, let’s add some style to the links:

.nav ul li a {
	font-family: "Oswald", sans-serif;
	font-size: 14px;
	text-decoration: none;
	display: block;
	padding: 7px 12px 7px 20px;
	color: #65c0bb;
	-webkit-transition: color .3s linear;
	-moz-transition: color .3s linear;
	-ms-transition: color .3s linear;
	-o-transition: color .3s linear;
}

Also, let’s take care of their hover state:

.nav ul li:hover>a, .nav ul li a:hover {
	color: #4db6b0;
}

.nav ul li:hover {
	background: #f7f7f7;
}

Now the menu looks like this:

It looks quite good, but there seems to be a problem: the first submenu has a bottom border, but the second doesn’t. So, let’s fix that:

.nav ul ul li:last-child {
	border-bottom: 1px solid #ccc;
}

You are probably wondering why I didn’t set the border for the unordered list. That’s because the unordered list has a 12 pixels padding to its left and the border would be longer than necessary. This is how it would look if I’d set a border for it:

step 6-e

Here’s how it looks instead:

step 6-4

Another bug pops up: the shadow of the second level submenu is bigger than the one of the first level submenu. Let’s solve this one as well:

.nav ul ul li {
	box-shadow: 3px 3px 0 rgba(0, 0, 0, .1);
}

step 6-5

The menu looks quite nice now, but it still feels like something’s missing. That’s right, in the preview image the submenus had some little arrows pointing to their parent links. We’re going to create them by using pseudo-elements. The pseudo-elements will be just some 9 by 9 pixels squares, rotated by 45 degrees. Their background and border must match those of the submenus. First, some base styling (we’re going to set their dimensions and positioning context):

.nav ul::after, .nav ul::before {
	content: "";
	display: block;
	z-index: 1;
	position: absolute;
	height: 9px;
	width: 9px;
}

First, let’s deal with first level submenu:

.nav>li>ul::after {
	border: 1px solid #ccc;
	background: #eee;
	border-right: 0;
	border-bottom: 0;
	top: 5px;
	left: 25px;
	-webkit-transform: rotate(45deg);
	-moz-transform: rotate(45deg);
	-o-transform: rotate(45deg);
	-ms-transform: rotate(45deg);
}

step 6-6

It looks nice, but that border is a little bit annoying. We can get rid of that by overlaping it with another pseudo-element which matches the background of the submenu:

.nav>li>ul::before {
	height: 1px;
	width: 12px;
	background: #eee;
	border-right: 0;
	border-bottom: 0;
	top: 10px;
	left: 24px;
	z-index: 99;
}

Let’s do the same thing for the higher level submenus:

.nav ul ul::after {
	border: 1px solid #ccc;
	background: #eee;
	border-right: 0;
	border-bottom: 0;
	top: 20px;
	left: 8px;
	position: absolute;
	-webkit-transform: rotate(-45deg);
	-moz-transform: rotate(-45deg);
	-o-transform: rotate(-45deg);
	-ms-transform: rotate(-45deg);
}

.nav ul ul::before {
	height: 10px;
	width: 1px;
	background: #eee;
	z-index: 99;
	top: 20px;
	left: 12px;
}

step 6-8

Well, that’s it! I hope you liked the tutorial and you learned some new things!

Subscribe to Unlock

Subscribe to our newsletter to get access to this content. Your email will not be sold/rented. Unsubscribe at any time.
- required fields

Enter the e-mail, which was subscribed

After that, the content becomes available again!
- fields are mandatory

On your email was sent a letter

Simply click on the link in the email and you will get access to the hidden content!

There is no information about this email in our database. Please subscribe by clicking on button below.

Your e-mail is not in our database. Subscribe by clicking the button below
I'm already a subscriber

Tiberiu Butnaru

I am Tiberiu Butnaru, a front end developer and Computer Science student from Romania. I love doing all kinds of experiments with HTML5 and CSS3. You can connect with me on Twitter at @tiberiualex

Newsletter

18 Comments
  1. Maarten Jan 25, 1:45 pm

    That’s very cool! I can really use this in my next project!

    Reply
    +12
  2. Agustín Jan 25, 3:20 pm

    Amazing design! A very gentle color selection! Loved it!

    Reply
    +1
    • Tiberiu Jan 25, 10:03 pm

      Thanks! I had a pretty hard time choosing the colors, as this was one of my first vintage designs.

      Reply
      +1
  3. Harish Jan 25, 4:23 pm

    Really neat Tiberiu. Thanks for the tutorial.

    Reply
    +1
  4. tjam Jan 25, 5:38 pm

    Instead of making pattern image in Photoshop you could use tools like draw2base – http://kandeefloss.com/draw2base/ or patternify http://www.patternify.com/ and get base64 code immediately.

    Reply
    +4
    • Tiberiu Jan 25, 10:43 pm

      Thanks for the tip! :)

      Reply
      0
    • Nicole Ingram Jun 13, 6:11 am

      Hi how do you then download that file into a pattern file for photoshop? or do you just paste this file into photoshop?

      Reply
      0
  5. Earl Varona Jan 26, 8:09 pm

    Very nice. I’ve always wondered how to make that little arrow in CSS and you can probably have the hover effect on it with the first-child. Thanks. :)

    Reply
    0
  6. Patrick Groot Jan 31, 11:09 am

    Can’t make the menu to work but great tut.

    The list only shows home and portfolio, maybe something wrong with the list.

    Reply
    0
    • Sonja Mar 5, 1:25 pm

      Yeah I had the same problem at first. I think there was on closing ul tag that didn’t belong there, after I deleted that it worked for me.
      Just check over the HTML again.

      Reply
      0
    • Sonja Mar 5, 1:36 pm

      I love this tutorial, only I can’t get rid off that border in the tiny arrow, and I don’t understand why…
      When I changed the color of the overlapping piece I still couldn’t see it. It’s just not there. Any ideas what I am doing wrong?

      Reply
      0
    • Aston Jul 15, 5:20 pm

      Hey …!!
      You need to check out 21 line in the step 1 close that “ul” tag you will get it right

      Reply
      0
  7. Niels Feb 16, 10:21 pm

    I wouldn’t call it a “retro” design, since this isn’t exactly something that comes from webdesign from the past, it’s something new.

    But it looks nice nonetheless.

    Reply
    0
  8. Kate Feb 17, 6:34 am

    I absolutely adore this menu. It is exactly what I had in mind for a layout of mine. I love the retro feel and the dropdown with the arrow. Thanks for sharing.

    Reply
    0
  9. Robin Henderickx Feb 20, 4:48 pm

    I saw there was a typo in your body tag on line 23: where te openingtag sould be, there is a closing tag ;)

    GREAT tutorial, it helped me a LOT ^^

    Reply
    0
  10. Marcelle Apr 6, 9:25 pm

    I was unable to implement the menu on my blog and I really loved it. Could you help me?

    Reply
    0

Leave a Reply

*
* Minimum length: 20 characters