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