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