d261c7d3b8f03907920f6ace404047e7ec2bc53b
[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 "FEGaussianBlur.h"
30 #include "IntSize.h"
31 #include "LengthFunctions.h"
32 #include <wtf/text/TextStream.h>
33
34 namespace WebCore {
35
36 static inline IntSize outsetSizeForBlur(float stdDeviation)
37 {
38     auto kernelSize = FEGaussianBlur::calculateUnscaledKernelSize({ stdDeviation, stdDeviation });
39
40     // We take the half kernel size and multiply it with three, because we run box blur three times.
41     return {
42         3 * kernelSize.width() / 2,
43         3 * kernelSize.height() / 2
44     };
45 }
46
47 bool FilterOperations::operator==(const FilterOperations& other) const
48 {
49     size_t size = m_operations.size();
50     if (size != other.m_operations.size())
51         return false;
52     for (size_t i = 0; i < size; i++) {
53         if (*m_operations[i] != *other.m_operations[i])
54             return false;
55     }
56     return true;
57 }
58
59 bool FilterOperations::operationsMatch(const FilterOperations& other) const
60 {
61     size_t size = operations().size();
62     if (size != other.operations().size())
63         return false;
64     for (size_t i = 0; i < size; ++i) {
65         if (!operations()[i]->isSameType(*other.operations()[i]))
66             return false;
67     }
68     return true;
69 }
70
71 bool FilterOperations::hasReferenceFilter() const
72 {
73     for (auto& operation : m_operations) {
74         if (operation->type() == FilterOperation::REFERENCE)
75             return true;
76     }
77     return false;
78 }
79
80 bool FilterOperations::hasOutsets() const
81 {
82     for (auto& operation : m_operations) {
83         auto type = operation->type();
84         if (type == FilterOperation::BLUR || type == FilterOperation::DROP_SHADOW)
85             return true;
86     }
87     return false;
88 }
89
90 FilterOutsets FilterOperations::outsets() const
91 {
92     FilterOutsets totalOutsets;
93     for (auto& operation : m_operations) {
94         switch (operation->type()) {
95         case FilterOperation::BLUR: {
96             auto& blurOperation = downcast<BlurFilterOperation>(*operation);
97             float stdDeviation = floatValueForLength(blurOperation.stdDeviation(), 0);
98             IntSize outsetSize = outsetSizeForBlur(stdDeviation);
99             FilterOutsets outsets(outsetSize.height(), outsetSize.width(), outsetSize.height(), outsetSize.width());
100             totalOutsets += outsets;
101             break;
102         }
103         case FilterOperation::DROP_SHADOW: {
104             auto& dropShadowOperation = downcast<DropShadowFilterOperation>(*operation);
105             IntSize outsetSize = outsetSizeForBlur(dropShadowOperation.stdDeviation());
106             FilterOutsets outsets {
107                 std::max(0, outsetSize.height() - dropShadowOperation.y()),
108                 std::max(0, outsetSize.width() + dropShadowOperation.x()),
109                 std::max(0, outsetSize.height() + dropShadowOperation.y()),
110                 std::max(0, outsetSize.width() - dropShadowOperation.x())
111             };
112             totalOutsets += outsets;
113             break;
114         }
115         default:
116             break;
117         }
118     }
119     return totalOutsets;
120 }
121
122 bool FilterOperations::hasFilterThatAffectsOpacity() const
123 {
124     for (auto& operation : m_operations) {
125         if (operation->affectsOpacity())
126             return true;
127     }
128     return false;
129 }
130
131 bool FilterOperations::hasFilterThatMovesPixels() const
132 {
133     for (auto& operation : m_operations) {
134         if (operation->movesPixels())
135             return true;
136     }
137     return false;
138 }
139
140 bool FilterOperations::hasFilterThatShouldBeRestrictedBySecurityOrigin() const
141 {
142     for (auto& operation : m_operations) {
143         if (operation->shouldBeRestrictedBySecurityOrigin())
144             return true;
145     }
146     return false;
147 }
148
149 TextStream& operator<<(TextStream& ts, const FilterOperations& filters)
150 {
151     for (size_t i = 0; i < filters.size(); ++i) {
152         auto filter = filters.at(i);
153         if (filter)
154             ts << *filter;
155         else
156             ts << "(null)";
157         if (i < filters.size() - 1)
158             ts << " ";
159     }
160     return ts;
161 }
162
163 } // namespace WebCore