CSS: не наносите несколько теней друг на друга
Есть ли способ не наносить несколько теней друг на друга? Я пытаюсь использовать тень для имитации обводки вокруг элемента. Мне нужно было использовать тень, потому что она работала, когда у нас были маски или изображения с прозрачными пятнами, и я не мог найти другой способ сделать то же самое. Другой вариант использования цвета фона не работал для изображений PNG с прозрачностью.
Проблема в том, что когда я определяю, например, тень для левой и верхней сторон, вместо того, чтобы делать хорошее перекрытие, он фактически применяет дополнительную тень к элементу, где было перекрытие. Это заканчивается с плоскими краями вместо закругленных, как они должны были быть. Обратите внимание, что хотя это округленное изображение, мы позволяем пользователям определять свои собственные маски и загружать файлы png, чтобы оно могло иметь практически любую форму.
Вот пример проблемы jsfiddle: https://jsfiddle.net/a67fqne2/17/
Сам код:
<div style="width: 391.6px; height: 369.6px; left: 239.6px; top: 28.9px; position: absolute; cursor: default; z-index: 52;">
<div class="imageOuterStyleWrapper">
<div class="imageMiddleStyleWrapper" style="filter: drop-shadow(rgb(139, 69, 19) 19.6px 0px 0px) drop-shadow(rgb(139, 69, 19) 0px 19.6px 0px) drop-shadow(rgb(139, 69, 19) -19.6px 0px 0px) drop-shadow(rgb(139, 69, 19) 0px -19.6px 0px); width: calc(100% - 39.16px); height: calc(100% - 39.16px); padding: 19.58px;">
<div class="imageInnerStyleWrapper" style="clip-path: url("#Mask-2");"><img style="width: 100%; height: 133.083%; left: 0%; top: -16.5415%;" src="https://plicimgstaging-d6d.kxcdn.com?url=https%3A%2F%2Fplic-staging.s3.amazonaws.com%2Fphotos%2F4dbf8644-0751-4e9f-afaf-f5c4a00f6bae%2Foriginal%2FAbdulKarim-Naeemah-01195-00007.JPG%3FX-Amz-Algorithm%3DAWS4-HMAC-SHA256%26X-Amz-Credential%3DAKIAIZ3GLWGI2FROIRRA%252F20180712%252Fus-east-1%252Fs3%252Faws4_request%26X-Amz-Date%3D20180712T165503Z%26X-Amz-SignedHeaders%3DHost%26X-Amz-Expires%3D604800%26X-Amz-Signature%3Dd32c01d8dd7f82206b5f15028fc90a1ff6d5aadd71738648d8e477799e27816b&op=resize,rotate&deg=auto&mode=clip&filter=bilinear&q=90&w=450&plic-ref=plic-books-dev&sig=4ee24d37aa0356fe61c9500d370ca81944338fe5"></div>
</div>
</div>
</div>
<svg width="0" height="0" id="globalMaskDefinitions">
<defs>
<clipPath id="Mask-2" clipPathUnits="objectBoundingBox">
<circle x="0" y="0" width="1" height="1" cx=".5px" cy=".5px" r=".5px"/>
</clipPath>
</defs>
</svg>
1 ответ
Проблема связана с формой, которая в данном случае является кругом, поэтому все эти drop-shadow
с разных позиций не будет сохранять форму круга.
Вот лучшая иллюстрация проблемы, если вы измените цвета:
div,
img {
width: 100%;
height: 100%;
box-sizing: content-box;
}
.imageOuterStyleWrapper {
width: 391.6px;
height: 369.6px;
}
.imageMiddleStyleWrapper {
filter: drop-shadow(red 19.6px 0px 0px)
drop-shadow(blue 0px 19.6px 0px)
drop-shadow(green -19.6px 0px 0px)
drop-shadow(yellow 0px -19.6px 0px);
width: calc(100% - 39.16px);
height: calc(100% - 39.16px);
padding: 19.58px;
}
.imageInnerStyleWrapper {
clip-path: url("#Mask-2");
}
<div class="imageOuterStyleWrapper">
<div class="imageMiddleStyleWrapper">
<div class="imageInnerStyleWrapper" style=""><img src="http://wiesmann.codiferes.net/share/bitmaps/test_pattern.svg"></div>
</div>
<svg width="0" height="0">
<defs>
<clipPath id="Mask-2" clipPathUnits="objectBoundingBox">
<circle x="0" y="0" width="1" height="1" cx=".5px" cy=".5px" r=".5px"/>
</clipPath>
</defs>
</svg>
Но если форма была квадратной, у вас не будет этой проблемы, и она будет отлично работать:
div,
img {
width: 100%;
height: 100%;
box-sizing: content-box;
}
.imageOuterStyleWrapper {
width: 391.6px;
height: 369.6px;
}
.imageMiddleStyleWrapper {
filter: drop-shadow(red 19.6px 0px 0px)
drop-shadow(blue 0px 19.6px 0px)
drop-shadow(green -19.6px 0px 0px)
drop-shadow(yellow 0px -19.6px 0px);
width: calc(100% - 39.16px);
height: calc(100% - 39.16px);
padding: 19.58px;
}
<div class="imageOuterStyleWrapper">
<div class="imageMiddleStyleWrapper">
<div class="imageInnerStyleWrapper" style=""><img src="http://wiesmann.codiferes.net/share/bitmaps/test_pattern.svg"></div>
</div>
Идея исправления состоит в том, чтобы использовать SVG и полагаться на штрих, чтобы получить то, что вы хотите:
<svg width="200" height="200">
<defs>
<pattern id="img" patternUnits="userSpaceOnUse" width="200" height="200">
<image xlink:href="http://wiesmann.codiferes.net/share/bitmaps/test_pattern.svg" x="0" y="0" width="200" height="200" />
</pattern>
</defs>
<circle fill="url(#img)" cx="100px" cy="100px" r="80px" stroke="rgb(139, 69, 19)" stroke-width="10"/>
</svg>
Я использовал круг, но вы можете легко использовать любую форму:
<svg width="200" height="200">
<defs>
<pattern id="img" patternUnits="userSpaceOnUse" width="200" height="200">
<image xlink:href="http://wiesmann.codiferes.net/share/bitmaps/test_pattern.svg" x="0" y="0" width="200" height="200" />
</pattern>
</defs>>
<polygon points="190,10 60,100 160,190" fill="url(#img)" stroke="rgb(139, 69, 19)" stroke-width="10" />
</svg>