No doubt you’ve seen the beautiful text field if you’re one of Gmail’s 2 billion active users:
data:image/s3,"s3://crabby-images/e2eb6/e2eb62b71dc4b284288006942c7eac4d4969d8eb" alt=""
It’s fluid, it’s intuitive, it’s colorful 🎨.
It’s Material Design: the wildly popular UI design system powering YouTube, WhatsApp, and many other apps with billions of users.
data:image/s3,"s3://crabby-images/5739f/5739f1c35e89896d5df6f55102fab8232153192e" alt=""
Let’s embark on a journey of recreating it from scratch with pure vanilla HTML, CSS, and JavaScript.
1. Start: Create basic input and label
As always we start with the critical HTML foundation, the skeleton:
The text input, a label, and a wrapper for both:
<!-- For text animation -- soon -->
<div class="input-container">
<input
type="text"
id="fname"
name="fname"
value=""
aria-labelledby="label-fname"
/>
<label class="label" for="fname" id="label-fname">
<div class="text">First Name</div>
</label>
</div>
data:image/s3,"s3://crabby-images/9b259/9b259c344c599e99646c6cc42de19332e9f17f2c" alt=""
2. Style input and label
I find it pretty satisfying: using CSS to gradually flesh out a stunning UI on the backs of a solid HTML foundation.
Let’s start:
Firs the <input>
and its container:
.input-container {
position: relative; /* parent of .label */
}
input {
height: 48px;
width: 280px;
border: 1px solid #c0c0c0;
border-radius: 4px;
box-sizing: border-box;
padding: 16px;
}
.label {
/* to stack on input */
position: absolute;
top: 0;
bottom: 0;
left: 16px; /* match input padding */
/* center in .input-container */
display: flex;
align-items: center;
}
.label .text {
position: absolute;
width: max-content;
}
data:image/s3,"s3://crabby-images/aac6c/aac6c77e100b9a78ee62f7a25e26a96856c8fe28" alt=""
3. Remove pointer events
It resembles a text field now, but look what happens when I try focusing:
data:image/s3,"s3://crabby-images/c99fe/c99fe02b6041779ccdf3268fae3f9706bd9b0fdd" alt=""
The label is part of the text field and the cursor should reflect that:
Solution? cursor: text
.label {
...
cursor: text;
/* Prevent blocking <input> focus */
pointer-events: none;
}
data:image/s3,"s3://crabby-images/e140b/e140bccf7fa9ec8ec622b47c7436bf66b90c4148" alt=""
4. Style input font
Now it’s time to customize font settings:
If you know Material Design well, you know Roboto is at the center of everything — much to the annoyance of some.
We’ll grab the embed code from Google Fonts:
data:image/s3,"s3://crabby-images/141ed/141ed9f18a9166f4b380efd0e04ba082dc66f3e0" alt=""
Embed:
data:image/s3,"s3://crabby-images/99ce7/99ce7f03eb817fab36bd8e9ff1dd514fa5c35c4b" alt=""
Use:
input,
.label .text {
font-family: 'Roboto';
font-size: 16px;
}
data:image/s3,"s3://crabby-images/b2ab1/b2ab1a914f1ee2f374093a2e0e7fa549c05f4233" alt=""
5. Style input on focus
You’ll do this with the :focus
selector:
CSS
input:focus {
outline: none;
border: 2px solid blue;
}
✅
data:image/s3,"s3://crabby-images/10663/10663f3de99a82e03c182d3d7bbf0544198c75e7" alt=""
6. Fluidity magic: Style label on input focus
On focus the label does 3 things:
- Shrinks
- Move to top input border
- Match input border color
Of course we can do all these with CSS:
input:focus + .label .text {
/* 1. Shrinks */
font-size: 12px;
/* 2. Move to top input border */
transform: translate(0, -100%);
top: 15%;
padding-left: 4px;
padding-right: 4px;
/* 3. Match input border color */
background-color: white;
color: #0b57d0;
}
All we need to complete the fluidity is CSS transition
:
label .text {
transition: all 0.15s ease-out;
}
data:image/s3,"s3://crabby-images/7bfaf/7bfafff1e5259a735f4978e81af37a266bd8d0f9" alt=""
7. One more thing
Small issue: The label always goes to the original position after the input loses focus:
data:image/s3,"s3://crabby-images/7cdb2/7cdb2992486406b73f02419f7966a8d11c627205" alt=""
:focus
which goes away on focus lost.But this should only happen when there’s no input yet.
CSS can’t fix this alone, we’re going to deploy the entire 3-tiered army of web dev.
HTML: input
value
to zero.
<input
type="text"
id="fname"
name="fname"
value=""
aria-labelledby="label-fname"
/>
CSS: :not
selector to give unfocused input label same position and size when not empty:
input:focus + .label .text,
/* ✅ no input yet */
:not(input[value='']) + .label .text {
/* 1. Shrink */
font-size: 12px;
transform: translate(0, -100%);
/* 2. Move to top */
top: 15%;
padding-left: 4px;
padding-right: 4px;
/* 3. Active color */
background-color: white;
color: #0b57d0;
}
And JavaScript: Sync initial input
value
attribute with user input
const input = document.getElementById('fname');
input.addEventListener('input', () => {
input.setAttribute('value', input.value);
});
const input = document.getElementById('fname');
input.addEventListener('input', () => {
input.setAttribute('value', input.value);
});
✅
data:image/s3,"s3://crabby-images/9607f/9607f301ab6e6ea4eb4b3fec8322118d5d90d7b4" alt=""
That’s it! We’ve successfully created an outlined Material Design text field.
With React or Vue it’ll be pretty easy to abstract everything we’ve done into a reusable component.
Here’s the link to the full demo: CodePen
Every Crazy Thing JavaScript Does
A captivating guide to the subtle caveats and lesser-known parts of JavaScript.
data:image/s3,"s3://crabby-images/ea530/ea530111f6f82839f5fb82e185233134334567be" alt="Every Crazy Thing JavaScript Does"