summaryrefslogtreecommitdiff
path: root/Source/WebCore/css/CSSCrossfadeValue.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'Source/WebCore/css/CSSCrossfadeValue.cpp')
-rw-r--r--Source/WebCore/css/CSSCrossfadeValue.cpp171
1 files changed, 101 insertions, 70 deletions
diff --git a/Source/WebCore/css/CSSCrossfadeValue.cpp b/Source/WebCore/css/CSSCrossfadeValue.cpp
index 26472263a..c324dc112 100644
--- a/Source/WebCore/css/CSSCrossfadeValue.cpp
+++ b/Source/WebCore/css/CSSCrossfadeValue.cpp
@@ -11,10 +11,10 @@
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
- * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
- * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
@@ -29,6 +29,7 @@
#include "AnimationUtilities.h"
#include "CSSImageValue.h"
+#include "CachedImage.h"
#include "CachedResourceLoader.h"
#include "CrossfadeGeneratedImage.h"
#include "ImageBuffer.h"
@@ -44,31 +45,59 @@ static inline double blendFunc(double from, double to, double progress)
return blend(from, to, progress);
}
-static bool subimageKnownToBeOpaque(CSSValue& value, const RenderElement* renderer)
+static bool subimageKnownToBeOpaque(const CSSValue& value, const RenderElement& renderer)
{
- if (value.isImageValue())
- return toCSSImageValue(value).knownToBeOpaque(renderer);
+ if (is<CSSImageValue>(value))
+ return downcast<CSSImageValue>(value).knownToBeOpaque(&renderer);
- if (value.isImageGeneratorValue())
- return toCSSImageGeneratorValue(value).knownToBeOpaque(renderer);
+ if (is<CSSImageGeneratorValue>(value))
+ return downcast<CSSImageGeneratorValue>(value).knownToBeOpaque(renderer);
ASSERT_NOT_REACHED();
return false;
}
+inline CSSCrossfadeValue::SubimageObserver::SubimageObserver(CSSCrossfadeValue& owner)
+ : m_owner(owner)
+{
+}
+
+void CSSCrossfadeValue::SubimageObserver::imageChanged(CachedImage*, const IntRect*)
+{
+ m_owner.crossfadeChanged();
+}
+
+inline CSSCrossfadeValue::CSSCrossfadeValue(Ref<CSSValue>&& fromValue, Ref<CSSValue>&& toValue, Ref<CSSPrimitiveValue>&& percentageValue, bool prefixed)
+ : CSSImageGeneratorValue(CrossfadeClass)
+ , m_fromValue(WTFMove(fromValue))
+ , m_toValue(WTFMove(toValue))
+ , m_percentageValue(WTFMove(percentageValue))
+ , m_subimageObserver(*this)
+ , m_isPrefixed(prefixed)
+{
+}
+
+Ref<CSSCrossfadeValue> CSSCrossfadeValue::create(Ref<CSSValue>&& fromValue, Ref<CSSValue>&& toValue, Ref<CSSPrimitiveValue>&& percentageValue, bool prefixed)
+{
+ return adoptRef(*new CSSCrossfadeValue(WTFMove(fromValue), WTFMove(toValue), WTFMove(percentageValue), prefixed));
+}
+
CSSCrossfadeValue::~CSSCrossfadeValue()
{
if (m_cachedFromImage)
- m_cachedFromImage->removeClient(&m_crossfadeSubimageObserver);
+ m_cachedFromImage->removeClient(m_subimageObserver);
if (m_cachedToImage)
- m_cachedToImage->removeClient(&m_crossfadeSubimageObserver);
+ m_cachedToImage->removeClient(m_subimageObserver);
}
String CSSCrossfadeValue::customCSSText() const
{
StringBuilder result;
- result.appendLiteral("-webkit-cross-fade(");
+ if (m_isPrefixed)
+ result.appendLiteral("-webkit-cross-fade(");
+ else
+ result.appendLiteral("cross-fade(");
result.append(m_fromValue->cssText());
result.appendLiteral(", ");
result.append(m_toValue->cssText());
@@ -78,142 +107,144 @@ String CSSCrossfadeValue::customCSSText() const
return result.toString();
}
-IntSize CSSCrossfadeValue::fixedSize(const RenderElement* renderer)
+FloatSize CSSCrossfadeValue::fixedSize(const RenderElement& renderer)
{
- float percentage = m_percentageValue->getFloatValue();
+ float percentage = m_percentageValue->floatValue();
float inversePercentage = 1 - percentage;
- CachedResourceLoader* cachedResourceLoader = renderer->document().cachedResourceLoader();
- CachedImage* cachedFromImage = cachedImageForCSSValue(m_fromValue.get(), cachedResourceLoader);
- CachedImage* cachedToImage = cachedImageForCSSValue(m_toValue.get(), cachedResourceLoader);
+ // FIXME: Skip Content Security Policy check when cross fade is applied to an element in a user agent shadow tree.
+ // See <https://bugs.webkit.org/show_bug.cgi?id=146663>.
+ auto options = CachedResourceLoader::defaultCachedResourceOptions();
+
+ auto& cachedResourceLoader = renderer.document().cachedResourceLoader();
+ auto* cachedFromImage = cachedImageForCSSValue(m_fromValue, cachedResourceLoader, options);
+ auto* cachedToImage = cachedImageForCSSValue(m_toValue, cachedResourceLoader, options);
if (!cachedFromImage || !cachedToImage)
- return IntSize();
+ return FloatSize();
- IntSize fromImageSize = cachedFromImage->imageForRenderer(renderer)->size();
- IntSize toImageSize = cachedToImage->imageForRenderer(renderer)->size();
+ FloatSize fromImageSize = cachedFromImage->imageForRenderer(&renderer)->size();
+ FloatSize toImageSize = cachedToImage->imageForRenderer(&renderer)->size();
// Rounding issues can cause transitions between images of equal size to return
// a different fixed size; avoid performing the interpolation if the images are the same size.
if (fromImageSize == toImageSize)
return fromImageSize;
- return IntSize(fromImageSize.width() * inversePercentage + toImageSize.width() * percentage,
- fromImageSize.height() * inversePercentage + toImageSize.height() * percentage);
+ return fromImageSize * inversePercentage + toImageSize * percentage;
}
bool CSSCrossfadeValue::isPending() const
{
- return CSSImageGeneratorValue::subimageIsPending(m_fromValue.get())
- || CSSImageGeneratorValue::subimageIsPending(m_toValue.get());
+ return CSSImageGeneratorValue::subimageIsPending(m_fromValue)
+ || CSSImageGeneratorValue::subimageIsPending(m_toValue);
}
-bool CSSCrossfadeValue::knownToBeOpaque(const RenderElement* renderer) const
+bool CSSCrossfadeValue::knownToBeOpaque(const RenderElement& renderer) const
{
- return subimageKnownToBeOpaque(*m_fromValue, renderer) && subimageKnownToBeOpaque(*m_toValue, renderer);
+ return subimageKnownToBeOpaque(m_fromValue, renderer)
+ && subimageKnownToBeOpaque(m_toValue, renderer);
}
-void CSSCrossfadeValue::loadSubimages(CachedResourceLoader* cachedResourceLoader)
+void CSSCrossfadeValue::loadSubimages(CachedResourceLoader& cachedResourceLoader, const ResourceLoaderOptions& options)
{
- CachedResourceHandle<CachedImage> oldCachedFromImage = m_cachedFromImage;
- CachedResourceHandle<CachedImage> oldCachedToImage = m_cachedToImage;
+ auto oldCachedFromImage = m_cachedFromImage;
+ auto oldCachedToImage = m_cachedToImage;
- m_cachedFromImage = CSSImageGeneratorValue::cachedImageForCSSValue(m_fromValue.get(), cachedResourceLoader);
- m_cachedToImage = CSSImageGeneratorValue::cachedImageForCSSValue(m_toValue.get(), cachedResourceLoader);
+ m_cachedFromImage = CSSImageGeneratorValue::cachedImageForCSSValue(m_fromValue, cachedResourceLoader, options);
+ m_cachedToImage = CSSImageGeneratorValue::cachedImageForCSSValue(m_toValue, cachedResourceLoader, options);
if (m_cachedFromImage != oldCachedFromImage) {
if (oldCachedFromImage)
- oldCachedFromImage->removeClient(&m_crossfadeSubimageObserver);
+ oldCachedFromImage->removeClient(m_subimageObserver);
if (m_cachedFromImage)
- m_cachedFromImage->addClient(&m_crossfadeSubimageObserver);
+ m_cachedFromImage->addClient(m_subimageObserver);
}
if (m_cachedToImage != oldCachedToImage) {
if (oldCachedToImage)
- oldCachedToImage->removeClient(&m_crossfadeSubimageObserver);
+ oldCachedToImage->removeClient(m_subimageObserver);
if (m_cachedToImage)
- m_cachedToImage->addClient(&m_crossfadeSubimageObserver);
+ m_cachedToImage->addClient(m_subimageObserver);
}
- m_crossfadeSubimageObserver.setReady(true);
+ // FIXME: Unclear why this boolean adds any value; for now keeping it around to avoid changing semantics.
+ m_subimagesAreReady = true;
}
-PassRefPtr<Image> CSSCrossfadeValue::image(RenderElement* renderer, const IntSize& size)
+Image* CSSCrossfadeValue::image(RenderElement& renderer, const FloatSize& size)
{
if (size.isEmpty())
- return 0;
+ return nullptr;
- CachedResourceLoader* cachedResourceLoader = renderer->document().cachedResourceLoader();
- CachedImage* cachedFromImage = cachedImageForCSSValue(m_fromValue.get(), cachedResourceLoader);
- CachedImage* cachedToImage = cachedImageForCSSValue(m_toValue.get(), cachedResourceLoader);
+ // FIXME: Skip Content Security Policy check when cross fade is applied to an element in a user agent shadow tree.
+ // See <https://bugs.webkit.org/show_bug.cgi?id=146663>.
+ auto options = CachedResourceLoader::defaultCachedResourceOptions();
+
+ auto& cachedResourceLoader = renderer.document().cachedResourceLoader();
+ auto* cachedFromImage = cachedImageForCSSValue(m_fromValue, cachedResourceLoader, options);
+ auto* cachedToImage = cachedImageForCSSValue(m_toValue, cachedResourceLoader, options);
if (!cachedFromImage || !cachedToImage)
return Image::nullImage();
- Image* fromImage = cachedFromImage->imageForRenderer(renderer);
- Image* toImage = cachedToImage->imageForRenderer(renderer);
+ auto* fromImage = cachedFromImage->imageForRenderer(&renderer);
+ auto* toImage = cachedToImage->imageForRenderer(&renderer);
if (!fromImage || !toImage)
return Image::nullImage();
- m_generatedImage = CrossfadeGeneratedImage::create(fromImage, toImage, m_percentageValue->getFloatValue(), fixedSize(renderer), size);
-
- return m_generatedImage.release();
+ m_generatedImage = CrossfadeGeneratedImage::create(*fromImage, *toImage, m_percentageValue->floatValue(), fixedSize(renderer), size);
+ return m_generatedImage.get();
}
-void CSSCrossfadeValue::crossfadeChanged(const IntRect&)
+inline void CSSCrossfadeValue::crossfadeChanged()
{
- for (auto it = clients().begin(), end = clients().end(); it != end; ++it)
- it->key->imageChanged(static_cast<WrappedImagePtr>(this));
+ if (!m_subimagesAreReady)
+ return;
+ for (auto& client : clients())
+ client.key->imageChanged(this);
}
-void CSSCrossfadeValue::CrossfadeSubimageObserverProxy::imageChanged(CachedImage*, const IntRect* rect)
+bool CSSCrossfadeValue::traverseSubresources(const std::function<bool (const CachedResource&)>& handler) const
{
- if (m_ready)
- m_ownerValue->crossfadeChanged(*rect);
-}
-
-bool CSSCrossfadeValue::hasFailedOrCanceledSubresources() const
-{
- if (m_cachedFromImage && m_cachedFromImage->loadFailedOrCanceled())
+ if (m_cachedFromImage && handler(*m_cachedFromImage))
return true;
- if (m_cachedToImage && m_cachedToImage->loadFailedOrCanceled())
+ if (m_cachedToImage && handler(*m_cachedToImage))
return true;
return false;
}
-PassRefPtr<CSSCrossfadeValue> CSSCrossfadeValue::blend(const CSSCrossfadeValue& from, double progress) const
+RefPtr<CSSCrossfadeValue> CSSCrossfadeValue::blend(const CSSCrossfadeValue& from, double progress) const
{
ASSERT(equalInputImages(from));
- RefPtr<StyleCachedImage> toStyledImage = StyleCachedImage::create(m_cachedToImage.get());
- RefPtr<StyleCachedImage> fromStyledImage = StyleCachedImage::create(m_cachedFromImage.get());
- auto fromImageValue = CSSImageValue::create(m_cachedFromImage->url(), fromStyledImage.get());
- auto toImageValue = CSSImageValue::create(m_cachedToImage->url(), toStyledImage.get());
+ if (!m_cachedToImage || !m_cachedFromImage)
+ return nullptr;
- RefPtr<CSSCrossfadeValue> crossfadeValue = CSSCrossfadeValue::create(std::move(fromImageValue), std::move(toImageValue));
+ auto fromImageValue = CSSImageValue::create(*m_cachedFromImage);
+ auto toImageValue = CSSImageValue::create(*m_cachedToImage);
- double fromPercentage = from.m_percentageValue->getDoubleValue();
+ double fromPercentage = from.m_percentageValue->doubleValue();
if (from.m_percentageValue->isPercentage())
fromPercentage /= 100.0;
- double toPercentage = m_percentageValue->getDoubleValue();
+ double toPercentage = m_percentageValue->doubleValue();
if (m_percentageValue->isPercentage())
toPercentage /= 100.0;
- crossfadeValue->setPercentage(CSSPrimitiveValue::create(blendFunc(fromPercentage, toPercentage, progress), CSSPrimitiveValue::CSS_NUMBER));
- return crossfadeValue.release();
+ auto percentageValue = CSSPrimitiveValue::create(blendFunc(fromPercentage, toPercentage, progress), CSSPrimitiveValue::CSS_NUMBER);
+
+ return CSSCrossfadeValue::create(WTFMove(fromImageValue), WTFMove(toImageValue), WTFMove(percentageValue), from.isPrefixed() && isPrefixed());
}
bool CSSCrossfadeValue::equals(const CSSCrossfadeValue& other) const
{
- return equalInputImages(other)
- && compareCSSValuePtr(m_percentageValue, other.m_percentageValue);
+ return equalInputImages(other) && compareCSSValue(m_percentageValue, other.m_percentageValue);
}
bool CSSCrossfadeValue::equalInputImages(const CSSCrossfadeValue& other) const
{
- return compareCSSValuePtr(m_fromValue, other.m_fromValue)
- && compareCSSValuePtr(m_toValue, other.m_toValue);
+ return compareCSSValue(m_fromValue, other.m_fromValue) && compareCSSValue(m_toValue, other.m_toValue);
}
} // namespace WebCore