/* * Copyright (C) 2011 Apple Inc. All rights reserved. * Copyright (C) 2013 Adobe Systems Incorporated. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * 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 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 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 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "CSSCrossfadeValue.h" #include "AnimationUtilities.h" #include "CSSImageValue.h" #include "CachedImage.h" #include "CachedResourceLoader.h" #include "CrossfadeGeneratedImage.h" #include "ImageBuffer.h" #include "RenderElement.h" #include "StyleCachedImage.h" #include "StyleGeneratedImage.h" #include namespace WebCore { static inline double blendFunc(double from, double to, double progress) { return blend(from, to, progress); } static bool subimageKnownToBeOpaque(CSSValue& value, const RenderElement* renderer) { if (is(value)) return downcast(value).knownToBeOpaque(renderer); if (is(value)) return downcast(value).knownToBeOpaque(renderer); ASSERT_NOT_REACHED(); return false; } CSSCrossfadeValue::~CSSCrossfadeValue() { if (m_cachedFromImage) m_cachedFromImage->removeClient(&m_crossfadeSubimageObserver); if (m_cachedToImage) m_cachedToImage->removeClient(&m_crossfadeSubimageObserver); } String CSSCrossfadeValue::customCSSText() const { StringBuilder result; result.appendLiteral("-webkit-cross-fade("); result.append(m_fromValue->cssText()); result.appendLiteral(", "); result.append(m_toValue->cssText()); result.appendLiteral(", "); result.append(m_percentageValue->cssText()); result.append(')'); return result.toString(); } FloatSize CSSCrossfadeValue::fixedSize(const RenderElement* renderer) { float percentage = m_percentageValue->getFloatValue(); float inversePercentage = 1 - percentage; // FIXME: Skip Content Security Policy check when cross fade is applied to an element in a user agent shadow tree. // See . ResourceLoaderOptions options = CachedResourceLoader::defaultCachedResourceOptions(); CachedResourceLoader& cachedResourceLoader = renderer->document().cachedResourceLoader(); CachedImage* cachedFromImage = cachedImageForCSSValue(m_fromValue.get(), cachedResourceLoader, options); CachedImage* cachedToImage = cachedImageForCSSValue(m_toValue.get(), cachedResourceLoader, options); if (!cachedFromImage || !cachedToImage) return FloatSize(); 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 FloatSize(fromImageSize.width() * inversePercentage + toImageSize.width() * percentage, fromImageSize.height() * inversePercentage + toImageSize.height() * percentage); } bool CSSCrossfadeValue::isPending() const { return CSSImageGeneratorValue::subimageIsPending(m_fromValue.get()) || CSSImageGeneratorValue::subimageIsPending(m_toValue.get()); } bool CSSCrossfadeValue::knownToBeOpaque(const RenderElement* renderer) const { return subimageKnownToBeOpaque(*m_fromValue, renderer) && subimageKnownToBeOpaque(*m_toValue, renderer); } void CSSCrossfadeValue::loadSubimages(CachedResourceLoader& cachedResourceLoader, const ResourceLoaderOptions& options) { CachedResourceHandle oldCachedFromImage = m_cachedFromImage; CachedResourceHandle oldCachedToImage = m_cachedToImage; m_cachedFromImage = CSSImageGeneratorValue::cachedImageForCSSValue(m_fromValue.get(), cachedResourceLoader, options); m_cachedToImage = CSSImageGeneratorValue::cachedImageForCSSValue(m_toValue.get(), cachedResourceLoader, options); if (m_cachedFromImage != oldCachedFromImage) { if (oldCachedFromImage) oldCachedFromImage->removeClient(&m_crossfadeSubimageObserver); if (m_cachedFromImage) m_cachedFromImage->addClient(&m_crossfadeSubimageObserver); } if (m_cachedToImage != oldCachedToImage) { if (oldCachedToImage) oldCachedToImage->removeClient(&m_crossfadeSubimageObserver); if (m_cachedToImage) m_cachedToImage->addClient(&m_crossfadeSubimageObserver); } m_crossfadeSubimageObserver.setReady(true); } RefPtr CSSCrossfadeValue::image(RenderElement* renderer, const FloatSize& size) { if (size.isEmpty()) return nullptr; // FIXME: Skip Content Security Policy check when cross fade is applied to an element in a user agent shadow tree. // See . ResourceLoaderOptions options = CachedResourceLoader::defaultCachedResourceOptions(); CachedResourceLoader& cachedResourceLoader = renderer->document().cachedResourceLoader(); CachedImage* cachedFromImage = cachedImageForCSSValue(m_fromValue.get(), cachedResourceLoader, options); CachedImage* cachedToImage = cachedImageForCSSValue(m_toValue.get(), cachedResourceLoader, options); if (!cachedFromImage || !cachedToImage) return Image::nullImage(); Image* fromImage = cachedFromImage->imageForRenderer(renderer); Image* 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; } void CSSCrossfadeValue::crossfadeChanged(const IntRect&) { for (auto it = clients().begin(), end = clients().end(); it != end; ++it) it->key->imageChanged(static_cast(this)); } void CSSCrossfadeValue::CrossfadeSubimageObserverProxy::imageChanged(CachedImage*, const IntRect* rect) { if (m_ready) m_ownerValue->crossfadeChanged(*rect); } bool CSSCrossfadeValue::traverseSubresources(const std::function& handler) const { if (m_cachedFromImage && handler(*m_cachedFromImage)) return true; if (m_cachedToImage && handler(*m_cachedToImage)) return true; return false; } RefPtr CSSCrossfadeValue::blend(const CSSCrossfadeValue& from, double progress) const { ASSERT(equalInputImages(from)); if (!m_cachedToImage || !m_cachedFromImage) return nullptr; RefPtr toStyledImage = StyleCachedImage::create(m_cachedToImage.get()); RefPtr fromStyledImage = StyleCachedImage::create(m_cachedFromImage.get()); auto fromImageValue = CSSImageValue::create(m_cachedFromImage->url(), fromStyledImage.get()); auto toImageValue = CSSImageValue::create(m_cachedToImage->url(), toStyledImage.get()); RefPtr crossfadeValue = CSSCrossfadeValue::create(WTFMove(fromImageValue), WTFMove(toImageValue)); double fromPercentage = from.m_percentageValue->getDoubleValue(); if (from.m_percentageValue->isPercentage()) fromPercentage /= 100.0; double toPercentage = m_percentageValue->getDoubleValue(); if (m_percentageValue->isPercentage()) toPercentage /= 100.0; crossfadeValue->setPercentage(CSSPrimitiveValue::create(blendFunc(fromPercentage, toPercentage, progress), CSSPrimitiveValue::CSS_NUMBER)); return crossfadeValue; } bool CSSCrossfadeValue::equals(const CSSCrossfadeValue& other) const { return equalInputImages(other) && compareCSSValuePtr(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); } } // namespace WebCore