Video sound sometimes keeps playing in page cache
[WebKit-https.git] / Source / WebCore / css / FontFaceSet.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 "FontFaceSet.h"
28
29 #include "DOMPromiseProxy.h"
30 #include "Document.h"
31 #include "EventLoop.h"
32 #include "FontFace.h"
33 #include "FrameLoader.h"
34 #include "JSDOMBinding.h"
35 #include "JSDOMPromiseDeferred.h"
36 #include "JSFontFace.h"
37 #include "JSFontFaceSet.h"
38 #include <wtf/IsoMallocInlines.h>
39
40 namespace WebCore {
41
42 WTF_MAKE_ISO_ALLOCATED_IMPL(FontFaceSet);
43
44 Ref<FontFaceSet> FontFaceSet::create(Document& document, const Vector<RefPtr<FontFace>>& initialFaces)
45 {
46     Ref<FontFaceSet> result = adoptRef(*new FontFaceSet(document, initialFaces));
47     result->suspendIfNeeded();
48     return result;
49 }
50
51 Ref<FontFaceSet> FontFaceSet::create(Document& document, CSSFontFaceSet& backing)
52 {
53     Ref<FontFaceSet> result = adoptRef(*new FontFaceSet(document, backing));
54     result->suspendIfNeeded();
55     return result;
56 }
57
58 FontFaceSet::FontFaceSet(Document& document, const Vector<RefPtr<FontFace>>& initialFaces)
59     : ActiveDOMObject(document)
60     , m_backing(CSSFontFaceSet::create())
61     , m_readyPromise(makeUniqueRef<ReadyPromise>(*this, &FontFaceSet::readyPromiseResolve))
62 {
63     m_backing->addClient(*this);
64     for (auto& face : initialFaces)
65         add(*face);
66 }
67
68 FontFaceSet::FontFaceSet(Document& document, CSSFontFaceSet& backing)
69     : ActiveDOMObject(document)
70     , m_backing(backing)
71     , m_readyPromise(makeUniqueRef<ReadyPromise>(*this, &FontFaceSet::readyPromiseResolve))
72 {
73     if (document.frame())
74         m_isFirstLayoutDone = document.frame()->loader().stateMachine().firstLayoutDone();
75
76     if (m_isFirstLayoutDone && !backing.hasActiveFontFaces())
77         m_readyPromise->resolve(*this);
78
79     m_backing->addClient(*this);
80 }
81
82 FontFaceSet::~FontFaceSet()
83 {
84     m_backing->removeClient(*this);
85 }
86
87 FontFaceSet::Iterator::Iterator(FontFaceSet& set)
88     : m_target(set)
89 {
90 }
91
92 RefPtr<FontFace> FontFaceSet::Iterator::next()
93 {
94     if (m_index == m_target->size())
95         return nullptr;
96     return m_target->backing()[m_index++].wrapper();
97 }
98
99 FontFaceSet::PendingPromise::PendingPromise(LoadPromise&& promise)
100     : promise(makeUniqueRef<LoadPromise>(WTFMove(promise)))
101 {
102 }
103
104 FontFaceSet::PendingPromise::~PendingPromise() = default;
105
106 bool FontFaceSet::has(FontFace& face) const
107 {
108     return m_backing->hasFace(face.backing());
109 }
110
111 size_t FontFaceSet::size() const
112 {
113     return m_backing->faceCount();
114 }
115
116 FontFaceSet& FontFaceSet::add(FontFace& face)
117 {
118     if (!m_backing->hasFace(face.backing()))
119         m_backing->add(face.backing());
120     return *this;
121 }
122
123 bool FontFaceSet::remove(FontFace& face)
124 {
125     bool result = m_backing->hasFace(face.backing());
126     if (result)
127         m_backing->remove(face.backing());
128     return result;
129 }
130
131 void FontFaceSet::clear()
132 {
133     while (m_backing->faceCount())
134         m_backing->remove(m_backing.get()[0]);
135 }
136
137 void FontFaceSet::load(const String& font, const String& text, LoadPromise&& promise)
138 {
139     auto matchingFacesResult = m_backing->matchingFacesExcludingPreinstalledFonts(font, text);
140     if (matchingFacesResult.hasException()) {
141         promise.reject(matchingFacesResult.releaseException());
142         return;
143     }
144     auto matchingFaces = matchingFacesResult.releaseReturnValue();
145
146     if (matchingFaces.isEmpty()) {
147         promise.resolve({ });
148         return;
149     }
150
151     for (auto& face : matchingFaces)
152         face.get().load();
153
154     for (auto& face : matchingFaces) {
155         if (face.get().status() == CSSFontFace::Status::Failure) {
156             promise.reject(NetworkError);
157             return;
158         }
159     }
160
161     auto pendingPromise = PendingPromise::create(WTFMove(promise));
162     bool waiting = false;
163
164     for (auto& face : matchingFaces) {
165         pendingPromise->faces.append(face.get().wrapper());
166         if (face.get().status() == CSSFontFace::Status::Success)
167             continue;
168         waiting = true;
169         ASSERT(face.get().existingWrapper());
170         m_pendingPromises.add(face.get().existingWrapper(), Vector<Ref<PendingPromise>>()).iterator->value.append(pendingPromise.copyRef());
171     }
172
173     if (!waiting)
174         pendingPromise->promise->resolve(pendingPromise->faces);
175 }
176
177 ExceptionOr<bool> FontFaceSet::check(const String& family, const String& text)
178 {
179     return m_backing->check(family, text);
180 }
181     
182 auto FontFaceSet::status() const -> LoadStatus
183 {
184     switch (m_backing->status()) {
185     case CSSFontFaceSet::Status::Loading:
186         return LoadStatus::Loading;
187     case CSSFontFaceSet::Status::Loaded:
188         return LoadStatus::Loaded;
189     }
190     ASSERT_NOT_REACHED();
191     return LoadStatus::Loaded;
192 }
193
194 void FontFaceSet::startedLoading()
195 {
196     // FIXME: Fire a "loading" event asynchronously.
197 }
198
199 void FontFaceSet::didFirstLayout()
200 {
201     m_isFirstLayoutDone = true;
202     if (!m_backing->hasActiveFontFaces() && !m_readyPromise->isFulfilled())
203         m_readyPromise->resolve(*this);
204 }
205
206 void FontFaceSet::completedLoading()
207 {
208     if (m_isFirstLayoutDone && !m_readyPromise->isFulfilled())
209         m_readyPromise->resolve(*this);
210 }
211
212 void FontFaceSet::faceFinished(CSSFontFace& face, CSSFontFace::Status newStatus)
213 {
214     if (!face.existingWrapper())
215         return;
216
217     auto pendingPromises = m_pendingPromises.take(face.existingWrapper());
218     if (pendingPromises.isEmpty())
219         return;
220
221     for (auto& pendingPromise : pendingPromises) {
222         if (pendingPromise->hasReachedTerminalState)
223             continue;
224         if (newStatus == CSSFontFace::Status::Success) {
225             if (pendingPromise->hasOneRef()) {
226                 pendingPromise->promise->resolve(pendingPromise->faces);
227                 pendingPromise->hasReachedTerminalState = true;
228             }
229         } else {
230             ASSERT(newStatus == CSSFontFace::Status::Failure);
231             pendingPromise->promise->reject(NetworkError);
232             pendingPromise->hasReachedTerminalState = true;
233         }
234     }
235 }
236
237 FontFaceSet& FontFaceSet::readyPromiseResolve()
238 {
239     return *this;
240 }
241
242 }