Layout

Layout comparisons

24 old vs modern layout CSS techniques, side by side.

Browser compatibility:
Layout
Advanced

Path shapes without SVG clip paths

Old clip-path: path("M 0 0 L 800 0...");
/* px values — not responsive */
Modern clip-path: shape(
  from 0% 0%, line to 100% 0%,
  curve to 0% 80% via 50% 105%);
see modern →
Layout
Beginner

Scaling elements without transform hacks

Old transform: scale(0.5);
margin-bottom: -50%; /* hack */
Modern .thumb { zoom: 0.5; }
see modern →
Layout
Beginner

Preventing layout shift from scrollbar appearance

Old body { overflow-y: scroll; }
/* or hardcode the scrollbar width */
body { padding-right: 17px; }
Modern body {
  scrollbar-gutter: stable;
}
/* scrollbar space always reserved */
see modern →
Layout
Beginner

Media query ranges without min-width and max-width

Old @media (min-width: 600px)
         and (max-width: 1200px) {
  /* styles */
}
Modern @media (600px <= width <= 1200px) {
  /* styles */
}
see modern →
Layout
Beginner

Preventing scroll chaining without JavaScript

Old // JS: block page scroll when inside modal
modal.addEventListener('wheel', e =>
  e.preventDefault(), { passive: false })
Modern .modal-content {
  overflow-y: auto;
  overscroll-behavior: contain;
}
/* page stays still */
see modern →
Layout
Beginner

Responsive images without the background-image hack

Old .card-image {
  background-image: url(...);
  background-size: cover;
  background-position: center;
}
Modern img {
  object-fit: cover;
  width: 100%;
  height: 200px;
}
see modern →
Layout
Beginner

Scrollbar styling without -webkit- pseudo-elements

Old /* webkit only */
::-webkit-scrollbar { width: 8px; }
::-webkit-scrollbar-thumb { background: #888; }
Modern * {
  scrollbar-width: thin;
  scrollbar-color: #888 transparent;
}
see modern →
Layout
Beginner

Mobile viewport height without the 100vh hack

Old .hero {
  height: 100vh;
}
/* overflows on mobile */
Modern .hero {
  height: 100dvh;
}
/* adapts to browser chrome */
see modern →
Layout
Beginner

Auto-growing textarea without JavaScript

Old // JS: resize on every keystroke
el.addEventListener('input', () => {
  el.style.height = 'auto';
  el.style.height = el.scrollHeight + 'px'; })
Modern textarea {
  field-sizing: content;
  min-height: 3lh;
}
/* grows with content, no JS */
see modern →
Layout
Beginner

Corner shapes beyond rounded borders

Old .card {
  clip-path: polygon(
    ... /* 20+ points */
  );
}
Modern .card {
  border-radius: 2em;
  corner-shape: squircle;
}
see modern →
Layout
Beginner

Filling available space without calc workarounds

Old .full {
  width: calc(100% - 40px);
  /* or width: 100% and overflow */
}
Modern .full {
  width: stretch;
}
/* fills container, keeps margins */
see modern →
Layout
Advanced

Carousel navigation without a JavaScript library

Old // Swiper.js or Slick carousel
new Swiper('.carousel', {
  navigation: { /* … */ },
  pagination: { /* … */ },
});
Modern .carousel::scroll-button(right) {
  content: "➡";
}
.carousel li::scroll-marker {
  content: '';
}
see modern →
Layout
Intermediate

Hover tooltips without JavaScript events

Old // JS: mouseenter + mouseleave
btn.addEventListener('mouseenter',
  () => showTooltip())
/* + focus, blur, positioning */
Modern <button interestfor="tip">Hover me</button>
<div id="tip" popover=hint>
  Tooltip content
</div>
see modern →
Layout
Beginner

Positioning shorthand without four properties

Old .overlay {
  top: 0; right: 0;
  bottom: 0; left: 0;
}
Modern .overlay {
  position: absolute;
  inset: 0;
}
see modern →
Layout
Advanced

Tooltip positioning without JavaScript

Old /* Popper.js / Floating UI: compute rect,
position: fixed, update on scroll */
.tooltip { position: fixed; }
Modern .trigger { anchor-name: --tip; }
.tooltip {
  position-anchor: --tip;
  top: anchor(bottom);
}
see modern →
Layout
Intermediate

Scroll snapping without a carousel library

Old // Slick, Swiper, or scroll/touch JS
$('.carousel').slick({ … })
touchstart / scroll handlers
Modern .carousel { scroll-snap-type: x mandatory; }
.carousel > * { scroll-snap-align: start; }
/* no lib, no touch handlers */
see modern →
Layout
Intermediate

Direction-aware layouts without left and right

Old margin-left: 1rem;
padding-right: 1rem;
[dir="rtl"] .box { margin-right: ... }
Modern margin-inline-start: 1rem;
padding-inline-end: 1rem;
border-block-start: 1px solid;
see modern →
Layout
Beginner

Naming grid areas without line numbers

Old float: left; /* clearfix, margins */
grid-column: 1 / 3;
grid-row: 2;
Modern .layout {
  display: grid;
  grid-template-areas: "header header" "sidebar main" "footer footer";
}
see modern →
Layout
Advanced

Aligning nested grids without duplicating tracks

Old .child-grid {
  grid-template-columns: 1fr 1fr 1fr;
/* duplicate parent tracks */
}
Modern .child-grid {
  display: grid;
  grid-template-columns: subgrid;
}
see modern →
Layout
Beginner

Spacing elements without margin hacks

Old .grid > * { margin-right: 16px; }
.grid > *:last-child { margin-right: 0; }
Modern .grid {
  display: flex;
  gap: 16px;
}
see modern →
Layout
Beginner

Aspect ratios without the padding hack

Old .wrapper { padding-top: 56.25%; position: relative; }
.inner { position: absolute; inset: 0; }
Modern .video-wrapper {
  aspect-ratio: 16 / 9;
}
see modern →
Layout
Beginner

Sticky headers without JavaScript scroll listeners

Old // JS: scroll listener + getBoundingClientRect
// then add/remove .fixed class
.header.fixed { position: fixed; }
Modern .header {
  position: sticky;
  top: 0;
}
see modern →
Layout
Intermediate

Responsive components without media queries

Old @media (max-width: 768px) {
  .card { … }
}
/* viewport, not container */
Modern @container (width < 400px) {
  .card { flex-direction: column; }
}
see modern →
Layout
Beginner

Centering elements without the transform hack

Old position: absolute;
top: 50%; left: 50%;
transform: translate(-50%,-50%);
Modern .parent {
  display: grid;
  place-items: center;
}
see modern →

Other categories

New CSS drops.

Join 600+ readers who've survived clearfix hacks.

ESC