How to do a dashed border with a gradient and a border-radius in css?
Closest I could get to was with an svg background-image:
This looks fine if you have a 100px to 100px box (see 2cnd picture). But the border should fit on any box. As soon as you increase width and height to another aspect ratio, it breaks, especially the border-radius (see 1st picture).
i also tried several different solutions, but nothing works as expected...
data:image/s3,"s3://crabby-images/c1ff6/c1ff646087a3faf54c28d44481a9d016695cd65b" alt="No description"
36 Replies
data:image/s3,"s3://crabby-images/3ac81/3ac81fb8645f93e04f0c684be1d382c1be621049" alt="No description"
I would try a pseudo element with a 2px dashed transparent border, bg clip/origin to the border-box and mask padding-box exclude. Then the bg image of conic gradient . I'm on mobile and I'm trying not to hand out solutions anymore anyways (no one learns if they just copy and paste) but lmk if you have questions.
Does this approach only work with a white background? In my case the background has to be transparent
One method is to place the "border" gradient on a pseudo element and use a mask to only allow the edges to be visible, leaving the content area transparent.
the responsive squishing and stretching of the corners can be fixed by using
border-image
instead of background-image
This is how i would do it https://codepen.io/tok124/pen/ZYEYxEZ
Just make sure to include the vendor prefixes so that it works on chromium based browsers
data:image/s3,"s3://crabby-images/3ca8c/3ca8c4e0480f71d2377b1918f9275c5710271049" alt="No description"
i have autoprefixer enabled in codepen so you don't really see the prefixes unless you compile the code
Yea the mask will make it transparent except for the gradient in the borderbox “padding-box exclude “
Unprefixed for
mask
is pretty well supported ! The purple border shows versions when prefix was required. Although with autoprefixer it doesn’t hurt to include anyway ✨data:image/s3,"s3://crabby-images/ad635/ad635f622df93d318b42cab35aec0123f560e53d" alt="No description"
Yeah, but you need it for the mask-composite 🙂
Aha that’s right , how quickly I forget ; just used it a few months ago and had to prefix for exclude
Yeah 😄
Switching to
border-image
doesn't show anything anymore, do i need to adapt the svg?That really looked promising, but as soon as you change the width/height the apperance breaks :/
data:image/s3,"s3://crabby-images/84bbf/84bbf91a2269c702543f5a873ceefc00cf4c5238" alt="No description"
I also think
border-image
doesn't account for border-radius
?Strange... Works fine for me...
Correct
@oemerSee here, works perfectly fine
Maybe you changed the width/height of the pseudo element and thats why it looked messed up...
But anyway, if it does look messed up on any width/height, just adjust the mask gradient
If you compare it to a normal css border, the line lenght and distance is different:
data:image/s3,"s3://crabby-images/410f3/410f35ac9fb6333dc823617fd9f437bb6917fdc0" alt="No description"
But don't I have to adjust it for any case where i use this style then? The card could have different dimensions for different use cases
@Tok124 (CSS Nerd)
I really think this is onlypossible with javascript, where you get the dimensions of the container and do some calculations with it.. which i would like to avoid because of SSR
Yep you can’t use border radius on border-image
Yeah. this is because i have used % unit. Idk if there is any other unit that would work better, perhaps vmin/vmax unit, i never really using these units, but maybe they could be useful here, otherwise you may have to go with em/rem/px unit
the radius of the corner is already included in the SVG image so
border-radius
wouldn't be necessary. or am i missing something?But yeah if you have a box like this, then 10% on the width is greater than 10% on the height, as you can probably imagine, that's why you end up with this problem, it's due to the % unit in mask gradient
data:image/s3,"s3://crabby-images/f7a98/f7a98ed949dba156e543d0c45b954242e37d5004" alt="No description"
But yeah if no other unit works you may need to use JS to calculate the correct % amount for the height
I see, if play around with values, it changes yeah
Yeah 🙂
would love to avoid javascript, but with px values it seems to be tricky
Otherwise you can use a conic-gradient, that works too, it may work a little better, but then there is a new problem, the corners may look kinda strange...
data:image/s3,"s3://crabby-images/12301/123013a9a32fb68b3dbf1cff388158deffacb3c3" alt="No description"
also interesting, but there are always buggy artefacts depending on which size you go for unfortunatelyy.. not only the corners
data:image/s3,"s3://crabby-images/8d309/8d309c4b6d68dd5b3fe5e3b9b01762bf90d72e17" alt="No description"
data:image/s3,"s3://crabby-images/9484a/9484aeb6d966f20a59282d1e9bc7c8e8427ca87e" alt="No description"
Hopefully CSS will support this in the future
Regardless, I learned a lot here, you guys are awesome
I got such an interesting result with a repeating conic gradient too, it looked like a pattern
For the future take a look at the
border-area
value for background-clip
. It's only in Safari right now but hopefully Chrome and Firefox will start to play around with it this year. To be honest it's not the best solution for this (it's a bit clunky and hacky to reuse background and set the border to transparent) but it's something we got a browser to actually work on... https://github.com/w3c/csswg-drafts/issues/9456#issuecomment-2325278729
also tinkered with your svg a bit to get it to work in border-image
as expected. Needed to remove the viewbox to make it a fixed size: https://codepen.io/jsnkuhn/pen/RNwNYdzOh wow, that looks really good! I am honestly baffled how deeply you guys understand CSS. I just checked all diffs. You are using a fixed height, not percent. You added
32 / 32px round
to the border-image
prop. You removed the viewBox
and also vector scaling. If you miss one of those, it just breaks.
I am really wondering how you learned this? Do you guys just do trial and error or is this your profession? It seems really complicated and very detailed.
In my company i am the kind of "CSS guy" but i feel really noobish compared to you guysupdated the pen I made above to use the
border-image-*
long hands to give a better idea of what's going on. MDN article is probably the best place to find more info: https://developer.mozilla.org/en-US/docs/Web/CSS/border-image