If you’ve been in the web development game for longer, you might recall the days when CSS was utterly confusing and you had to come up with hacks and workarounds to make things work. Luckily, these days are over and new features such as container queries, cascade layers, CSS nesting, the :has selector, grid and subgrid, and even new color spaces make CSS more powerful than ever before.
With all these lovely new CSS features on the horizon, in this post, we dive into the world of CSS with a few helpful techniques, a deep-dive into specificity, hanging punctuation, and self-modifying CSS variables. We hope they’ll come in handy in your work.
Cascade And Specificity Primer
Many fear the cascade and specificity in CSS. However, the concept isn’t as hard to get to grips with as one might think. To help you get more comfortable with two of the most fundamental parts of CSS, Andy Bell wrote a wonderful primer on the cascade and specificity.
The guide explains how certain CSS property types will be prioritized over others and dives deeper into specificity scoring to help you assess how likely it is that the CSS of a specific rule will apply. Andy uses practical examples to illustrate the concepts and simplifies the underlying mental model to make it easy to adopt and utilize. A power boost for your CSS skills.
Testing HTML With Modern CSS
Have you ever considered testing HTML with CSS instead of JavaScript? CSS selectors today are so powerful that it is actually possible to test for most kinds of HTML patterns using CSS alone. A proponent of the practice, Heydon Pickering summarized everything you need to know about testing HTML with CSS, whether you want to test accessibility, uncover HTML bloat, or check the general usability.
As Heydon points out, testing with CSS has quite some benefits. Particularly if you work in the browser and prefer exploring visual regressions and inspector information over command line logs, testing with CSS could be for you. It also shines in situations where you don’t have direct access to a client’s stack: Just provide a test stylesheet, and clients can locate instances of bad patterns you have identified for them without having to onboard you to help them do so. Clever!
Self-Modifying CSS Variables
The CSS spec for custom properties does not allow a custom property to reference itself — although there are quite some use cases where such a feature would be useful. To close the gap, Lea Verou proposed an inherit() function in 2018, which the CSSWG added to the specs in 2021. It hasn’t been edited-in yet, but Roman Komarov found a workaround that makes it possible to start involving its behavior.
Roman’s approach uses container-style queries as a way to access the previous state of a custom property. It can be useful when you want to cycle through various hues without having a static list of values, to match the border-radius visually, or to nest menu lists, for example. The workaround is still strictly experimental (so do not use it in production!), but since it is likely that style queries will gain broad browser support before inherit(), it has great potential.
Hanging Punctuation In CSS
hanging-punctuation is a neat little CSS property. It extends punctuation marks such as opening quotes to cater to nice, clean blocks of text. And while it’s currently only supported in Safari, it doesn’t hurt to include it in your code, as the property is a perfect example of progressive enhancement: It leaves things as they are in browsers that don’t support it and adds the extra bit of polish in browsers that do.
Jeremy Keith shares a little gotcha to help you fix an unintended side effect of hanging punctuation. (Large preview) (Image credit: MDN Web Docs)
Jeremy Keith noticed an unintended side-effect of hanging-punctuation, though. When you apply it globally, it’s also applied to form fields. So, if the text in a form field starts with a quotation mark or some other piece of punctuation, it’s pushed outside the field and hidden. Jeremy shares a fix for it: Add input, textarea { hanging-punctuation: none; } to prevent your quotation marks from disappearing. A small tip that can save you a lot of headaches.
Fixing aspect-ratio Issues
The aspect-ratio property shines in fluid environments. It can handle anything from inserting a square-shaped <div> to matching the 16:9 size of a <video>, without you thinking in exact dimensions. And most of the time, it does so flawlessly. However, there are some things that can break aspect-ratio. Chris Coyier takes a closer look at three reasons why your aspect-ratio might not work as expected.
As Chris explains, one potential breakage is setting both dimensions — which might seem obvious, but it can be confusing if one of the dimensions is set from somewhere you didn’t expect. Stretching and content that forces height can also lead to unexpected results. A great overview of what to look out for when aspect-ratio breaks.
Masonry Layout With CSS
CSS Grid has taken layouts on the web to the next level. However, as powerful as CSS is today, not every layout that can be imagined is feasible. Masonry layout is one of those things that can’t be accomplished with CSS alone. To change that, the CSS Working Group is asking for your help.
There are currently two approaches in discussion at the CSS Working Group about how CSS should handle masonry-style layouts — and they are asking for insights from real-world developers and designers to find the best solution.
The first approach would expand CSS Grid to include masonry, and the second approach would be to introduce a masonry layout as a display: masonry display type. Jen Simmons summarized what you need to know about the ongoing debate and how you can contribute your thoughts on which direction CSS should take.
Before you come to a conclusion, also be sure to read Rachel Andrew’s post on the topic. She explains why the Chrome team has concerns about implementing a masonry layout as a part of the CSS Grid specification and clarifies what the alternate proposal enables.
Boost Your CSS Skills
If you’d like to dive deeper into CSS, we’ve got your back — with a few friendly events and SmashingConfs coming up this year:
We’d be absolutely delighted to welcome you to one of our special Smashing experiences — be it online or in person!
Smashing Weekly Newsletter
With our weekly newsletter, we aim to bring you useful, practical tidbits and share some of the helpful things that folks are working on in the web industry. There are so many talented folks out there working on brilliant projects, and we’d appreciate it if you could help spread the word and give them the credit they deserve!
Also, by subscribing, there are no third-party mailings or hidden advertising, and your support really helps us pay the bills. ❤️
Interested in sponsoring? Feel free to check out our partnership options and get in touch with the team anytime — they’ll be sure to get back to you as soon as they can.
Lately, I’ve been thinking about the basics of web development. Actually, I’ve been thinking about them for some time now, at least since I started teaching beginning web development in 2020.
I’m fascinated by the basics. They’re an unsung hero, really, as there is no developer worth their salt who would be where they are without them. Yet, they often go unnoticed.
The basics exist in some sort of tension between the utmost importance and the incredibly banal.
You might even think of them as the vegetable side on your dinner plate — wholesome but perhaps bland without the right seasoning.
Who needs the basics of HTML and CSS, some say, when we have tools that abstract the way they’re written and managed? We now have site builders that require no technical knowledge. We have frameworks with enough syntactic sugar to give your development chops a case of cavities. We have libraries packed with any number of pre-established patterns that can be copy-pasted without breaking a sweat. The need to “learn” the basics of HTML and CSS is effectively null when the number of tools that exist to supplant them is enough to fill a small galaxy of stars.
Rachel Andrew wrote one of my all-time favorite posts back in 2019, equating the rise of abstractions with an increase in complexity and a profound loss of inroads for others to enter the web development field:
“We have already lost many of the entry points that we had. We don’t have the forums of parents teaching each other HTML and CSS, in order to make a family album. Those people now use Facebook or perhaps run a blog on wordpress.com or SquareSpace with a standard template. We don’t have people customising their MySpace profile or learning HTML via Neopets. We don’t have the people, usually women, entering the industry because they needed to learn HTML during that period when an organisation’s website was deemed part of the duties of the administrator.”
There’s no moment more profound in my web development career than the time I changed the background color of a page from default white to some color value I can’t remember (but know for a fact it would never be dodgerblue). That, and my personal “a-ha!” moment when realizing that everything in CSS is a box. Nothing guided me with the exception of “View Source,” and I’d bet the melting Chapstick in my pocket that you’re the same if you came up around the turn of the 21st century.
Where do you go to learn HTML and CSS these days? Even now, there are few dedicated secondary education programs (or scholarships, for that matter) to consider. We didn’t have bootcamps back in the day, but you don’t have to toss a virtual stone across many pixels to find one today.
There are excellent and/or free tutorials, too. Here, I’ll link a few of ’em up for you:
Anyway, my point is that there are more resources than ever for learning web development, but still painfully few entry points to get there. The resources we have for learning the basics are great, but many are either growing stale, are quick hits without a clear learning path, or assume the learner has at least some technical knowledge. I can tell you, as someone who has hit the Publish button on thousands of front-end tutorials, that the vast majority — if not all — of them are geared toward those who are already on the career path.
It was always a bit painful when someone would email CSS-Tricks asking where to get started learning CSS because, well, you’d imagine CSS-Tricks being the perfect home for something like that, and yet, there’s nothing. It’s just the reality, even if many of us (myself included) cut our chops with sites like CSS-Tricks, Smashing Magazine, and A List Apart. We were all learning together at that time, or so it seemed.
What we need are more pathways for deep learning.
Learning Experience Design (LXD) is a real thing that I’d position somewhere between what we know as UX Design and the practice of accessibility. There’s a focus on creating delightful experiences, sure, but the real aim of LDX is to establish learning paths that universally account for different types of learners (e.g., adults and children) and learning styles (e.g., visual and experiential). According to LDX, learners have a set of needs not totally unlike those that Maslow’s hierarchy of needs identifies for all humans, and there are different models for determining those needs, perhaps none more influential than Bloom’s Taxonomy.
These are things that many front-end tutorials, bootcamps, videos, and programs are not designed for. It’s not that the resources are bad (nay, most are excellent); it’s that they are serving different learners and learning types than what a day-one beginner needs. And let’s please not rely on AI to fill the gaps in human experiences!
Like I said, I’ve been thinking about this a lot. Like, a lot a lot. In fact, I recently published an online course purely dedicated to learning the basics of front-end development, creatively named TheBasics.dev. I’d like to think it’s not just another tutorial because it’s a complete set of lessons that includes reading, demonstrations, videos, lab exercises, and assessments, i.e., a myriad of ways to learn. I’d also like to think that this is more than just another bootcamp because it is curricula designed with the intention to develop new knowledge through reflective practices, peer learning, and feedback.
Anyway, I’m darn proud of The Basics, even if I’m not exactly the self-promoting type, and writing about it is outside of my comfort zone. If you’re reading this, it’s very likely that you, too, work on the front end. The Basics isn’t for you exactly, though I’d argue that brushing up on fundamentals is never a bad thing, regardless of your profession, but especially in front-end development, where standards are well-documented but ever-changing as well.
The Basics is more for your clients who do not know how to update the website they paid you to make. Or the friend who’s learning but still keeps bugging you with questions about the things they’re reading. Or your mom, who still has no idea what it is you do for a living. It’s for those whom the entry points are vanishing. It’s for those who could simply sign up for a Squarespace account but want to actually understand the code it spits out so they have more control to make a site that uniquely reflects them.
If you know a person like that, I would love it if you’d share The Basics with them.
Long live the basics! Long live the “a-ha!” moments that help us all fall in love with the World Wide Web.
Establishing layouts in CSS is something that we, as developers, often delegate to whatever framework we’re most comfortable using. And even though it’s possible to configure a framework to get just what we need out of it, how often have you integrated an entire CSS library simply for its layout features? I’m sure many of us have done it at some point, dating back to the days of 960.gs, Bootstrap, Susy, and Foundation.
Modern CSS features have significantly cut the need to reach for a framework simply for its layout. Yet, I continue to see it happen. Or, I empathize with many of my colleagues who find themselves re-creating the same Grid or Flexbox layout over and over again.
In this article, we will gain greater control over web layouts. Specifically, we will create four CSS classes that you will be able to take and use immediately on just about any project or place where you need a particular layout that can be configured to your needs.
While the concepts we cover are key, the real thing I want you to take away from this is the confidence to use CSS for those things we tend to avoid doing ourselves. Layouts used to be a challenge on the same level of styling form controls. Certain creative layouts may still be difficult to pull off, but the way CSS is designed today solves the burdens of the established layout patterns we’ve been outsourcing and re-creating for many years.
What We’re Making
We’re going to establish four CSS classes, each with a different layout approach. The idea is that if you need, say, a fluid layout based on Flexbox, you have it ready. The same goes for the three other classes we’re making.
And what exactly are these classes? Two of them are Flexbox layouts, and the other two are Grid layouts, each for a specific purpose. We’ll even extend the Grid layouts to leverage CSS Subgrid for when that’s needed.
Within those two groups of Flexbox and Grid layouts are two utility classes: one that auto-fills the available space — we’re calling these “fluid” layouts — and another where we have greater control over the columns and rows — we’re calling these “repeating” layouts.
Finally, we’ll integrate CSS Container Queries so that these layouts respond to their own size for responsive behavior rather than the size of the viewport. Where we’ll start, though, is organizing our work into Cascade Layers, which further allow you to control the level of specificity and prevent style conflicts with your own CSS.
Setup: Cascade Layers & CSS Variables
A technique that I’ve used a few times is to define Cascade Layers at the start of a stylesheet. I like this idea not only because it keeps styles neat and organized but also because we can influence the specificity of the styles in each layer by organizing the layers in a specific order. All of this makes the utility classes we’re making easier to maintain and integrate into your own work without running into specificity battles.
I think the following three layers are enough for this work:
@layer reset, theme, layout;
Notice the order because it really, really matters. The reset layer comes first, making it the least specific layer of the bunch. The layout layer comes in at the end, making it the most specific set of styles, giving them higher priority than the styles in the other two layers. If we add an unlayered style, that one would be added last and thus have the highest specificity.
Figure 1: Inspecting Cascade Layers in Chrome’s DevTools. (Large preview)
Let’s briefly cover how we’ll use each layer in our work.
Reset Layer
The reset layer will contain styles for any user agent styles we want to “reset”. You can add your own resets here, or if you already have a reset in your project, you can safely move on without this particular layer. However, do remember that un-layered styles will be read last, so wrap them in this layer if needed.
I’m just going to drop in the popular box-sizing declaration that ensures all elements are sized consistently by the border-box in accordance with the CSS Box Model.
This layer provides variables scoped to the :root element. I like the idea of scoping variables this high up the chain because layout containers — like the utility classes we’re creating — are often wrappers around lots of other elements, and a global scope ensures that the variables are available anywhere we need them. That said, it is possible to scope these locally to another element if you need to.
Now, whatever makes for “good” default values for the variables will absolutely depend on the project. I’m going to set these with particular values, but do not assume for a moment that you have to stick with them — this is very much a configurable system that you can adapt to your needs.
Here are the only three variables we need for all four layouts:
Notice: The variables are prefixed with layout-, which I’m using as an identifier for layout-specific values. This is my personal preference for structuring this work, but please choose a naming convention that fits your mental model — naming things can be hard!
Layout Layer
This layer will hold our utility class rulesets, which is where all the magic happens. For the grid, we will include a fifth class specifically for using CSS Subgrid within a grid container for those possible use cases.
Now that all our layers are organized, variables are set, and rulesets are defined, we can begin working on the layouts themselves. We will start with the “repeating” layouts, one based on CSS Grid and the other using Flexbox.
Repeating Grid And Flex Layouts
I think it’s a good idea to start with the “simplest” layout and scale up the complexity from there. So, we’ll tackle the “Repeating Grid” layout first as an introduction to the overarching technique we will be using for the other layouts.
Repeating Grid
If we head into the @layout layer, that’s where we’ll find the .repeating-grid ruleset, where we’ll write the styles for this specific layout. Essentially, we are setting this up as a grid container and applying the variables we created to it to establish layout columns and spacing between them.
It’s not too complicated so far, right? We now have a grid container with three equally sized columns that take up one fraction (1fr) of the available space with a gap between them.
This is all fine and dandy, but we do want to take this a step further and turn this into a system where you can configure the number of columns and the size of the gap. I’m going to introduce two new variables scoped to this grid:
--_grid-repeat: The number of grid columns.
--_repeating-grid-gap: The amount of space between grid items.
Did you notice that I’ve prefixed these variables with an underscore? This was actually a JavaScript convention to specify variables that are “private” — or locally-scoped — before we had const and let to help with that. Feel free to rename these however you see fit, but I wanted to note that up-front in case you’re wondering why the underscore is there.
Notice: These variables are set to the variables in the @theme layer. I like the idea of assigning a global variable to a locally-scoped variable. This way, we get to leverage the default values we set in @theme but can easily override them without interfering anywhere else the global variables are used.
Now let’s put those variables to use on the style rules from before in the same .repeating-grid ruleset:
What happens from here when we apply the .repeating-grid to an element in HTML? Let’s imagine that we are working with the following simplified markup:
If we were to apply a background-color and height to those divs, we would get a nice set of boxes that are placed into three equally-sized columns, where any divs that do not fit on the first row automatically wrap to the next row.
See the Pen [Layout Utility: Repeating Grid [forked]](https://codepen.io/smashingmag/pen/gOJrqmL) by Geoff Graham.
Now, of course, we don’t have to have just three columns. Let’s say we want a product grid where we want to change the repeating columns from 3 to 5 while updating the gap from 2vw to 3vw using the same HTML, only with a new class we can use override those values.
See how this is shaping up? We have a grid layout based on a set of globally-scoped variables that we can re-assign to variables that are locally-scoped to the utility class and further customized with a class of our own that adds context to the element’s purpose and allows you to adjust the responsive behavior.
The benefit is that we can overwrite our default values without polluting the HTML with superfluous classes. This is the overarching approach we will also use in the three other layout classes. Next up is the “Repeating Flex” version of what we just made.
Repeating Flex
The “Repeating Grid” layout is great, but you might not always want equally-sized columns. CSS Grid is certainly capable of auto-filling elements with whatever space is available, but Flexbox is extremely proficient at it.
Let’s say we have the same five divs from before. That leaves us with two divs on the second row next to an empty column on the right. Perhaps we want those last two leftover divs to stretch out and take up the space in the empty column.
Figure 2: Flexible items automatically fill any remaining space that would otherwise be presented as an empty third column on the second row. (Large preview)
Time to put the process we established with the Repeating Grid layout to use in this Repeating Flex layout. This time, we jump straight to defining the private variables on the .repeating-flex ruleset in the @layout layer since we already know what we’re doing.
Again, we have two locally-scoped variables used to override the default values assigned to the globally-scoped variables. Now, we apply them to the style declarations.
We’re only using one of the variables to set the gap size between flex items at the moment, but that will change in a bit. For now, the important thing to note is that we are using the flex-wrap property to tell Flexbox that it’s OK to let additional items in the layout wrap into multiple rows rather than trying to pack everything in a single row.
But once we do that, we also have to configure how the flex items shrink or expand based on whatever amount of available space is remaining. Let’s nest those styles inside the parent ruleset:
If you’re wondering why I’m using the universal selector (*), it’s because we can’t assume that the layout items will always be divs. Perhaps they are <article> elements, <section>s, or something else entirely. The child combinator (>) ensures that we’re only selecting elements that are direct children of the utility class to prevent leakage into other ancestor styles.
The flex shorthand property is one of those that’s been around for many years now but still seems to mystify many of us. Before we unpack it, did you also notice that we have a new locally-scoped --_gap-repeater-calc variable that needs to be defined? Let’s do this:
Whoa, we actually created a second variable that --_gap-repeater-calc can use to properly calculate the third flex value, which corresponds to the flex-basis property, i.e., the “ideal” size we want the flex items to be.
If we take out the variable abstractions from our code above, then this is what we’re looking at:
Hopefully, this will help you see what sort of math the browser has to do to size the flexible items in the layout. Of course, those values change if the variables’ values change. But, in short, elements that are direct children of the .repeating-flex utility class are allowed to grow (flex-grow: 1) and shrink (flex-shrink: 1) based on the amount of available space while we inform the browser that the initial size (i.e., flex-basis) of each flex item is equal to some calc()-ulated value.
Because we had to introduce a couple of new variables to get here, I’d like to at least explain what they do:
--_gap-count: This stores the number of gaps between layout items by subtracting 1 from --_flex-repeat. There’s one less gap in the number of items because there’s no gap before the first item or after the last item.
--_gap-repeater-calc: This calculates the total gap size based on the individual item’s gap size and the total number of gaps between items.
From there, we calculate the total gap size more efficiently with the following formula:
Let’s break that down further because it’s an inception of variables referencing other variables. In this example, we already provided our repeat-counting private variable, which falls back to the default repeater by setting the --layout-default-repeat variable.
This sets a gap, but we’re not done yet because, with flexible containers, we need to define the flex behavior of the container’s direct children so that they grow (flex-grow: 1), shrink (flex-shrink: 1), and with a flex-basis value that is calculated by multiplying the repeater by the total number of gaps between items.
Next, we divide the individual gap size (--_repeating-flex-gap) by the number of repetitions (--_flex-repeat)) to equally distribute the gap size between each item in the layout. Then, we multiply that gap size value by one minus the total number of gaps with the --_gap-count variable.
And that concludes our repeating grids! Pretty fun, or at least interesting, right? I like a bit of math.
Before we move to the final two layout utility classes we’re making, you might be wondering why we want so many abstractions of the same variable, as we start with one globally-scoped variable referenced by a locally-scoped variable which, in turn, can be referenced and overridden again by yet another variable that is locally scoped to another ruleset. We could simply work with the global variable the whole time, but I’ve taken us through the extra steps of abstraction.
I like it this way because of the following:
I can peek at the HTML and instantly see which layout approach is in use: .repeating-grid or .repeating-flex.
It maintains a certain separation of concerns that keeps styles in order without running into specificity conflicts.
This gives me all of the context I need: the type of layout, what it is used for, and where to find the variables. I think that’s handy, but you certainly could get by without the added abstractions if you’re looking to streamline things a bit.
Fluid Grid And Flex Layouts
All the repeating we’ve done until now is fun, and we can manipulate the number of repeats with container queries and media queries. But rather than repeating columns manually, let’s make the browser do the work for us with fluid layouts that automatically fill whatever empty space is available in the layout container. We may sacrifice a small amount of control with these two utilities, but we get to leverage the browser’s ability to “intelligently” place layout items with a few CSS hints.
Fluid Grid
Once again, we’re starting with the variables and working our way to the calculations and style rules. Specifically, we’re defining a variable called --_fluid-grid-min that manages a column’s minimum width.
Let’s take a rather trivial example and say we want a grid column that’s at least 400px wide with a 20px gap. In this situation, we’re essentially working with a two-column grid when the container is greater than 820px wide. If the container is narrower than 820px, the column stretches out to the container’s full width.
If we want to go for a three-column grid instead, the container’s width should be about 1240px wide. It’s all about controlling the minimum sizing values in the gap.
The display is set to grid, and the gap between items is based on the --fluid-grid-gap variable. The magic is taking place in the grid-template-columns declaration.
This grid uses the repeat() function just as the .repeating-grid utility does. By declaring auto-fit in the function, the browser automatically packs in as many columns as it possibly can in the amount of available space in the layout container. Any columns that can’t fit on a line simply wrap to the next line and occupy the full space that is available there.
Then there’s the minmax() function for setting the minimum and maximum width of the columns. What’s special here is that we’re nesting yet another function, min(), within minmax() (which, remember, is nested in the repeat() function). This a bit of extra logic that sets the minimum width value of each column somewhere in a range between --_fluid-grid-min and 100%, where 100% is a fallback for when --_fluid-grid-min is undefined or is less than 100%. In other words, each column is at least the full 100% width of the grid container.
The “max” half of minmax() is set to 1fr to ensure that each column grows proportionally and maintains equally sized columns.
See the Pen [Fluid grid [forked]](https://codepen.io/smashingmag/pen/GRaZzMN) by utilitybend.
That’s it for the Fluid Grid layout! That said, please do take note that this is a strong grid, particularly when it is combined with modern relative units, e.g. ch, as it produces a grid that only scales from one column to multiple columns based on the size of the content.
Fluid Flex
We pretty much get to re-use all of the code we wrote for the Repeating Flex layout for the Fluid Flex layout, but only we’re setting the flex-basis of each column by its minimum size rather than the number of columns.
That completes the fourth and final layout utility — but there’s one bonus class we can create to use together with the Repeating Grid and Fluid Grid utilities for even more control over each layout.
Optional: Subgrid Utility
Subgrid is handy because it turns any grid item into a grid container of its own that shares the parent container’s track sizing to keep the two containers aligned without having to redefine tracks by hand. It’s got full browser support and makes our layout system just that much more robust. That’s why we can set it up as a utility to use with the Repeating Grid and Fluid Grid layouts if we need any of the layout items to be grid containers for laying out any child elements they contain.
This works like a charm since the variable informs the subgrid how much it can grow.
Method 2: Using The :has() Pseudo-Class
This approach leads to verbose CSS, but sacrificing brevity allows us to automate the layout so it handles practically anything we throw at it without having to update an inline style in the markup.
The :has() selector checks if a subgrid row is the last child item in the container when that item is either the first, second, third, fourth, fifth, and so on item. For example, the second declaration:
…is pretty much saying, “If this is the second subgrid item and it happens to be the last item in the container, then set the number of rows to 2.”
Whether this is too heavy-handed, I don’t know; but I love that we’re able to do it in CSS.
The final missing piece is to declare a container on our children. Let’s give the columns a general class name, .grid-item, that we can override if we need to while setting each one as a container we can query for the sake of updating its layout when it is a certain size (as opposed to responding to the viewport’s size in a media query).
That’s a wild-looking selector, but the verbosity is certainly kept to a minimum thanks to the :is() pseudo-class, which saves us from having to write this as a larger chain selector. It essentially selects the direct children of the other utilities without leaking into .subgrid-rows and inadvertently selecting its direct children.
The container property is a shorthand that combines container-name and container-type into a single declaration separated by a forward slash (/). The name of the container is set to one of our variables, and the type is always its inline-size (i.e., width in a horizontal writing mode).
The container-type property can only be applied to grid containers — not grid items. This means we’re unable to combine it with the grid-template-rows: subgrid value, which is why we needed to write a more complex selector to exclude those instances.
Demo
Check out the following demo to see how everything comes together.
See the Pen [Grid system playground [forked]](https://codepen.io/smashingmag/pen/mdYPvLR) by utilitybend.
The demo is pulling in styles from another pen that contains the full CSS for everything we made together in this article. So, if you were to replace the .fluid-flex classname from the parent container in the HTML with another one of the layout utilities, the layout will update accordingly, allowing you to compare them.
Those classes are the following:
.repeating-grid,
.repeating-flex,
.fluid-grid,
.fluid-flex.
And, of course, you have the option of turning any grid items into grid containers using the optional .subgrid-rows class in combination with the .repeating-grid and .fluid-grid utilities.
Conclusion: Write Once And Repurpose
This was quite a journey, wasn’t it? It might seem like a lot of information, but we made something that we only need to write once and can use practically anywhere we need a certain type of layout using modern CSS approaches. I strongly believe these utilities can not only help you in a bunch of your work but also cut any reliance on CSS frameworks that you may be using simply for its layout configurations.
This is a combination of many techniques I’ve seen, one of them being a presentation Stephanie Eckles gave at CSS Day 2023. I love it when people handcraft modern CSS solutions for things we used to work around. Stephanie’s demonstration was clean from the start, which is refreshing as so many other areas of web development are becoming ever more complex.
After learning a bunch from CSS Day 2023, I played with Subgrid on my own and published different ideas from my experiments. That’s all it took for me to realize how extensible modern CSS layout approaches are and inspired me to create a set of utilities I could rely on, perhaps for a long time.
By no means am I trying to convince you or anyone else that these utilities are perfect and should be used everywhere or even that they’re better than <framework-du-jour>. One thing that I do know for certain is that by experimenting with the ideas we covered in this article, you will get a solid feel of how CSS is capable of making layout work much more convenient and robust than ever.
Create something out of this, and share it in the comments if you’re willing — I’m looking forward to seeing some fresh ideas!
Media queries have been around almost as long as CSS itself — and with no flex, no grid, no responsive units, and no math functions, media queries were the most pragmatic choice available to make a somewhat responsive website.
In the early 2010s, with the proliferation of mobile devices and the timely publication of Ethan Marcotte’s classic article “Responsive Web Design”, media queries became much needed for crafting layouts that could morph across screens and devices. Even when the CSS Flexbox and Grid specifications rolled out, media queries for resizing never left.
While data on the actual usage of media queries is elusive, the fact that they have grown over time with additional features that go well beyond the viewport and into things like user preferences continues to make them a bellwether ingredient for responsive design.
Today, there are more options and tools in CSS for establishing layouts that allow page elements to adapt to many different conditions besides the size of the viewport. Some are more widely used — Flexbox and Grid for certain — but also things like responsive length units and, most notably, container queries, a concept we will come back to in a bit.
But media queries are still often the de facto tool that developers reach for. Maybe it’s muscle memory, inconsistent browser support, or that we’re stuck in our ways, but adoption of the modern approaches we have for responsive interfaces seems slow to take off.
To be clear, I am all for media queries. They play a significant role in the work we do above and beyond watching the viewport size to make better and more accessible user experiences based on a user’s OS preferences, the type of input device they’re using, and more.
But should media queries continue to be the gold standard for responsive layouts? As always, it depends, but
It is undeniable that media queries have evolved toward accessibility solutions, making space for other CSS features to take responsibility for responsiveness.
“
The Problem With Media Queries
Media queries seemed like a great solution for most responsive-related problems, but as the web has grown towards bigger and more complex layouts, the limits of media queries are more prevalent than ever.
Problem #1: They Are Viewport-Focused
When writing media query breakpoints where we want the layout to adapt, we only have access to the viewport’s properties, like width or orientation. Sometimes, all we need is to tweak a font size, and the viewport is our best bud for that, but most times, context is important.
Components on a page share space with others and are positioned relative to each other according to normal document flow. If all we have access to is the viewport width, knowing exactly where to establish a particular breakpoint becomes a task of compromises where some components will respond well to the adapted layout while others will need additional adjustments at that specific breakpoint.
So, there we are, resizing our browser and looking for the correct breakpoint where our content becomes too squished.
The following example probably has the worst CSS you will see in a while, but it helps to understand one of the problems with media queries.
See the Pen [Old Media Queries [forked]](https://codepen.io/smashingmag/pen/xxNwbob) by Monknow.
It’s a pretty embarrassing example, but why exactly is it bad?
If we try to convert the CSS verbatim, it would say, When the page width is smaller than 600px, these elements will grow and collapse. This statement is completely agnostic to the element’s contents or its siblings. What happens if there are more siblings or just one? Or what happens if the element is inside a smaller container? The media query simply lacks the information we need to account for these things, which leads us to a second problem with media queries.
Problem #2: They Are Difficult To Manage
Nowadays everything is a component. Like nomads, components wander from place to place, sharing space with other components and bringing along their ever-changing content. Again, media queries are completely unaware of the context surrounding a component, and the burden lies on the developer to find that sweet spot for each and every case.
This is cumbersome work because the ideal breakpoint in a media query is kind of a hardcoded _magic number that we find by resizing our page, and also because it will differ depending on the context surrounding each component, necessitating multiple media queries. If we want to change an element’s content or container, then the media query needs to change, too.
And where do you manage media queries in your stylesheet? Some developers will plop them in at the end, while others may manage them in partial files and rely on a preprocessor to compile them.
The recent CSS Nesting feature makes things cleaner now that we can attach a media query to a component in the same style rule, but now we have to do that for each and every component in the system, which makes for more and more instances to account for when editing styles. This leads to the next problem with media queries as a responsive silver bullet.
Problem #3: They Aren’t That Responsive
Most times, we want an element to resize itself fluidly with the screen, but writing a new media query each time a component “breaks” at a specific breakpoint is a lot to manage. If a component becomes too tall on a narrow screen when the viewport is between 960px and 970px, do we really need to write a new set of styles that we now have to look after?
I wouldn’t exactly call that responsive design but rather some form of adaptive design based on hard numbers for a super-specific situation. There’s no fluidity in that.
You Don’t Need Media Queries (For Resizing)
Fortunately, we’re no longer living in 2012, and there are far better options than media queries, most of them largely adopted and widely supported, led by the likes of Flexbox and Grid, responsive units and math functions, while others like container queries are on the cusp, but still in relatively early days.
I am not going to act like I discovered hot water by pointing out that these modern CSS features exist and are now commonplace tools used by nearly every CSS developer. However, media queries still find their way into CSS for resizing, particularly in situations where a clamp() function or a bit of creative thinking with responsive units would not only accomplish the task but do it better than a media query because they are designed for this purpose.
So rather than me trying to teach all these not-entirely-new CSS features (you are awesome, and I am confident you are already aware of them), my focus is on swapping your existing media queries for modern responsive techniques.
Flexbox
Flexbox and media queries are often paired together so that Flexbox establishes a layout in a certain direction, and media queries are used to change direction at certain viewport widths.
See the Pen [Using Media Queries with Flex Items [forked]](https://codepen.io/smashingmag/pen/MWdaYNx) by Monknow.
This very simple — but common — pattern runs into each of the three problems we discussed earlier:
It’s viewport-focused.
We are only considering the viewport width when choosing where the container changes direction. I found the magic breakpoint number is 700px after testing, so that is where we would need to establish a new media query.
It’s hard to reuse and manage.
Since we are only considering the viewport width, the element can’t be used inside a smaller container or may look awkward if the element has different content within it or around it.
It isn’t that responsive.
We have a breakpoint at 700px, so devices with narrow screens beyond the threshold may squish the content too much while others get the optimal experience.
If we try to fix it by adding more media queries, we’ll be back at Problem #2.
The best solution for this case is to avoid media queries altogether. I would replace them with the flex shorthand property that allows the <article> elements to grow and shrink based on the available space up to a certain point that’s set to 400px.
main {
display: flex;
flex-flow: row wrap;
}
main article {
flex: 1 1 400px;
}
If we translated the CSS, the former example with the media query says, When the viewport is smaller than 700px, I will make the elements wrap. Why? I don’t really know. Again, the query is aloof to an article’s context. If we translate the updated example, it says something along the lines of, No matter where the element is, I will try my best to make it 400px but will adjust it if the available space changes.
Resize the following demo. See how much nicer the articles flow as the screen changes size?
See the Pen [Flex Items without Media Queries [forked]](https://codepen.io/smashingmag/pen/jOobPNe) by Monknow.
And we pulled it off with less code and zero magic numbers.
Grid
The last example is nice, but you may notice that the last flexible item is able to take all the available space in the last row instead of remaining the same size as its siblings. If you want all flexible items to be the same size, you would have to mess with their width and maybe again with media queries. In most cases where you find yourself slapping a width on a flexible item, it’s a sign that you are better off switching to Grid instead, as we can establish specific tracks for columns and rows.
Fortunately, we can make it happen with just two lines of CSS:
main {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(500px, 1fr));
}
To briefly break down what happened:
auto-fit fits as many columns as it can and also expands them if there is any leftover space.
minmax specifies a minimum width for the columns, 500px in this case.
See the Pen [Grid without Media Queries [forked]](https://codepen.io/smashingmag/pen/RwmWPwV) by Monknow.
Note: Sara Soueidan has what may be the best explanation of this approach, and it is definitely worth a read. It may just be your new favorite CSS snippet.
Math Functions & Responsive Units
Math functions and responsive units cover most problems related to resizing elements. They set responsive limits without having to painstakingly define specific breakpoints. They are fully supported in modern browsers and already in widespread use, so we’ll simply summarize the what and why of what’s available.
Using the min() function, we can make elements resize depending on a responsive unit like viewport width (vw) or a relative unit like a percentage to establish an upper limit so they don’t grow too much. This element will try to cover its parent full width but won’t grow past 300px:
.min {
height: 400px;
width: min(100%, 300px);
}
We can make the height resize along the width using the aspect-ratio property:
Using the max() function, we can apply a lower limit. The following CSS allows the element to increase its size to cover up to half of its parent element but won’t ever shrink below 300px;
.max {
height: 400px;
width: max(50%, 300px);
}
It’s a bit of a mind-bender, right? We use min() to establish a maximum width and max() to establish a minimum.
Then there is the very popular clamp() function that establishes both maximum and minimum limits — but with the added bonus of setting an “ideal” size as the middle argument. We’re “clamping” values with a range it adheres to while attempting to hit that ideal target.
The element in the following demo tries to cover the full available width of its parent element but will not go above 300px or below 200px.
See the Pen [math functions [forked]](https://codepen.io/smashingmag/pen/wvbKaBj) by Monknow.
Making a website that looks great no matter the device relies on more than responsive units or math functions alone. We need the combination of all techniques to create a seamless responsive experience. You can sort of think of it like the Performance API in JavaScript that is a group of standards that work together for performance-related work.
What we have is a group of CSS specifications built around responsive design. They aren’t necessarilyreplacingCSS media queries but are additive and designed to work together for the best possible coverage.
For example, we may want a font-size value to increase or decrease depending on the width of the viewport. Easy enough with media queries, but now we have additional ways to approach this that could be more efficient or maintainable depending on your project.
We certainly could use media queries to update the font-size value at specific browser widths. We’d likely need to write more than one to get the right size at each breakpoint, but it is possible and valid.
See the Pen [Resizing text using media queries [forked]](https://codepen.io/smashingmag/pen/oNRjXXz) by Monknow.
Instead of updating the font-size with fixed pixels at multiple breakpoints, we can reach for responsive length units instead. For example, the vw unit is relative to the viewport width, where each unit represents 1% of the current browser width.
But we can go further than that because viewport units alone will not save us from font sizes that are far too small and large for their contexts. Let’s combine them with the clamp() function to establish minimum and maximum limits with our ideal size defined.
See the Pen [Resizing text individually using clamp() and vw [forked]](https://codepen.io/smashingmag/pen/yLWYNNw) by Monknow.
But wait! We can improve this even more by declaring this directly on the <html> element’s font-size, making all other fonts resize by the same factor. Then, using the rem unit, we can write how big or small each element font-size should be or opt out and use clamp(), or even fixed pixel units in specific elements.
It’s worth noting that the difference between rem and em units is that the former is relative to the “root” element, i.e., <html>, while the latter is relative to the selector’s parent element.
See the Pen [Resizing text by the same factor [forked]](https://codepen.io/smashingmag/pen/YzbyXyZ) by Monknow.
So, yes, none of this is meant to be used in isolation or as a one-to-one replacement for media queries. Building responsive interfaces takes a village, so to speak, and we have a knapsack of hammers, wrenches, nails, and screws we can use to put it all together.
Hello, Container Queries
Media queries are adept at modifying layouts on a page-wide basis. Take, for example, a shopping cart. When the viewport width is wide enough to accommodate it, we can display the included products in a wide <table> where they can breathe comfortably.
That same layout in mobile simply does not work. Tables have their own set of responsive challenges as it is, and while there is no shortage of solutions, we may be able to consider another layout using modern techniques that are way less engineered.
We are doing much more than simply changing the width or height of elements! Border colors, element visibility, and flex directions need to be changed, and it can only be done through a media query, right? Well, even in cases where we have to completely switch a layout depending on the viewport size, we can better achieve it with container queries.
Again, Problem #1 of media queries is that they only consider the viewport size when making decisions and are completely ignorant of an element’s surrounding context.
“
That may not be a big concern if all we’re talking about is a series of elements that are allowed to take up the full page width because the full page width is very much related to the viewport size, making media queries a perfectly fine choice for making adjustments.
See the Pen [Responsive Cards Using Media Queries [forked]](https://codepen.io/smashingmag/pen/ExzVjPj) by Monknow.
But say we want to display those same elements as part of a multi-column layout where they are included in a narrow column as an <aside> next to a larger column containing a <main> element. Now we’re in trouble.
A more traditional solution is to write a series of media queries depending on where the element is used and where its content breaks. But media queries completely miss the relationship between the <main> and <aside> elements, which is a big deal since the size of one influences the size of the other according to normal document flow.
See the Pen [Responsive Cards Using Media Queries Inside Container [forked]](https://codepen.io/smashingmag/pen/gOJapPo) by Monknow.
The .cards element is in the context of the <aside> element and is squished as a result of being in a narrow column. What would be great is to change the layout of each .card component when the .cards element that contains them reaches a certain size rather than when the viewport is a certain size.
That’s where container queries come into play, allowing us to conditionally apply styles based on an element’s size. We register an element as a “container,” which, in our current example, is the unordered list containing the series of .card components. We’re essentially giving the parent selector a great deal of power to influence the current layout.
.cards {
container-name: cards;
}
Container queries monitor an element by its size, and we need to tell the browser exactly how to interpret that size by giving .cards a container-type, which can be the container’s size (i.e., in the block and inline directions) or its inline-size (i.e., the total length in the inline direction). There’s a normal value that removes sizing considerations but allows the element to be queried by its styles.
We can simplify things down a bit using the container shorthand property.
.cards {
container: cards / inline-size;
}
Now, we can adjust the layout of the .card components when the .cards container is a certain inline size. Container queries use the same syntax as media queries but use the @container at-rule instead of @media.
Now, each .card is a flexible container that flows in the column direction when the width of the .cards container is less than 700px. Any wider than that, we have the same to lay them out in a row direction instead.
See the Pen [Responsive Cards Using Container Queries [forked]](https://codepen.io/smashingmag/pen/VwOvLap) by Monknow.
Style queries are a cousin to container queries in the sense that we can query the container’s styles and conditionally apply style changes to its children, say changing a child element’s color to white when the container’s background-color is set to a dark color. We’re still in the early days, and support for style queries and browser support is still evolving.
I hope this gives you a sense of how amazing it is that we have this context-aware way of establishing responsive layouts. Containers are a completely new idea in CSS (although we’ve used the term synonymously with “parent element” for ages) that is novel and elegant.
So, Are Media Queries Useless?
NO! While media queries have been the go-to solution for responsive design, their limitations are glaringly obvious now that we have more robust tools in CSS that are designed to solve those limits.
That doesn’t make media queries obsolete — merely a different tool that’s part of a larger toolset for building responsive interfaces. Besides, media queries still address vital accessibility concerns thanks to their ability to recognize a user’s visual and motion preferences — among other settings — at the operating system level.
So, yes, keep using media queries! But maybe reach for them sparingly since CSS has a lot more to offer us.
We generally use a CSS variable as a placeholder for some value we plan to reuse — to avoid repeating the same value and to easily update that value across the board if it needs to be updated.
:root {
--mix: color-mix(in srgb, #8A9B0F, #fff 25%);
}
div {
box-shadow: 0 0 15px 25px var(--mix);
}
We can register custom properties in CSS using @property. The most common example you’ll likely find demonstrates how @property can animate the colors of a gradient, something we’re unable to do otherwise since a CSS variable is recognized as a string and what we need is a number format that can interpolate between two numeric values. That’s where @property allows us to define not only the variable’s value but its syntax,initial value, and inheritance, just like you’ll find documented in CSS specifications.
For example, here’s how we register a custom property called --circleSize, which is formatted as a percentage value that is set to 10% by default and is not inherited by child elements.
@property --circleSize {
syntax: "<percentage>";
inherits: false;
initial-value: 10%;
}
div { /* red div */
clip-path: circle(var(--circleSize) at center bottom);
transition: --circleSize 300ms linear;
}
section:hover div {
--circleSize: 125%;
}
In this example, a circle() function is used to clip the <div> element into — you guessed it — a circle. The size value of the circle()’s radius is set to the registered custom property, --circleSize, which is then independently changed on hover using a transition. The result is something close to Material Design’s ripple effect, and we can do it because we’ve told CSS to treat the custom property as a percentage value rather than a string:
See the Pen [CSS @property [forked]](https://codepen.io/smashingmag/pen/PovwepK) by Preethi Sam.
The freedom to define and spec our own CSS properties gives us new animating superpowers that were once only possible with JavaScript, like transitioning the colors of a gradient.
“
Here’s an idea I have that uses the same basic idea as the ripple, only it chains multiple custom properties together that are formatted as colors, lengths, and angle degrees for a more complex animation where text slides up the container as the text changes colors.
See the Pen [Text animation with @property [forked]](https://codepen.io/smashingmag/pen/rNgavyb) by Preethi Sam.
Let’s use this demo as an exercise to learn more about defining custom properties with the @property at-rule, combining what we just saw in the ripple with the concept of interpolating gradient values.
The HTML contains Chinese characters we’re going to animate. These Chinese characters are marked up with <ruby> tags so that their English translations can be supplied in <rt> tags. The idea is that .scrolling-text is the component’s parent container and, in it, is a child element holding the sliding text characters that allow the characters to slide in and out of view.
Vertical Sliding
In CSS, let’s make the characters slide vertically on hover. What we’re making is a container with a fixed height we can use to clip the characters out of view when they overflow the available space.
Setting the .scrolling-text container’s width to min-content gives the characters a tight fit, stacking them vertically in a single column. The container’s height is set 1lh. And since we’ve set overflow: hidden on the container, only one character is shown in the container at any given point in time.
Tip: You can also use the HTML <pre> element or either the white-space or text-wrap properties to control how text wraps.
On hover, the text moves -2lh, or double the height of a single text character in the opposite, or up, direction. So, basically, we’re sliding things up by two characters in order to animate from the first character to the third character when the container holding the text is in a hovered state.
How often do you find yourself using repeating gradients in your work? The fun part, though, is what comes after it. See, we’re setting a transparent color on the text and that allows the repeating-linear-gradient() to show through it. But since text is a box like everything else in CSS, we clip the background at the text itself to make it look like the text is cut out of the gradient.
See the Pen [A gradient text (Note: View in Safari or Chrome) [forked]](https://codepen.io/smashingmag/pen/BaeyxZJ) by Preethi Sam.
Pretty neat, right? Now, it looks like our text characters have a striped pattern painted on them.
Animating The Gradient
This is where we take the same animated gradient concept covered in other tutorials and work it into what we’re doing here. For that, we’ll first register some of the repeating-linear-gradient() values as custom properties. But unlike the other implementations, ours is a bit more complex because we will animate several values rather than, say, updating the hue.
Instead, we’re animating two colors, a length, and an angle.
We want to update the values of our registered custom properties when the container that holds the text is hovered or in focus. All that takes is re-declaring the properties with the updated values.
To be super clear about what’s happening, these are the custom properties and values that update on hover:
--c1: Starts with a color value of rgb(224, 236, 236) and updates to pink.
--c2: Starts with a color value of rgb(92, 198, 162) and updates to transparent.
--l: Starts with length value 5px and updates to 100%.
--a: Starts with an angle value of 180deg and updates to 90deg.
So, the two colors used in the gradient transition into other colors while the overall size of the gradient increases and rotates. It’s as though we’re choreographing a short dance routine for the gradient.
Refining The Transition
All the while, the .text element containing the characters slides up to reveal one character at a time. The only thing is that we have to tell CSS what will transition on hover, which we do directly on the .text element:
Yes, I could just as easily have used the all keyword to select all of the transitioning properties. But I prefer taking the extra step of declaring each one individually. It’s a little habit to keep the browser from having to watch for too many things, which could slow things down even a smidge.
Final Demo
Here’s the final outcome once again:
See the Pen [Text animation with @property [forked]](https://codepen.io/smashingmag/pen/qBGEYXO) by Preethi Sam.
I hope this little exercise not only demonstrates the sorts of fancy things we can make with CSS custom properties but also helps clarify the differences between custom properties and standard variables. Standard variables are excellent placeholders for more maintainable code (and a few fancy tricks of their own) but when you find yourself needing to update one value in a property that supports multiple values — such as colors in a gradient — the @property at-rule is where it’s at because it lets us define variables with a custom specification that sets the variable’s syntax, initial value, and inheritance behavior.
When we get to amend values individually and independently with a promise of animation, it both helps streamline the code and opens up new possibilities for designing elaborate animations with relatively nimble code.
“
That’s why @property is a useful CSS standard to keep in mind and keep ready to use when you are thinking about animations that involve isolated value changes.
You have for sure googled “how to create [shape_name]withCSS” at least once in your front-end career if it’s not something you already have bookmarked. And the number of articles and demos you will find out there is endless.
Good, right? Copy that code and drop it into the ol’ stylesheet. Ship it!
The problem is that you don’t understand how the copied code works. Sure, it got the job done, but many of the most widely used CSS shape snippets are often dated and rely on things like magic numbers to get the shapes just right. So, the next time you go into the code needing to make a change to it, it either makes little sense or is inflexible to the point that you need an entirely new solution.
So, here it is, your one-stop modern guide for how to create shapes in CSS! We are going to explore the most common CSS shapes while highlighting different CSS tricks and techniques that you can easily re-purpose for any kind of shape. The goal is not to learn how to create specific shapes but rather to understand the modern tricks that allow you to create any kind of shape you want.
Table of Contents
You can jump directly to the topic you’re interested in to find relevant shapes or browse the complete list. Enjoy!
I get asked this question often, and my answer is always the same: Use SVG if youcan! I have nothing against SVG. It’s just another approach for creating shapes using another syntax with another set of considerations. If SVG was my expertise, then I would be writing about that instead!
CSS is my field of expertise, so that’s the approach we’re covering for drawing shapes with code. Choosing CSS or SVG is typically a matter of choice. There may very well be a good reason why SVG is a better fit for your specific needs.
Many times, CSS will be your best bet for decorative things or when you’re working with a specific element in the markup that contains real content to be styled. Ultimately, though, you will need to consider what your project’s requirements are and decide whether a CSS shape is really what you are looking for.
Your First Resource
Before we start digging into code, please spend a few minutes over at my CSS Shape website. You will find many examples of CSS-only shapes. This is an ever-growing collection that I regularly maintain with new shapes and techniques. Bookmark it and use it as a reference as we make our way through this guide.
Is it fairly easy to modify and tweak the CSS for those shapes?
Yes! The CSS for each and every shape is optimized to be as flexible and efficient as possible. The CSS typically targets a single HTML element to prevent you from having to touch too much markup besides dropping the element on the page. Additionally, I make liberal use of CSS variables that allow you to modify things easily for your needs.
Most of you don’t have time to grasp all the techniques and tricks to create different shapes, so an online resource with ready-to-use snippets of code can be a lifesaver!
Clipping Shapes In CSS
The CSS clip-path property — and its polygon() function — is what we commonly reach for when creating CSS Shapes. Through the creation of common CSS shapes, we will learn a few tricks that can help you create other shapes easily.
Hexagons
Let’s start with one of the easiest shapes; the hexagon. We first define the shape’s dimensions, then provide the coordinates for the six points and we are done.
Easy, right? But what if I told you that there’s an even easier way to do it? Instead of six points, we can get by with just four.
A little-known trick with the polygon() function is that we are allowed to set points that are outside the [0% 100%] range. In other words, we can cut outside the element — which becomes super useful for this shape as well many others, as we’ll see.
Figure 1: Clipping a hexagon with four points. (Large preview)
We’re basically drawing the shape of a diamond where two of the points are set way outside the bounds of the hexagon we’re trying to make. This is perhaps the very first lesson for drawing CSS shapes: Allow yourself to think outside the box — or at least the shape’s boundaries.
Did you notice that I updated the aspect-ratio property in there? I’m using a trigonometric function, cos(), to replace the magic number 0.866. The exact value of the ratio is equal to cos(30deg) (or sin(60deg)). Besides, cos(30deg) is a lot easier to remember than 0.866.
Here’s something fun we can do: swap the X and Y coordinate values. In other words, let’s change the polygon() coordinates from this pattern:
clip-path: polygon(X1 Y1, X2 Y2, ..., Xn Yn)
…to this, where the Y values come before the X values:
clip-path: polygon(Y1 X1, Y2 X2, ..., Yn Xn)
What we get is a new variation of the hexagon:
See the Pen [Another variation of the hexagon shape](https://codepen.io/t_afif/pen/BaEZrrP) by Temani Afif.
Swapping the X and Y values will make a kind of switch between the vertical and horizontal axes, which will help to get a different shape. Note that I have also updated the ratio to 1/cos(30deg) instead of cos(30deg). Since we are switching both axes, the new ratio needs to be equal to its inverse, i.e., R (or R/1) becomes 1/R.
And since our CSS is nothing more than a single style rule on a single selector, we can apply it to more than a <div>. For example, the following demo includes both variations of the original hexagon, plus a third example that sets the styles on an <img> element.
See the Pen [CSS-only hexagon shapes (the modern way)](https://codepen.io/t_afif/pen/KKEMjxV) by Temani Afif.
There we go, our first shape! We are also walking away with two valuable lessons about creating shapes with CSS:
The polygon() function accepts points outside the [0% 100%] range.
This allows us to clip shapes with fewer points in some cases but also opens up possibilities for creating additional shapes.
Switching axes is a solid approach for creating shape variations.
In the case of a hexagon, swapping the values on the X and Y axes changes the hexagon’s direction.
Octagons
An octagon is another geometric shape and it is very close in nature to the hexagon. Instead of working with six sides, we’re working with eight to get what looks like the shape of a common traffic stop sign.
Let’s take the first lesson we learned from working with hexagons and clip the element with coordinates outside the shape’s boundaries to keep our clipping efficient. Believe it or not, we can actually establish all eight octagon sides with only four points, just like we used only four points to establish the hexagon’s six sides.
Figure 2: Clipping an octagon with four points. (Large preview)
I know that visualizing the shape with outside points can be somewhat difficult because we’re practically turning the concept of clipping on its head. But with some practice, you get used to this mental model and develop muscle memory for it.
Notice that the CSS is remarkably similar to what we used to create a hexagon:
Except for the small trigonometric formula, the structure of the code is identical to the last hexagon shape — set the shape’s dimensions, then clip the points. And notice how I saved the math calculation as a CSS variable to avoid repeating that code.
If math isn’t really your thing — and that’s totally fine! — remember that the formulas are simply one part of the puzzle. There’s no need to go back to your high school geometry textbooks. You can always find the formulas you need for specific shapes in my online collection. Again, that collection is your first resource for creating CSS shapes!
And, of course, we can apply this shape to an <img> element as easily as we can a <div>:
See the Pen [CSS-only octagon shapes (the modern way)](https://codepen.io/t_afif/pen/LYaxqEg) by Temani Afif.
The most obvious difference is that the variable containing the math function (--o) is removed, and we have a new one, --w, for setting the shape’s dimensions.
But notice that we’re now setting margin on the shape and declaring a margin-box keyword at the end of the clip-path. It means that the reference for the polygon() is now set to the margin-box instead of the default border-box.
If you look back at Figure 2, notice that the four points used to draw the octagon are outside the shape’s boundaries and have the same distance from those boundaries. Instead of accounting for that distance inside the clip-path, the updated code declares it on the margin property, which makes the values of the coordinates easier to define.
All the --o variables are removed from the clip-path, and the margin property gets that same value. I had to introduce a new variable, --w, to set the element’s size dimensions because I couldn’t rely on a percentage value. In this particular case, you will end with some margin around the element, but this trick can really help simplify calculations.
If you don’t want the extra margin, you can add padding instead and apply the same amount of padding as a negative margin. That’s another trick to keep polygons simple in a way that works well with images. Here is a demo showing different shapes created with the same clip-path value.
See the Pen [Different shapes using the same polygon](https://codepen.io/t_afif/pen/oNOOWqz) by Temani Afif.
Creating a star shape is always a bit tricky, even if you are comfortable using clip-path with the polygon() function. Clipping requires very precise values, so we either find a ready-to-use snippet of CSS or fuss with it ourselves until we get it right.
And if I were to ask you how many points we need to cut the shape, you might reasonably respond that 10 points are needed. And you are technically correct. But we can do better using only five points!
Figure 3: Drawing a star shape with five points instead of 10 points. (Large preview)
It may sound impossible to make a star out of only five points, but it’s perfectly possible, and the trick is how the points inside polygon() are ordered. If we were to draw a star with pencil on paper in a single continuous line, we would follow the following order:
Figure 4: Drawing a star with a single line illustrates the order of the points we need to create the shape. (Large preview)
It’s the same way we used to draw stars as kids — and it fits perfectly in CSS with polygon()! This is another hidden trick about clip-path with polygon(), and it leads to another key lesson for drawing CSS shapes: the lines we establish can intersect. Again, we’re sort of turning a concept on its head, even if it’s a pattern we all grew up making by hand.
I am using trigonometric functions again for accuracy without resorting to magic numbers, but even if we calculate the values, the code is still better than the traditional 10-point approach:
Since we have a symmetrical shape, note that the second and fifth points on the star share the same Y coordinates. The same is true for the third and fourth points. And notice, too, that the X values have the same distance to the center (79% - 50% = 50% - 21%). If we add those up, we see that the sum is equal to 100% (79% + 21% = 100%).
That leads us to yet another major lesson on drawing CSS shapes: Consider the shape’s symmetry because that’s a big hint that there may be duplicated values. This will reduce your effort in calculating/finding the different values.
We already cut the number of points once from 10 to five. Now, there are only three points to remember — the remaining two can be figured out from there, thanks to symmetry.
Go back to the hexagon and octagon shapes and look for symmetry. You will notice repeated values as well, and the clip-path will suddenly look easier to remember!
Polygons & Starbursts
We’ve already covered stars, hexagons, and octagons, but what if you are working with an unknown number of points or sides? You may want a solution that is capable of adjusting the number for whatever situation it is used for. For this, we can consider more generic shapes like polygons and starbursts.
Geometric shapes come with a number of points and sides. (Large preview)
The funny thing is that starbursts are basically the exact same thing as polygons, just with half the points that we can move inward.
Figure 6.
I often advise people to use my online generators for shapes like these because the clip-path coordinates can get tricky to write and calculate by hand.
That said, I really believe it’s still a very good idea to understand how the coordinates are calculated and how they affect the overall shape. I have an entire article on the topic for you to learn the nuances of calculating coordinates.
Parallelograms & Trapezoids
Another common shape we always build is a rectangle shape where we have one or two slanted sides. They have a lot of names depending on the final result (e.g., parallelogram, trapezoid, skewed rectangle, and so on), but all of them are built using the same CSS technique.
Figure 7: Parallelograms and a trapezoid. (Large preview)
First, we start by creating a basic rectangle by linking the four corner points together:
This code produces nothing because our element is already a rectangle. Also, note that 0 and 100% are the only values we’re using.
Next, offset some values to get the shape you want. Let’s say our offset needs to be equal to 10px. If the value is 0, we update it with 10px, and if it’s 100% we update it with calc(100% - 10px). As simple as that!
But which value do I need to update and when?
Try and see! Open your browser’s developer tools and update the values in real-time to see how the shape changes, and you will understand what points you need to update. I would lie if I told you that I write all the shapes from memory without making any mistakes. In most cases, I start with the basic rectangle, and I add or update points until I get the shape I want. Try this as a small homework exercise and create the shapes in Figure 11 by yourself. You can still find all the correct code in my online collection for reference.
We just worked with a number of shapes that required us to figure out a number of points and clip-path by plotting their coordinates in a polygon(). In this section, we will cover circular and curvy shapes while introducing the other property you will use the most when creating CSS shapes: the mask property.
Like the previous section, we will create some shapes while highlighting the main tricks you need to know. Don’t forget that the goal is not to learn how to create specific shapes but to learn the tricks that allow you to create any kind of shape.
Circles & Holes
When talking about the mask property, gradients are certain to come up. We can, for example, “cut” (but really “mask”) a circular hole out of an element with a radial-gradient:
mask: radial-gradient(50px, #0000 98%, #000);
Why aren’t we using a simple background instead? The mask property allows us more flexibility, like using any color we want and applying the effect on a variety of other elements, such as <img>. If the color and flexible utility aren’t a big deal, then you can certainly reach for the background property instead of cutting a hole.
Here’s the mask working on both a <div> and <img>:
See the Pen [Hole shape](https://codepen.io/t_afif/pen/OJGgGve) by Temani Afif.
It’s here that I’d like to call out yet another lesson for creating shapes in CSS: The colors we use in gradients are completely unimportant when working with mask.
All we care about is the color value’s alpha channel because transparency is what is mask-ed out of the element, establishing the circular hole in the center. The gradient’s opaque colors preserve the visibility of the rest of the element. That’s why you will often see me using a black color value (e.g., #000) for the visible part and a transparent color (e.g., #0000) for the invisible part.
Notice the hard color stops in the gradient. A smooth transition between colors would lead to blurry lines. If we remove that transition and sharply change from one color to another, we get smooth, sharp edges. But not totally! I prefer to keep a very small transition (98% instead of 100%) to avoid jagged edges.
And with a simple radial-gradient, we can achieve a lot of shapes, like cutting a circle from the top or bottom of an element.
See the Pen [Circular cut from the top & bottom](https://codepen.io/t_afif/pen/MWRvBOL) by Temani Afif.
Rather than dissecting the code for that last example, I want you to peek at the CSS and see for yourself how the radial-gradient is configured. You will notice that we went from a simple hole to a fancy border decoration by making only a few changes.
Border Edges
The previous demo is one example of many fancy borders we can create. We can go wavy, spiked, scalloped, and more!
Once again, it’s all about CSS masks and gradients. In the following articles, I provide you with examples and recipes for many different possibilities:
This is another instance where CSS gradients are the perfect fit for mask-ing shapes. You’ve probably seen this type of shape a gazillion times because it’s a common pattern for animated loading indications.
Figure 9: Circular progress element with rounded edges and gradient coloration. (Large preview)
This time, we are going to introduce another technique which is “composition”. It’s an operation we perform between two gradient layers. We either use mask-composite to define it, or we declare the values on the mask property.
The figure below illustrates the gradient configuration and the composition between each layer.
Figure 10: Combining radial and conical gradients to establish the final shape. (Large preview)
We start with a radial-gradient to create a full circle shape. Then we use a conic-gradient to create the shape below it. Between the two gradients, we perform an “intersect” composition to get the unclosed circle. Then we tack on two more radial gradients to the mask to get those nice rounded endpoints on the unclosed circle. This time we consider the default composition, “add”.
Gradients aren’t something new as we use them a lot with the background property but “composition” is the new concept I want you to keep in mind. It’s a very handy one that unlocks a lot of possibilities.
Even if the code looks a bit complex at first glance, the use of CSS variables makes things easier to adjust. That’s an important CSS technique I am using in most of the shapes I have created. Many of them require complex formulas and a lot of gradients, but in the end, all you have to do is adjust a few variables to control the shape. So, let’s not spend too much time explaining the math expressions. I want to focus on the tricks and techniques because the CSS concepts are what is important; remember, you can always grab the math. How CSS uses it is key.
Notice that we can achieve the same result using different gradient combinations. It’s weird because the syntax looks completely different. This snippet accomplishes the same visual result.
I added border-radius in there to round the element and added padding equal to the border’s thickness. Then, if you check the gradient used in the mask, you will see that I have changed the radial-gradient with a linear-gradient containing a single transparent color that covers the element’s content-box.
Sure, there are two more variables using this approach, but I did simplify the overall gradient at the same time. Yet another valid approach for the same effect.
See the Pen [Untitled](https://codepen.io/t_afif/pen/WNWErpV) by Temani Afif.
This time we’re combining two gradients in our mask. One is a black-to-transparent repeating-conic-gradient and the other is a transparent linear-gradient configured to cover the element up to its content-box and the mask-composite property is set to intersect.
Tabs are a super common design pattern. Each tab is connected to a panel of content where clicking a tab reveals that panel of content. Tabs can be rectangular, but we often think of them as rounded, the way they are on actual paper file folders.
We could get clever and use a pseudo-element for the shape that’s positioned behind the set of panels, but that introduces more complexity and fixed values than we ought to have. Instead, we can continue using CSS masks to get the perfect shape with a minimal amount of reusable code.
It’s not really the rounded top edges that are difficult to pull off, but the bottom portion that curves inwards instead of rounding in like the top. And even then, we already know the secret sauce: using CSS masks by combining gradients that reveal just the parts we want.
Notice that the shape’s rounded top edges are equal to two times the radius (--r) value. If you’re wondering why we need a calculation here at all, it’s because we have a transparent border hanging out there, and we need to double the radius to account for it. The radius of the blue areas highlighted in Figure 13 is equal to 2 * R while the red area highlighted in the same figure is equal to 2 * R - R, or simply R.
We can actually optimize the code so that we only need two gradients — one linear and one radial — instead of three. I’ll drop that into the following demo for you to pick apart. Can you figure out how we were able to eliminate one of the gradients?
I’ll throw in two additional variations for you to investigate:
See the Pen [Rounded tab using CSS mask](https://codepen.io/t_afif/pen/JjVpPmr) by Temani Afif.
I’m often asked how I know when my code can be optimized more than it is. That’s truly the most difficult part of everything we’ve covered so far. I do not have any hard rules for how and when to optimize, and it’s not necessary to find the optimal solution, especially if you are a beginner. My advice is to first find the trivial and easy solution, even if it requires a lot of gradients. Then, with a lot of practice, you will be able to find better solutions.
Talking about practice, here’s your next bit of homework: try creating the shapes illustrated in Figure 15:
These aren’t tabs at all but tooltips! We can absolutely use the exact same masking technique we used to create the tabs for these shapes. Notice how the curves that go inward are consistent in each shape, no matter if they are positioned on the left, right, or both.
You can always find the code over at my online collection if you want to reference it.
More CSS Shapes
At this point, we’ve seen the main tricks to create CSS shapes. You will rely on mask and gradients if you have curves and rounded parts or clip-path when there are no curves. It sounds simple but there’s still more to learn, so I am going to provide a few more common shapes for you to explore.
Instead of going into a detailed explanation of the shapes in this section, I’m going to give you the recipes for how to make them and all of the ingredients you need to make it happen. In fact, I have written other articles that are directly related to everything we are about to cover and will link them up so that you have guides you can reference in your work.
Triangles
A triangle is likely the first shape that you will ever need. They’re used in lots of places, from play buttons for videos, to decorative icons in links, to active state indicators, to open/close toggles in accordions, to… the list goes on.
Please check out my article “CSS Shapes: The Triangle” on the Verpex blog for a full explanation of techniques with many examples and variations.
Hearts
Hearts are another classic shape that’s been tackled with older CSS techniques but have a better modern equivalent. We can pull this off more simply by combining border-image and clip-path:
There are many different types of ribbons, as you might imagine. So, rather than detail one I will provide you with four articles I have written detailing the general technique (more clips!) and a variety of fun variations for you to consider.
By the end, you can literally create as many variations as you can imagine.
Cutting Corners
Insert your obligatory joke about how we’re supposed to cut corners in life. However, when we cut corners out of squares and rectangles, the result is a nice decorative shape that also works as a frame for images.
We can cut all the corners or just specific ones. We can make circular cuts or sharp ones. We can even create an outline of the overall shape. Take a look at my online generator to play with the code, and check out my full article on the topic where I am detailing all the different cases.
Cut-Out Shapes
In addition to cutting corners, we can also cut a shape out of a rectangle. They are also called inverted shapes.
The technique is all about setting the CSS clip-path property with the shape’s coordinates in the polygon() function. So, technically, this is something you already know, thanks to the examples we’ve looked at throughout this guide.
See the Pen [“Cut-out shapes using clip-path”](https://codepen.io/t_afif/pen/gOJvdav) by Temani Afif ([@t_afif](https://codepen.io/t_afif))
I hope you see the pattern now: sometimes, we’re clipping an element or masking portions of it. The fact that we can sort of “carve” into things this way using polygon() coordinates and gradients opens up so many possibilities that would have required clever workarounds and super-specific code in years past.
See my article “How to Create a Section Divider Using CSS” on the freeCodeCamp blog for a deep dive into the concepts, which we’ve also covered here quite extensively already in earlier sections.
Inner Curves
Inverted radius, notch, rounded cut, bell curve… many names can be used to describe these shapes with smooth inner curves. They’re also another use case of using CSS mask to combine gradients for creating variations.
These shapes are pretty cool on their own. But like a few of the other shapes we’ve covered, this one works extremely well with images. If you need something fancier than the typical box, then masking the edges can come off like a custom-framed photo.
There’s a lot of math involved with this, specifically trigonometric functions. I have a two-part series that gets into the weeds if you’re interested in that side of things:
As always, remember that my online collection is your Number One resource for all things related to CSS shapes. The math has already been worked out for your convenience, but you also have the references you need to understand how it works under the hood.
Wavy & Zig-Zag boxes
We saw how to create a Zig-Zag and a Wave on one side but we can also have them on all the sides and make fancy boxes!
Cool right? Check the following articles if you want to understand the logic behind creating them:
I hope you see CSS Shapes differently now as a result of reading this comprehensive guide. We covered a few shapes, but really, it’s hundreds upon hundreds of shapes because you see how flexible they are to configure into a slew of variations.
At the end of the day, all of the shapes use some combination of different CSS concepts such as clipping, masking, composition, gradients, CSS variables, and so on. Not to mention a few hidden tricks like the one related to the polygon() function:
It accepts points outside the [0% 100%] range.
Switching axes is a solid approach for creating shape variations.
The lines we establish can intersect.
It’s not that many things, right? We looked at each of these in great detail and then whipped through the shapes to demonstrate how the concepts come together. It’s not so much about memorizing snippets than it is thoroughly understanding how CSS works and leveraging its features to produce any number of things, like shapes.
Don’t forget to bookmark my CSS Shape website and use it as a reference as well as a quick stop to get a specific shape you need for a project. I avoid re-inventing the wheel in my work, and the online collection is your wheel for snagging shapes made with pure CSS.
Please also use it as inspiration for your own shape-shifting experiments. And post a comment if you think of a shape that would be a nice addition to the collection.
Even though the CSS :has() pseudo-class is relatively new, we already know a lot about it, thanks to many, many articles and tutorials demonstrating its powerful ability to conditionally select elements based on their contents. We’ve all seen the card component and header examples, but the conditional nature of :has() actually makes it adept at working with form controls, which are pretty conditional in nature as well.
Let’s look specifically at the <select> element. With it, we can make a choice from a series of <option>s. Combined with :has(), we are capable of manipulating styles based on the selected <option>.
This is your standard <select> usage, producing a dropdown menu that contains options for user selection. And while it’s not mandatory, I’ve added the selected attribute to the first <option> to set it as the initial selected option.
Applying styles based on a user’s selection is not a new thing. We’ve had the Checkbox Hack in our pockets for years, using the :checked CSS pseudo-class to style the element based on the selected option. In this next example, I’m changing the element’s color and the background-color properties based on the selected <option>.
See the Pen [demo 01 – Using the :has selector on a dropdown menu](https://codepen.io/smashingmag/pen/oNOwded) by Amit Sheen.
But that’s limited to styling the current element, right? If a particular <option> is :checked, then we style its style. We can write a more complex selector and style child elements based on whether an <option> is selected up the chain, but that’s a one-way road in that we are unable to style up parent elements even further up the chain.
That’s where :has() comes in because styling up the chain is exactly what it is designed to do; in fact, it’s often called the “parent selector” for this reason (although “family selector” may be a better descriptor).
For example, if we want to change the background-color of the <select> element according to the value of the selected <option>, we select the element if it has a specific [value] that is :checked.
See the Pen [demo 02 – Using the :has selector on a dropdown menu](https://codepen.io/smashingmag/pen/eYoRopZ) by Amit Sheen.
Just how practical is this? One way I’m using it is to style mandatory <select> elements without a valid selected <option>. So, instead of applying styles if the element :has() a :checked state, I am applying styles if the required element does :not(:has(:checked)).
See the Pen [demo 02.1 – Using the :has selector on a dropdown menu](https://codepen.io/smashingmag/pen/jORLoVM) by Amit Sheen.
But why stop there? If we can use :has() to style the <select> element as the parent of an <option>, then we can also use it to style the parent of the <select>, as well as its parent, in addition to its parent, and even its parent… all the way up the chain to the :root element. We could even bring :has() all the way up the chain and sniff out whether any <select> child of the document :root:has() a particular <option> that is :checked:
:root:has(select [value="foo"]:checked) {
// Styles applied if <option value="foo"> is <select>-ed
}
This is useful for setting a custom property value dynamically or applying a set of styles for the whole page. Let’s make a little style picker that illustrates the idea of setting styles on an entire page.
See the Pen [demo 03 – Using the :has selector on a dropdown menu](https://codepen.io/smashingmag/pen/yLrXroO) by Amit Sheen.
How that last example works is that I added a class to each <select> element and referenced that class inside the :has() selector in order to prevent unwanted selections in the event that there are multiple <select> elements on the page.
And, of course, we don’t have to go all the way up to the :root element. If we’re working with a specific component, we can scope :has() to that component like in the following demo of a star rating component.
See the Pen [demo 05 – Using the :has selector on a dropdown menu](https://codepen.io/smashingmag/pen/rNbwvqz) by Amit Sheen.
We’d be doing :has() a great disservice if we only saw it as a “parent selector” rather than the great conditional operator it is for applying styles all the way up the chain. Seen this way, it’s more of a modern upgrade to the Checkbox Hack in that it sends styles up like we were never able to do before.
There are endless examples of using :has() to create style variations of a component according to its contents. We’ve even seen it used to accomplish the once-complicated linked card pattern. But now you have an example for using it to create dropdown menus that conditionally apply styles (or don’t) to a page or component based the currently selected option — depending on how far up the chain we scope it.
I’ve used this technique a few different ways — e.g., as form validation, a style picker, and star ratings — but I’m sure there are plenty of other ways you can imagine how to use it in your own work. And if you are using :has() on a <select> element for something different or interesting, let me know because I’d love to see it!
In a previous article, we played with CSS masks to create cool hover effects where the main challenge was to rely only on the <img> tag as our markup. In this article, pick up where we left off by “revealing” the image from behind a sliding door sort of thing — like opening up a box and finding a photograph in it.
See the Pen [Image gift box (hover to reveal)](https://codepen.io/smashingmag/pen/LYaPPPo) by Temani Afif.
Pretty neat, right? You might think this is an easy thing to pull off. All we really need is an overlay above the image that we translate, and, boom, we’re done, right?
That’s true. But if you check the code, you won’t find any additional elements in the markup other than the exact same <img> tag we used last time. Plus, we cannot even use pseudo-elements to make this work. This is what makes such an effect a bit more challenging.
Don’t look at the code right now. Let’s build it together by breaking the demo into isolated little CSS tricks.
The Image And Sliding Overlay
You would be correct in thinking it’s impossible to add an overlay to an image without an extra element. Instead, we are going to fake it and create the illusion of an overlay.
We have defined the width as a CSS variable (--s) and repurposed it to apply padding along the right side of the element. Combined with box-sizing: border-box, this will make the size of the content box equal to 0. In other words, we don’t see the image, but we see the background color since it covers the padding area.
On hover, let’s make the padding equal to 0:
See the Pen [Padding animation to reveal the image](https://codepen.io/smashingmag/pen/jOJrqXo) by Temani Afif.
Nothing surprising, right? By decreasing the padding, we increase the size of the content box and it slowly reveals the image. We’re basically squishing it vertically and allowing to widen back into place on hover.
Let’s add two more properties to the mix:
img {
object-fit: cover;
object-position: left;
}
See the Pen [Adding object-* properties](https://codepen.io/smashingmag/pen/oNVLLdM) by Temani Afif.
Tada! The effect looks much better now, and we have an overlay reveal animation even if, in reality, the overlay you see is the background, which is behind the image! The illusion is perfect.
Why does it behave like that? The logic is explained nicely over at MDN:
“The replaced content is sized to maintain its aspect ratio while filling the element’s entire content box. If the object’s aspect ratio does not match the aspect ratio of its box, then the object will be clipped to fit.”
In other words, the image will maintain its ratio while filling the content box. As a result, the image does not get distorted by the padding as we saw in the first demo — instead, it is clipped. Then, object-position: left aligns the position of the image to the left so it doesn’t move while the size of the content box increases as a result of the decreased padding on hover.
If we change the position to right, you get a different effect:
See the Pen [Using object-position: right](https://codepen.io/smashingmag/pen/xxBOOJW) by Temani Afif.
Instead of an overlay animation, we have a kind of sliding effect where the image enters from the left. This is directly related to another cool CSS trick that I used in a previous article to create a “pop-out” hover effect:
See the Pen [Fancy Pop Out Reveal hover effect!](https://codepen.io/smashingmag/pen/VwqWRyj) by Temani Afif.
You will notice that it’s pretty easy to switch between the different variations by toggling a couple of values in the CSS.
Sliding The Overlay Outside The Image
Now that we have our overlay, let’s try to slide it outside of the image. Instead of decreasing its size like we did previously, we want it to maintain its size and move it.
Cool, right? We have an overlay above our image that slides over to reveal the image — without using any extra elements in the markup or pseudo-elements in the styles!
We can do the same effect using a clip-path animation as well.
We define a box-shadow as having a widespread radius, but we won’t actually see it because it’s clipped. On hover, though, we update the inset() value to reveal the box-shadow on the right side of the image.
See the Pen [Using clip-path instead of box-shadow](https://codepen.io/smashingmag/pen/YzgWxxK) by Temani Afif.
Using the same technique, we can slide the overlay in whatever direction we want. Can you figure out how? Give it a shot by forking the Pen above and changing directions as an exercise before we move to the next part of our work.
Adding Borders
Borders can help create space around the image and get it close to a square box shape. Don’t forget that we want to create a 3D box in the end. But let’s see what happens when we add borders.
See the Pen [Adding border](https://codepen.io/smashingmag/pen/YzgWrvQ) by Temani Afif.
Hmm, not good. The border sits above the overlay, and the image isn’t a perfect square, at least initially. Even if that seems glitchy at first, it’s a logical outcome since the border is painted above the background, and its thickness adds up to the element’s total size.
What we need to do is adjust the padding to account for the border’s size. Then, let’s make the border transparent so that we can see the background color behind it.
First off, note that we’ve added the color-mix() function that allows us to define a new color variation from the original color value (--c:#8A9B0F) by mixing it with white to get a brighter shade. Then, we use that new color to create a gradient above the element’s background color, which is declared right after the gradient. The same color is also used for the box-shadow.
The idea is to decrease the size of the gradient the same way we do with the padding so that the background-color behind the gradient is revealed.
See the Pen [Adding gradient animation](https://codepen.io/smashingmag/pen/rNRLJpB) by Temani Afif.
That’s really nice! But did you catch the subtle visual issue? If you look closely, you can notice that the overlay is slightly out of alignment with the border.
This is because the padding has a transition that goes from s - 2*b to 0. Meanwhile, the background transitions from 100% (equivalent to --s) to 0. There’s a difference equal to 2*b. The background covers the entire area, while the padding covers less of it. We need to account for this.
Ideally, the padding transition would take less time to complete and have a small delay at the beginning to sync things up, but finding the correct timing won’t be an easy task. Instead, let’s increase the padding transition’s range to make it equal to the background.
The new variable, --h, transitions from s - b to -b on hover, so we have the needed range since the difference is equal to --s, making it equal to the background and clip-path transitions.
The trick is the min() function. When --h transitions from s - b to s - 2*b, the padding is equal to s - 2*b. No padding changes during that brief transition. Then, when --h reaches 0 and transitions from 0 to -b, the padding remains equal to 0 since, by default, it cannot be a negative value.
It would be more intuitive to use clamp() instead:
That said, we don’t need to specify the lower parameter since padding cannot be negative and will, by default, be clamped to 0 if you give it a negative value.
We are getting much closer to the final result!
See the Pen [Fixing the padding issue](https://codepen.io/smashingmag/pen/ExMgZbW) by Temani Afif.
Worth noting that we need to use @property to be able to apply a transition to the --h variable. The transition won’t work in Firefox at the time of this writing.
The 3D Effect
The last step is to add a touch of 3D to the effect. To better understand how we’re going to approach this, let’s temporarily remove the box-shadow, clip-path, and the linear-gradient() with the image in its revealed state.
See the Pen [The revealed image with border](https://codepen.io/smashingmag/pen/QWoKpyE) by Temani Afif.
Notice the semi-transparent black color values (e.g., #0008 and #0004). The slight bit of transparency blends with the colors behind it to create the illusion of a dark variation of the main color since the gradient is placed above the background color.
And lastly, we apply a clip-path to cut out the corners that establish the 3D box.
Let’s do the same thing with the linear gradient. We need to decrease its size so it covers the same area as it did before we introduced the depth so that it doesn’t overlap with the conic gradient:
See the Pen [Putting back the gradient animation](https://codepen.io/smashingmag/pen/VwRKpzN) by Temani Afif.
We are getting closer! The last piece we need to add back in from earlier is the clip-path transition that is combined with the box-shadow. We cannot reuse the same code we used before since we changed the clip-path value to create the 3D box shape. But we can still transition it to get the sliding result we want.
The idea is to have two points at the top that move up and down to reveal and hide the box-shadow while the other points remain fixed. Here is a small video to illustrate the movement of the points.
See that? We have five fixed points. The two at the top move to increase the area of the polygon and reveal the box shadow.
And we’re done! We’re left with a nice 3D frame around the image element with a cover that slides up and down on hover. And we did it with zero extra markup or reaching for pseudo-elements!
See the Pen [3D image with reveal effect](https://codepen.io/smashingmag/pen/GRejXMK) by Temani Afif.
This last demo is an optimized version of what we did together. I have written most of the formulas using the variable --h so that I only update one value on hover. It also includes another variation. Can you reverse-engineer it and see how its code differs from the one we did together?
One More 3D Example
Want another fancy effect that uses 3D effects and sliding overlays? Here’s one I put together using a different 3D perspective where the overlay splits open rather than sliding from one side to the other.
See the Pen [Image gift box II (hover to reveal)](https://codepen.io/smashingmag/pen/yLwBVGQ) by Temani Afif.
Your homework is to dissect the code. It may look complex, but if you trace the steps we completed for the original demo, I think you’ll find that it’s not a terribly different approach. The sliding effect still combines the padding, the object-* properties, and clip-path but with different values to produce this new effect.
Conclusion
I hope you enjoyed this little 3D image experiment and the fancy effect we applied to it. I know that adding an extra element (i.e., a parent <div> as a wrapper) to the markup would have made the effect a lot easier to achieve, as would pseudo-elements and translations. But we are here for the challenge and learning opportunity, right?
Limiting the HTML to only a single element allows us to push the limits of CSS to discover new techniques that can save us time and bytes, especially in those situations where you might not have direct access to modify HTML, like when you’re working in a CMS template. Don’t look at this as an over-complicated exercise. It’s an exercise that challenges us to leverage the power and flexibility of CSS.
The number of design resources and tools available on the market is increasing at a pace faster than one can count to 100. This makes it increasingly challenging to find the tools or resources you want and need to stay ahead of the competition.
To lend a helping hand, we’ve evaluated numerous free and premium resources and tools for designers and small business owners.
The variety of web design tools and resources featured in our list includes:
Website Builders – which you can use to quickly and easily create landing pages and multi-page websites, all without the frustration of hitting your head against the desk.
WordPress Themes – which allow you to build complex websites and e-stores known for their high conversion rates.
WordPress Plugins – which enable the incorporation of otherwise challenging-to-develop functionalities, helping your websites stand out.
Vector Illustrations – which can transform a dull website into a captivating one.
Font Identifiers – which help you identify appealing fonts used by brands like Nike or Hilton, allowing you to incorporate them into your own web projects.
More than half of the web design resources and tools listed here are either free or offer a free version or trial. They are presented in order of discussion:
What Do These 15 Best Web Design Tools & Resources for Designers and Small Businesses Have in Common?
They Exude Quality. The moment you install or download and begin using any of them, you’ll notice there’s something distinctively premium about them. This “special” quality is reflected in how effortless each interaction with the tool or resource feels.
They Are User-Friendly. From installation to downloading, using, or editing, all interactions and features are thoughtfully placed and designed.
They Deliver Value. Utilizing these tools will enable you to complete web design projects more quickly and enhance the aesthetics of your deliverables. By incorporating them into your workflow, you’re likely to secure higher-paying projects.
The 15 Best Web Design Tools & Resources for Designers and Small Business Owners
To simplify your search, we have meticulously compiled essential information about these products, including key features, ratings, user reviews, and access to immediate customer support resources.
In Brief: Brizy is the best and most intuitive website builder for any agency or designer in the hunt for a white label option.
Click on the video to get a firsthand look at one of Brizy’s most popular templates in action.
Notably, its standout feature is the white label option that allows your clients to build websites using a builder that you own, which prominently showcases your brand.
There’s much more to appreciate about Brizy. Highlights include:
Brizy’s intuitiveness: each tool or option is readily available precisely when and where you need it.
The capability to directly edit text, images, or any content in place.
The elimination of the need to deal with content creation through a disjointed sidebar, a common inconvenience with many other builders.
Brizy Builder also presents its users with a commendable array of demo/template/prebuilt websites. The “Natural Beauty” pre-built website, for instance, is attractive and inspirational, offering a robust foundation for websites aimed at beauty parlors, spas, and other service-oriented businesses.
Agencies and resellers will find value in the marketing integrations, reseller client management, and billing features.
Rating: 4.6/5 on Trustpilot
User Review:“I’ve been using Brizy for a while now, and their support team is outstanding. Quick, knowledgeable, and always ready to help. They’ve made our website design experience a breeze. I highly recommend Brizy for their exceptional support and user-friendly platform. Thank you, Brizy team!”
Types of Instant Support Materials: Support manual and YouTube videos
In Brief: The #1 Free Scheduling Online Software for Service-Oriented Businesses.
Click on the video for a firsthand look at one of Trafft’s most popular templates in action.
A highly valued feature among website building or enhancement tools is their adaptability; in this instance, the capability to operate in multiple languages. 70% of Trafft users highlight its powerful multilingual notifications system as the top feature.
The extensive library of prebuilt websites significantly contributes to the enjoyable experience of working with Trafft. The Trafft Barbershop template demonstrates the seamless integration of scheduling and marketing.
Exploring Trafft’s capabilities is a rewarding journey. Key features you’ll appreciate when you start using Trafft include:
The backend and frontend interface’s ease of use and innovative design.
The strength of the customization options available.
The versatility of the white label option.
The white label option is particularly advantageous for digital design agencies and web developers servicing clients. Another key user group includes individuals who require immediate confirmation for their appointments and schedules.
Rating: 5 stars on Captera
User Review:“Overall, Trafft makes a very good impression. For those looking to manage appointments within their business, Trafft is an indispensable tool. The UX and UI impressed us with their intuitiveness.”
Types of Instant Support Materials: Customer Support via Trafft’s ticketing system, Support manual, YouTube videos, email, and social media.
In Brief: WpDataTables is the best WordPress plugin for businesses and individuals tasked with managing complex data to create tables and charts.
Click on the video for a firsthand look at one of WpDataTables’ most popular uses in action.
The WordPress table plugin’s standout feature, its Multiple Database Connections, marks a significant advance in data management, transforming every table into a comprehensive data hub that can gather information from various databases and servers.
Aside from this, WpDataTables offers numerous features that facilitate the creation of custom, responsive, and easily editable tables and charts, streamlining data management processes. For instance, a financial team could effectively utilize the responsive top mutual fund template.
With WpDataTables, you will quickly realize you have access to:
A rich assortment of useful and practical functionalities within an intuitive interface.
Unmatched data management capabilities.
The expertise to adeptly handle complex data structures.
WpDataTables serves a broad spectrum of client needs effectively.
Separate database connections for engaging with specialized database systems.
Chart engines for visualizing data trends and comparisons, useful in marketing, finance, and environmental contexts.
Rating: 4.5/5 on WordPress.org
User Review:“My company deals with A LOT of data and spreadsheets. These spreadsheets act as master files shared among various departments, and I needed a way to link these spreadsheets with the data on WordPress. WP Data Tables provided THE PERFECT SOLUTION for bridging this gap.”
Types of Instant Support Materials: Support manual, YouTube videos, and the Facebook community.
In Brief: LayerSlider is the best WordPress slider plugin for creating websites with amazing animations. It is perfect for anyone who wants to save time and effort.
Click on the video for a firsthand look at one of LayerSlider’s most popular templates in action.
It’s common for a newly added tool or feature to quickly become a favorite among users.
The scroll effect in LayerSlider has become the plugin’s standout feature, prominently featured in the most recently released templates, including full-size landing pages and entire websites. Experience the Parallax City template to see the potential of these designs.
The versatility of LayerSlider contributes to its popularity, enabling the creation of simple sliders or slideshows as well as the integration of complex animated content.
You will appreciate:
The customizable interface, making LayerSlider feel like it was tailored specifically for you.
The access to connected online services, providing a plethora of visual content creation possibilities.
The Project Editor, simplifying the use of this plugin significantly.
LayerSlider particularly excels in creating content for marketing, offering astonishing effects for engaging customers through popups and banners.
User Review:“Great quality and importantly – great support!”
Types of Instant Support Materials: Manual, integrated help in Project Editor, ticket system, and questions or concerns sent via email
In Brief: Amelia is the ideal WordPress plugin for agencies and service-oriented businesses looking for an automated booking website.
Click on the video for a firsthand look at one of Amelia’s most popular templates in action.
Amelia’s automated notification system stands out as its prime feature. Users appreciate the simplicity with which they can categorize appointments as pending, approved, cancelled, rejected, or rescheduled. Additionally, sending birthday greetings or notices for upcoming events helps enhance client engagement and loyalty.
Amelia offers a variety of templates that are easily customizable. The massage therapist template, for instance, exemplifies a stylish and contemporary design.
Engaging with Amelia, you will notice:
The seamless navigation and innovative design of both the frontend and backend interfaces, highlighting its functionality and user-friendly approach.
The extensive customization options Amelia provides, along with its straightforward pricing plans.
The Amelia plugin is beneficial for any service-oriented business, including ticket sales agencies and event organizers. Developers and programming agencies will also find value in incorporating Amelia into their design toolsets.
Rating: 4.8 on Capterra
User Review:“Easy to learn, comprehensive (recurrent bookings, group bookings…), customization options for colors, fonts, various forms available, integrates well with Elementor. Ideal for the end user: logical, simple, includes customer accounts. Support is efficient and responsive.”
Types of Instant Support Materials: YouTube videos, Discord Group, and Support Manual
In Brief: Uncode is the #1 multiuse WordPress and WooCommerce theme for designers and creative agencies in search of an excellent solution for any project.
Click on the video for a firsthand look at one of Uncode’s most popular templates in action.
The extensive range of website building tools and options offered by this creative WooCommerce theme certainly contributes to its popularity. However, most users highlight the demo library as its standout feature. The demos exhibit remarkable attention to detail and can serve as significant sources of inspiration.
Choosing a single best feature would be difficult, so it’s best not to attempt it. The Portfolio Video Cover template from Uncode is among its most downloaded. Consider what possibilities it could open up for you.
You will quickly appreciate Uncode’s customization capabilities, the value of its demos and wireframes, and the exceptional customer support provided.
Uncode’s main user base includes:
Shop business owners who praise Uncode’s innovative WooCommerce features.
Agencies and freelancers who value the advanced customization options available.
Rating: 4.89/5
User Review:“Out of all the themes I’ve used, Uncode surpasses them all, in my opinion, and the latest update is simply astonishing! It offers the flexibility to create any design or type of site. It’s an exceptionally robust theme with numerous options and settings. I’m extremely satisfied with it! A heartfelt thanks!”
Types of Instant Support Materials: Support manual, YouTube videos, Facebook groups
In Brief: Slider Revolution is the best WordPress plugin for anyone looking to create jaw-dropping animated sliders.
Click on the video for a firsthand look at one of Slider Revolution’s most popular templates in action.
Slider Revolution’s premier feature is its capability to create visually stunning animated effects for WordPress sites without the need for additional effort or coding skills.
But the Slider Revolution plugin’s utility extends beyond just sliders. It allows you to:
Design home pages that immediately capture the attention of visitors.
Create visually appealing portfolios that demand a closer look.
Develop striking website sections that stand out.
If you’re seeking inspiration, explore Slider Revolution’s extensive library of over 250 templates. Many feature unique special effects not found on other websites and are fully optimized for various screen sizes. For instance, the Generative AI WordPress template is nothing short of revolutionary.
Slider Revolution is ideally suited for individual web designers, e-commerce sites, and small agencies.
Rating: 4.6/5 on Trustpilot
User Review:“Slider Revolution 6 is an incredibly fun tool to use, delivering exciting and fantastic looking animations to any site. Its versatility makes Slider Revolution an essential and powerful addition to any WordPress site!”
Types of Instant Support Materials: Support manual and YouTube videos
In Brief: The best resource for designers looking for premier illustrations that feature exceptional attention to detail.
Click on the video for a firsthand look at one of GetIllustrations’ most popular icons in action.
Is GetIllustrations’ top feature its collection of 21,500 vector illustrations, the free updates for one year, or the addition of new illustration packs every week? In fact, it’s the combination of all three.
With over 40 categories to explore, each filled with dozens, hundreds, and even up to 1,200 captivating illustrations, finding the perfect one for your needs is effortlessly easy. The illustrations are so well organized that browsing through them is a breeze.
Designed to cater to a wide range of clients, from students to businesses and developers, the collection includes pencil and basic ink illustrations, several 3D illustration categories, and specific themes like fitness, logistics, and eco illustrations, among others.
Exclusive to GetIllustrations, these illustrations enable users to craft truly unique projects.
Packs are available for purchase, with the Essential Illustrations pack being the most comprehensive. It includes Scenes, Avatars, and Elements, boasting a vast collection of one-color vector illustrations renowned for their depth and timeless appeal.
User Review:“Great product, and I appreciate the alternative to relying solely on Adobe stock for my app’s illustrations. Really impressed with the range of illustrations you offer!”
In Brief: The best online AI tool for generating and downloading full page websites through the use of prompt commands.
The Mobirise AI website builder’s premier feature allows for the generation of a website from a single prompt.
Provide a detailed description of your envisioned website, your offerings, and your target audience. The AI website builder leverages your input and intelligent algorithms to automatically create customized, aesthetically pleasing websites. This makes it an excellent choice for those without technical expertise or anyone in search of straightforward, efficient design solutions.
So advanced is Mobirise AI that it can understand instructions in Swahili, showcasing its impressive language capabilities.
Once the AI has generated a basic layout, you can use prompts to select a style, color scheme, typography, and more. Pre-generated text and content can also be edited to meet your specific requirements without the need for coding.
Important: The AI facilitates website creation but does not claim ownership of the final product.
Concerned about optimization for Google or mobile devices? Mobirise AI addresses all such concerns as well.
User Review:“Your team has excelled here! With just a brief paragraph from me, the complete concept of the site was grasped. What was generated, including images and text, perfectly aligned with our brand’s upscale messaging and class of images. Everything was executed exceptionally well.”
Types of Instant Support Materials: Support manual, YouTube videos, and the User Forum
In Brief: The most accurate font identifier available with no cost for searching.
There is a 90%+ chance that WhatFontIs will identify any free or licensed font you wish to find.
No other system matches this level of accuracy. WhatFontIs boasts a database of nearly 1 million free and commercial fonts, which is at least five times more than any other font identifier tool.
Users turn to WhatFontIs for various reasons, whether it’s to identify a specific font requested by a client or simply because they’ve encountered an appealing font and wish to know its name and where to find it. The search can be conducted regardless of the font’s publisher, producer, or foundry.
Just drag, drop, and upload a clear font image.
An AI-powered search engine swiftly identifies the font along with over 50 similar options.
The results will indicate where to download the free font and where to purchase a commercial one.
For the tool to accurately identify your font, the submitted image must be of high quality, and letters in cursive fonts should be separated. If these conditions are not met, the tool may not correctly identify the font.
User Review:“I found my font within seconds. Many thanks.”
In Brief: Blocksy is the #1 free WordPress theme for building beautiful, lightweight websites in 2024.
Click on the video for a firsthand look at one of Blocksy’s most popular templates in action.
Blocksy’s leading feature is evenly split among its user-friendly header and footer builder, WooCommerce integration with its extensive features, Gutenberg compatibility, and the advanced hooks system equipped with display conditions.
In other words, there’s hardly anything that you won’t find utterly impressive. (The high customer rating supports this assertion).
Moreover, Blocksy is available for free!
You’ll quickly come to realize that Blocksy:
Leverages the latest web technologies.
Delivers exceptional performance.
Seamlessly integrates with the most popular plugins.
Blocksy is versatile enough to create any type of website across any niche.
The demos of Blocksy are not just visually appealing but also highly useful for website building. The Daily News creative demo ranks among the top 5 most utilized.
Rating: 5/5 on WordPress.org
User Review:“An exceptional theme supported by a world-class customer service team! The responsiveness and helpfulness of the customer service team make this the best WordPress theme I’ve ever used. 🚀 With a wealth of features, I highly recommend this theme.”
Types of Instant Support Materials: Support manual, YouTube videos, and an easily navigable documentation section
In Brief: Total is the #1 WordPress theme for web designers and developers who want the flexibility to be able to design from scratch.
Click on the video for a firsthand look at one of Total’s most popular templates in action.
Total’s versatility is undoubtedly its standout feature, making it a comprehensive toolkit of design options, which justifies its name perfectly. Additionally, its exceptional support distinguishes Total from many other themes.
Engaging with Total will quickly reveal:
That Total is optimized for speed, with possibilities for further optimization to enhance performance.
An abundance of settings, numerous page builder element options, a font manager, custom post types, and more.
Support for dynamic templates for posts and archives.
The prebuilt sites offered by Total are of superior quality. Bolt, known for its minimalistic design, is among the top 5 most downloaded and utilized, adaptable to a variety of purposes.
Total is designed to meet the needs of beginners, developers, DIY enthusiasts, and essentially anyone in need of a flexible and powerful website solution.
Rating: 4.86/5
User Review:“I have utilized this theme across multiple sites, and the developer consistently keeps it updated and in excellent condition. Highly recommended!”
Types of Instant Support Materials: Support Manual
In Brief: Essential Grid is the best WordPress Gallery Plugin for designers and businesses who want to create a breathtaking gallery.
Click on the video for a firsthand look at one of Essential Grid’s most popular grid skins in action.
The hallmark of Essential Grid, its array of over 50 unique grid skins, is undoubtedly its standout feature.
This plugin offers significant value, as few web designers or developers would choose to create a gallery from scratch when such alternatives are available.
Using an Essential Grid gallery skin, you can easily achieve the desired gallery layout, and you might even find yourself enamored with a layout you hadn’t anticipated. For instance, the Funky Instagram cobbled layout social media grid might present an entirely new perspective on gallery design.
You’ll quickly appreciate Essential Grid’s ability to save time and effectively organize content streams.
Rating: 4.7/5 on Trustpilot
User Review:“An excellent plugin, operates flawlessly. Had some questions that needed answering, and the support team was incredibly friendly and helpful, not stopping until all issues were resolved and I was fully satisfied.”
Types of Instant Support Materials: Support manual and YouTube videos
In Brief: WoodMart is the ideal WooCommerce theme for designing niche ecommerce websites.
Click on the video for a firsthand look at one of WoodMart’s most popular templates in action.
A quick visit to the WoodMart website immediately highlights its standout feature: the custom layouts for shop, cart, and checkout pages are so impressively designed and realistic that you might momentarily forget you’re viewing a demo and attempt to place an order.
WoodMart is distinguished by numerous appealing aspects:
A vast array of options available for customization.
The ease with which layouts can be customized to add necessary branding, despite their near-perfect initial design.
The Theme Settings Search function and performance optimization features significantly save time.
Options like “Frequently Bought Together,” “Dynamic Discounts,” and social media integrations are highly favored by store owners and marketers.
Additionally, WoodMart offers a White Label option for further customization.
Identifying the most popular demos can be challenging, as many enjoy widespread usage. However, the Event Agency demo stands out as one of the top 5 most downloaded for clear reasons.
Rating: 4.93/5
User Review:“An excellent theme that’s easy to use and optimized for page speed insights. I’m very satisfied with this purchase.”
Types of Instant Support Materials: Support Manual and YouTube Videos
In Brief: XStore is the best WooCommerce theme for anyone looking to easily build a high-converting online store.
Click on the video for a firsthand look at one of XStore’s most popular pre-built websites in action.
XStore is clearly designed for shop owners and those aspiring to be, with its range of ready-made stores (pre-built websites) having long been a popular feature. However, the newly introduced selection of Sales Booster features has rapidly become the most celebrated aspect.
You’ll quickly come to value the Builders Panel and the intuitive XStore Control Panel, both of which offer extensive flexibility for building and customizing your store.
The immediate advantage provided by the pre-built websites is evident from the start. Opting for a pre-built website like the XStore Grocery Store, you’ll see how swiftly you can have an e-store operational.
Beyond the ready-to-use layouts, XStore grants you immediate access to the Single Product Builder, Checkout Builder, Cart Builder, Archive Products Builder, and 404 Page Builder, enriching your site-building toolkit.
Rating: 4.87/5
User Review:“For those seeking a WordPress theme that perfectly blends style with functionality, XStore is the answer. Its elegant design, straightforward customization options, and flawless WooCommerce integration make it a standout in the theme market.”
Types of Instant Support Materials: Support Manual and YouTube Videos
The best web design tools & resources all share several things in common.
They’re easy to use and set up.
They give a finished website a competitive edge in terms of both design and functionality.
They have excellent customer support.
Many of them have a free version, or as a minimum come with a lot of information so that you won’t suspect that you bought a completely different product from the one they described.
Are you looking to create a WordPress-powered website that serves as a classified platform? If so, you’re in the right place. This article showcases more than 20 WordPress themes, carefully selected to cater to a variety of classified site needs. Whether you’re a budding entrepreneur, a small business owner, or simply someone keen on setting up an efficient online classifieds portal, these themes are just what you need.
Our list includes both free and premium options, ensuring there’s something for every budget and requirement. Each theme is designed with user-friendliness in mind, making your site not only visually appealing but also easy to navigate and use.
Dive in and find the ideal WordPress theme to launch or enhance your classified website.
Best Listing is designed for professional directory websites. It offers a straightforward directory builder, making it easy to manage different types of directories. You’ll find features like a multi-directory system, unlimited custom fields, and ranked featured listings. Importing data is simple with CSV import capabilities.
Additionally, it includes classified ads functionality, perfect for starting a classified listing business. The responsive design ensures your site looks great on all devices, making it a great choice for launching a directory listing business.
The Classified Ads theme is notable for its striking red, multi-purpose HTML5 layout. It’s effective for classified ad websites, providing a responsive 2-column layout. Customize it with your own ad button text, header, and background https://assets.hongkiat.com/uploads/classified-wordpress-themes/colors.
The premium version includes PayPal integration for paid ads, Adsense JavaScript code, customizable user form fields, and various payment gateways. It’s browser-friendly and offers a user-friendly platform for classified ad listings.
For those looking to build a diverse range of sites, from Yellow Pages to real estate agencies, the Classified Ads Directory Theme is a perfect choice. It includes Open Street Maps support, making it ideal for real estate and location-based services. Check out its versatility in the demo at wpdirectorykit.com.
This theme serves various purposes, whether for realtors, brokers, hotels, or rental services, offering a comprehensive solution for directory-based websites.
Classified Listings is dynamic, suitable for a wide range of online classified ads platforms. Whether for real estate, jobs, or products, it offers an easy-to-use interface for posting and searching ads. The front-end submission system is user-friendly, perfect for a community-driven marketplace.
Its advanced search and filtering help users find listings easily. Customizable templates provide a unique look for each listing, and monetization options are available for revenue generation. It’s responsive, ensuring a smooth experience across devices.
CLClassified reinvents the classified ad website experience. Suitable for both beginners and experienced users, it focuses on the essentials of a classified ads site. This theme supports multiple revenue streams, including ad promotions and a store facility. It offers extensive customization, with Gutenberg compatibility for easy editing and unlimited color options.
Features like the WordPress Live Customizer and One-Click Demo Importer make setup a breeze. User-friendly elements like Ajax filters and autocomplete search enhance the overall experience. This fully responsive, mobile-friendly theme is also cross-browser compatible and supports unlimited custom fields.
Listdomer is tailored for listing, directory, and classified websites, adaptable to various industries like events and real estate. Integrated with the Listdom plugin, it offers diverse views such as grid, list, and masonry.
This theme is easy to use, requiring no technical knowledge, and features a powerful, customizable search form. Designing pages is effortless with Elementor’s drag-and-drop functionality. Its mobile compatibility and SEO-friendly design make it a top performer in search engine rankings.
ListingHive serves as a multipurpose theme for any directory or listing website. Ideal for a business directory, job board, or a classifieds site, its flexibility accommodates various listing types.
The theme’s user-friendly design simplifies site building and management, making it an efficient choice for diverse projects.
NexProperty is designed for real estate directory listings, suitable for agencies, realtors, and classified ads. It’s tailored for Elementor, allowing visual customization across the board.
The theme includes features for managing listings, categories, and fields, multimedia integration, and messaging support. Fully compatible with Elementor, it offers a visually appealing, customizable experience. Open Street Maps support enhances its real estate listing capabilities. With demo data for real estate and car dealerships, NexProperty provides a complete solution for classified directory or listing businesses.
TerraClasic is the first free theme for the TerraClassifieds plugin, ideal for classified ads. It emphasizes user-friendly ad posting and uses a classic contact form for advertiser communication. Future updates are set to include various payment methods and auction features.
This theme is a great starting point for those new to classified ads, offering a straightforward, functional solution.
AdForest distinguishes itself as a premium Classified Ads WordPress theme with a modern front-end UI. It offers various color options and WordPress functionalities, including Google Map integration.
Built for the modern era, AdForest is quick to set up and easy to customize. Constructed with HTML5, CSS3, Bootstrap 5, and JQuery 3.1, it promises a robust and responsive design. Extensive documentation is provided for easy setup.
AdForest is designed to meet all classified ad posting needs, making it an excellent choice for those aiming to elevate their classified business.
Adifier transforms the classified ads marketplace with its comprehensive and user-focused design. It’s built from the ground up to provide essential features, such as multiple ad types, advanced custom fields, and various monetization options.
Users can choose from a range of payment methods, enjoy a custom messaging and reviews system, and participate in auctions. A standout feature is the ad comparison based on custom fields, elevating the user experience.
Adifier also integrates social login and a user dashboard, making it a complete solution for a sophisticated classified ads marketplace.
For car dealership businesses, Avtorai emerges as the top pick. Tailored specifically for Car Dealer, Dealership, Car Listing, Classified, and Car Rental sites, this theme is responsive, retina-ready, and built on Bootstrap 4.
With features like 4 Home Page layouts, WPBakery Page Builder, and WooCommerce readiness, it’s highly functional. Avtorai supports one-click installation, touch-enabled carousels, Slider Revolution, and includes a variety of listing features like paid listings, a Trim Cars Database, and multiple layout options.
The frontend dashboard, shortcode elements, and compatibility with Google fonts, WPML, and Contact Form 7, make it versatile for car dealership websites.
Buyent stands as a fast and modern option for classified ads and directory websites. Its integration with the Elementor page builder makes homepage customization straightforward.
The theme shines with its monetization features, such as selling ads and ad banner slots. For a classified business looking to stand out, Buyent offers a user-friendly design and efficient monetization capabilities, making it an excellent choice.
CarSpot, a robust theme for car dealerships, caters to all sizes of automotive businesses. It includes key ad posting features, a modern gallery, a review system, and a car comparison feature.
With 6 premade demos, a one-click demo importer, and various customization options, CarSpot is ideal for creating a feature-rich website. Its exceptional vehicle search functionality and ad features are designed to boost business sales, making it a comprehensive choice for anyone in the car dealership industry seeking a strong online presence.
Clasifico offers a clean and modern solution for classified advertising and online listing sites. It’s highly responsive and customizable, featuring three homepage layouts, two header styles, and over 15 inner page variations. This flexibility allows for extensive personalization and design choices, making Clasifico a versatile choice for various advertising or online listing sites.
Classiads stands as a top choice in the classified ads website arena, thanks to its status as the best-selling theme on ThemeForest. Renowned for its flexibility and feature-rich environment, it offers a user-friendly interface that makes it a reliable choice for building a classified ads platform.
Ideal for those seeking an efficient and trusted solution, Classiads is a standout in its category.
Turning to Classiera, this theme is known for its modern and responsive design, making it a popular choice for classified ads websites. It’s SEO optimized and offers a range of features including various ad types and pricing plans.
With its ability to cater to diverse classified ad needs, Classiera is a versatile and professional option for creating classified ads websites.
ClassifiedPro, using the CubeWP Framework, offers unique versatility with support for three specific post types: General items, Automotive, and Real Estate. This theme suits both local classified websites for cash transactions and pickup, and re-commerce sites that support online transactions and shipping.
ClassifiedPro stands out for its dual functionality, enhancing revenue generation opportunities and making it a top choice for diverse classified ad requirements.
The Classified Ads WordPress Theme provides a comprehensive solution for classified ads websites. It’s designed to be both beautiful and powerful, with a fast performance that aims to meet all classified ad needs. Their team offers free installation of the latest version and demo content within 48 hours, providing a hassle-free setup and making it an attractive option for a user-friendly classified ads website.
Classify is a responsive and SEO-optimized Classified Advertising Theme for WordPress. Developed using Bootstrap, HTML5, and CSS3, it promises a modern and durable framework.
It’s versatile, particularly excelling in Directory and Classified listings, and integrates social media logins for enhanced user convenience. With lifetime support, free installation, and updates, Classify is a sustainable choice for long-term website development.
Classima, created with creativity and modern design, is ideal for classified listing and directory websites. Optimized for Gutenberg and built with Elementor Page Builder, it offers customizable features like multiple homepages and header styles. It supports various ad layouts, Ajax filters, and autocompletion for searches, making it a comprehensive and user-friendly theme.
Classima is responsive, mobile-friendly, and supports multiple payment options, ensuring it meets the needs of classified ads websites.
Hourty Complex Tower, designed for real estate agents and agencies, excels in showcasing single properties and apartment complexes. Its design is both attractive and functional, meeting a variety of real estate needs such as rentals, sales, and commercial projects.
With four homepage demos, Hourty is efficient for creating quality real estate websites. Its comprehensive functionalities and plugins are time-savers, making it a versatile choice for realtors, investors, and construction companies.
Knowhere Pro is a diverse directory WordPress theme, ideal for a city portal or specialized directories like restaurants, cafes, hotels, job boards, or classified ads. Its flexibility caters to a range of directory styles, from property listings to job portals.
With its diverse page blocks and sections, Knowhere Pro offers a universal solution for any directory-based website.
Lisfinity focuses on classified ads, offering a streamlined platform specifically for this niche. Its design and functionality are tailored to enhance classified listings, providing a focused solution for this sector.
Motodeal is a multipurpose theme for vehicle dealerships. It suits a variety of vehicles, from luxury cars to bikes, trucks, yachts, and even agricultural vehicles. This theme’s versatile design makes it an excellent platform for any vehicle dealership, providing an attractive and functional website.
Trade stands out for its customization and flexibility. It incorporates the Visual Composer Drag & Drop Page Builder and an extensive options panel. Built on Bootstrap 3.x, it offers a responsive, mobile-first experience. With unlimited color choices, Google Fonts, and cross-browser compatibility, Trade is user-friendly. Its detailed documentation and child theme inclusion simplify customization.
Additionally, it’s translation ready and WPML compatible, appealing to a broad audience.