Implement rendering support for the color-filter CSS property
[WebKit-https.git] / Source / WebCore / platform / graphics / filters / FilterOperations.cpp
1 /*
2  * Copyright (C) 2011-2017 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27 #include "FilterOperations.h"
28
29 #include "ColorUtilities.h"
30 #include "FEGaussianBlur.h"
31 #include "IntSize.h"
32 #include "LengthFunctions.h"
33 #include <wtf/text/TextStream.h>
34
35 namespace WebCore {
36
37 static inline IntSize outsetSizeForBlur(float stdDeviation)
38 {
39     auto kernelSize = FEGaussianBlur::calculateUnscaledKernelSize({ stdDeviation, stdDeviation });
40
41     // We take the half kernel size and multiply it with three, because we run box blur three times.
42     return {
43         3 * kernelSize.width() / 2,
44         3 * kernelSize.height() / 2
45     };
46 }
47
48 bool FilterOperations::operator==(const FilterOperations& other) const
49 {
50     size_t size = m_operations.size();
51     if (size != other.m_operations.size())
52         return false;
53     for (size_t i = 0; i < size; i++) {
54         if (*m_operations[i] != *other.m_operations[i])
55             return false;
56     }
57     return true;
58 }
59
60 bool FilterOperations::operationsMatch(const FilterOperations& other) const
61 {
62     size_t size = operations().size();
63     if (size != other.operations().size())
64         return false;
65     for (size_t i = 0; i < size; ++i) {
66         if (!operations()[i]->isSameType(*other.operations()[i]))
67             return false;
68     }
69     return true;
70 }
71
72 bool FilterOperations::hasReferenceFilter() const
73 {
74     for (auto& operation : m_operations) {
75         if (operation->type() == FilterOperation::REFERENCE)
76             return true;
77     }
78     return false;
79 }
80
81 bool FilterOperations::hasOutsets() const
82 {
83     for (auto& operation : m_operations) {
84         auto type = operation->type();
85         if (type == FilterOperation::BLUR || type == FilterOperation::DROP_SHADOW)
86             return true;
87     }
88     return false;
89 }
90
91 FilterOutsets FilterOperations::outsets() const
92 {
93     FilterOutsets totalOutsets;
94     for (auto& operation : m_operations) {
95         switch (operation->type()) {
96         case FilterOperation::BLUR: {
97             auto& blurOperation = downcast<BlurFilterOperation>(*operation);
98             float stdDeviation = floatValueForLength(blurOperation.stdDeviation(), 0);
99             IntSize outsetSize = outsetSizeForBlur(stdDeviation);
100             FilterOutsets outsets(outsetSize.height(), outsetSize.width(), outsetSize.height(), outsetSize.width());
101             totalOutsets += outsets;
102             break;
103         }
104         case FilterOperation::DROP_SHADOW: {
105             auto& dropShadowOperation = downcast<DropShadowFilterOperation>(*operation);
106             IntSize outsetSize = outsetSizeForBlur(dropShadowOperation.stdDeviation());
107             FilterOutsets outsets {
108                 std::max(0, outsetSize.height() - dropShadowOperation.y()),
109                 std::max(0, outsetSize.width() + dropShadowOperation.x()),
110                 std::max(0, outsetSize.height() + dropShadowOperation.y()),
111                 std::max(0, outsetSize.width() - dropShadowOperation.x())
112             };
113             totalOutsets += outsets;
114             break;
115         }
116         default:
117             break;
118         }
119     }
120     return totalOutsets;
121 }
122
123 bool FilterOperations::transformColor(Color& color) const
124 {
125     if (isEmpty())
126         return false;
127
128     FloatComponents components;
129     color.getRGBA(components.components[0], components.components[1], components.components[2], components.components[3]);
130
131     for (auto& operation : m_operations) {
132         if (!operation->transformColor(components))
133             return false;
134     }
135
136     color = Color(components.components[0], components.components[1], components.components[2], components.components[3]);
137     return true;
138 }
139
140 bool FilterOperations::hasFilterThatAffectsOpacity() const
141 {
142     for (auto& operation : m_operations) {
143         if (operation->affectsOpacity())
144             return true;
145     }
146     return false;
147 }
148
149 bool FilterOperations::hasFilterThatMovesPixels() const
150 {
151     for (auto& operation : m_operations) {
152         if (operation->movesPixels())
153             return true;
154     }
155     return false;
156 }
157
158 bool FilterOperations::hasFilterThatShouldBeRestrictedBySecurityOrigin() const
159 {
160     for (auto& operation : m_operations) {
161         if (operation->shouldBeRestrictedBySecurityOrigin())
162             return true;
163     }
164     return false;
165 }
166
167 TextStream& operator<<(TextStream& ts, const FilterOperations& filters)
168 {
169     for (size_t i = 0; i < filters.size(); ++i) {
170         auto filter = filters.at(i);
171         if (filter)
172             ts << *filter;
173         else
174             ts << "(null)";
175         if (i < filters.size() - 1)
176             ts << " ";
177     }
178     return ts;
179 }
180
181 } // namespace WebCore