8e78a66e3322f41487b9d05574a5dc319c72a199
[WebKit-https.git] / Source / WebCore / platform / graphics / ImageFrame.cpp
1 /*
2  * Copyright (C) 2016 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 "ImageFrame.h"
28
29 #include <wtf/NeverDestroyed.h>
30
31 namespace WebCore {
32
33 ImageFrame::ImageFrame()
34 {
35 }
36
37 ImageFrame::~ImageFrame()
38 {
39     clearImage();
40 }
41
42 const ImageFrame& ImageFrame::defaultFrame()
43 {
44     static NeverDestroyed<ImageFrame> sharedInstance;
45     return sharedInstance;
46 }
47
48 ImageFrame& ImageFrame::operator=(const ImageFrame& other)
49 {
50     if (this == &other)
51         return *this;
52
53     m_decodingStatus = other.m_decodingStatus;
54     m_size = other.m_size;
55
56 #if !USE(CG)
57     if (other.backingStore())
58         initialize(*other.backingStore());
59     else
60         m_backingStore = nullptr;
61     m_disposalMethod = other.m_disposalMethod;
62 #endif
63
64     m_nativeImage = other.m_nativeImage;
65     m_subsamplingLevel = other.m_subsamplingLevel;
66     m_decodingOptions = other.m_decodingOptions;
67
68     m_orientation = other.m_orientation;
69     m_duration = other.m_duration;
70     m_hasAlpha = other.m_hasAlpha;
71     return *this;
72 }
73
74 void ImageFrame::setDecodingStatus(DecodingStatus decodingStatus)
75 {
76     ASSERT(decodingStatus != DecodingStatus::Decoding);
77     m_decodingStatus = decodingStatus;
78 }
79
80 ImageFrame::DecodingStatus ImageFrame::decodingStatus() const
81 {
82     ASSERT(m_decodingStatus != DecodingStatus::Decoding);
83     return m_decodingStatus;
84 }
85
86 unsigned ImageFrame::clearImage()
87 {
88 #if !USE(CG)
89     if (hasBackingStore())
90         m_backingStore = nullptr;
91 #endif
92
93     if (!hasNativeImage())
94         return 0;
95
96     unsigned frameBytes = this->frameBytes();
97
98     clearNativeImageSubimages(m_nativeImage);
99     m_nativeImage = nullptr;
100     m_decodingOptions = { };
101
102     return frameBytes;
103 }
104
105 unsigned ImageFrame::clear()
106 {
107     unsigned frameBytes = clearImage();
108     *this = ImageFrame();
109     return frameBytes;
110 }
111
112 #if !USE(CG)
113 bool ImageFrame::initialize(const ImageBackingStore& backingStore)
114 {
115     if (&backingStore == this->backingStore())
116         return true;
117
118     m_backingStore = ImageBackingStore::create(backingStore);
119     return m_backingStore != nullptr;
120 }
121
122 bool ImageFrame::initialize(const IntSize& size, bool premultiplyAlpha)
123 {
124     if (size.isEmpty())
125         return false;
126
127     m_backingStore = ImageBackingStore::create(size, premultiplyAlpha);
128     return m_backingStore != nullptr;
129 }
130 #endif
131
132 IntSize ImageFrame::size() const
133 {
134 #if !USE(CG)
135     if (hasBackingStore())
136         return backingStore()->size();
137 #endif
138     return m_size;
139 }
140     
141 bool ImageFrame::hasNativeImage(const std::optional<SubsamplingLevel>& subsamplingLevel) const
142 {
143     return m_nativeImage && (!subsamplingLevel || *subsamplingLevel >= m_subsamplingLevel);
144 }
145
146 bool ImageFrame::hasFullSizeNativeImage(const std::optional<SubsamplingLevel>& subsamplingLevel) const
147 {
148     return hasNativeImage(subsamplingLevel) && (m_decodingOptions.isSynchronous() || m_decodingOptions.hasFullSize());
149 }
150
151 bool ImageFrame::hasDecodedNativeImageCompatibleWithOptions(const std::optional<SubsamplingLevel>& subsamplingLevel, const DecodingOptions& decodingOptions) const
152 {
153     return hasNativeImage(subsamplingLevel) && m_decodingOptions.isAsynchronousCompatibleWith(decodingOptions);
154 }
155
156 Color ImageFrame::singlePixelSolidColor() const
157 {
158     if (!hasNativeImage() || m_size != IntSize(1, 1))
159         return Color();
160
161     return nativeImageSinglePixelSolidColor(m_nativeImage);
162 }
163
164 }