Unreviewed, silence "enumeral and non-enumeral type in conditional expression" warning
[WebKit-https.git] / Source / WebCore / page / Chrome.cpp
1 /*
2  * Copyright (C) 2006-2017 Apple Inc. All rights reserved.
3  * Copyright (C) 2008, 2010 Nokia Corporation and/or its subsidiary(-ies)
4  * Copyright (C) 2012, Samsung Electronics. All rights reserved.
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public License
17  * along with this library; see the file COPYING.LIB.  If not, write to
18  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  */
21
22 #include "config.h"
23 #include "Chrome.h"
24
25 #include "ChromeClient.h"
26 #include "DOMWindow.h"
27 #include "Document.h"
28 #include "DocumentType.h"
29 #include "FileChooser.h"
30 #include "FileIconLoader.h"
31 #include "FileList.h"
32 #include "FloatRect.h"
33 #include "Frame.h"
34 #include "FrameLoaderClient.h"
35 #include "FrameTree.h"
36 #include "Geolocation.h"
37 #include "HTMLFormElement.h"
38 #include "HTMLInputElement.h"
39 #include "HTMLNames.h"
40 #include "HitTestResult.h"
41 #include "Icon.h"
42 #include "InspectorInstrumentation.h"
43 #include "Page.h"
44 #include "PageGroupLoadDeferrer.h"
45 #include "PopupOpeningObserver.h"
46 #include "RenderObject.h"
47 #include "ResourceHandle.h"
48 #include "Settings.h"
49 #include "StorageNamespace.h"
50 #include "WindowFeatures.h"
51 #include <JavaScriptCore/VM.h>
52 #include <wtf/SetForScope.h>
53 #include <wtf/Vector.h>
54
55 #if ENABLE(INPUT_TYPE_COLOR)
56 #include "ColorChooser.h"
57 #endif
58
59 #if ENABLE(DATALIST_ELEMENT)
60 #include "DataListSuggestionPicker.h"
61 #endif
62
63 #if PLATFORM(MAC) && ENABLE(GRAPHICS_CONTEXT_3D)
64 #include "GraphicsContext3DManager.h"
65 #endif
66
67 namespace WebCore {
68
69 using namespace HTMLNames;
70
71 Chrome::Chrome(Page& page, ChromeClient& client)
72     : m_page(page)
73     , m_client(client)
74 {
75 }
76
77 Chrome::~Chrome()
78 {
79     m_client.chromeDestroyed();
80 }
81
82 void Chrome::invalidateRootView(const IntRect& updateRect)
83 {
84     m_client.invalidateRootView(updateRect);
85 }
86
87 void Chrome::invalidateContentsAndRootView(const IntRect& updateRect)
88 {
89     m_client.invalidateContentsAndRootView(updateRect);
90 }
91
92 void Chrome::invalidateContentsForSlowScroll(const IntRect& updateRect)
93 {
94     m_client.invalidateContentsForSlowScroll(updateRect);
95 }
96
97 void Chrome::scroll(const IntSize& scrollDelta, const IntRect& rectToScroll, const IntRect& clipRect)
98 {
99     m_client.scroll(scrollDelta, rectToScroll, clipRect);
100     InspectorInstrumentation::didScroll(m_page);
101 }
102
103 IntPoint Chrome::screenToRootView(const IntPoint& point) const
104 {
105     return m_client.screenToRootView(point);
106 }
107
108 IntRect Chrome::rootViewToScreen(const IntRect& rect) const
109 {
110     return m_client.rootViewToScreen(rect);
111 }
112     
113 #if PLATFORM(IOS)
114
115 IntPoint Chrome::accessibilityScreenToRootView(const IntPoint& point) const
116 {
117     return m_client.accessibilityScreenToRootView(point);
118 }
119
120 IntRect Chrome::rootViewToAccessibilityScreen(const IntRect& rect) const
121 {
122     return m_client.rootViewToAccessibilityScreen(rect);
123 }
124
125 #endif
126
127 PlatformPageClient Chrome::platformPageClient() const
128 {
129     return m_client.platformPageClient();
130 }
131
132 void Chrome::contentsSizeChanged(Frame& frame, const IntSize& size) const
133 {
134     m_client.contentsSizeChanged(frame, size);
135 }
136
137 void Chrome::scrollRectIntoView(const IntRect& rect) const
138 {
139     m_client.scrollRectIntoView(rect);
140 }
141
142 void Chrome::setWindowRect(const FloatRect& rect) const
143 {
144     m_client.setWindowRect(rect);
145 }
146
147 FloatRect Chrome::windowRect() const
148 {
149     return m_client.windowRect();
150 }
151
152 FloatRect Chrome::pageRect() const
153 {
154     return m_client.pageRect();
155 }
156
157 void Chrome::focus() const
158 {
159     m_client.focus();
160 }
161
162 void Chrome::unfocus() const
163 {
164     m_client.unfocus();
165 }
166
167 bool Chrome::canTakeFocus(FocusDirection direction) const
168 {
169     return m_client.canTakeFocus(direction);
170 }
171
172 void Chrome::takeFocus(FocusDirection direction) const
173 {
174     m_client.takeFocus(direction);
175 }
176
177 void Chrome::focusedElementChanged(Element* element) const
178 {
179     m_client.focusedElementChanged(element);
180 }
181
182 void Chrome::focusedFrameChanged(Frame* frame) const
183 {
184     m_client.focusedFrameChanged(frame);
185 }
186
187 Page* Chrome::createWindow(Frame& frame, const FrameLoadRequest& request, const WindowFeatures& features, const NavigationAction& action) const
188 {
189     Page* newPage = m_client.createWindow(frame, request, features, action);
190     if (!newPage)
191         return nullptr;
192
193     if (auto* oldSessionStorage = m_page.sessionStorage(false))
194         newPage->setSessionStorage(oldSessionStorage->copy(newPage));
195     if (auto* oldEphemeralLocalStorage = m_page.ephemeralLocalStorage(false))
196         newPage->setEphemeralLocalStorage(oldEphemeralLocalStorage->copy(newPage));
197
198     return newPage;
199 }
200
201 void Chrome::show() const
202 {
203     m_client.show();
204 }
205
206 bool Chrome::canRunModal() const
207 {
208     return m_client.canRunModal();
209 }
210
211 void Chrome::runModal() const
212 {
213     // Defer callbacks in all the other pages in this group, so we don't try to run JavaScript
214     // in a way that could interact with this view.
215     PageGroupLoadDeferrer deferrer(m_page, false);
216
217     // JavaScript that runs within the nested event loop must not be run in the context of the
218     // script that called showModalDialog. Null out entryScope to break the connection.
219     SetForScope<JSC::VMEntryScope*> entryScopeNullifier { m_page.mainFrame().document()->vm().entryScope, nullptr };
220
221     TimerBase::fireTimersInNestedEventLoop();
222     m_client.runModal();
223 }
224
225 void Chrome::setToolbarsVisible(bool b) const
226 {
227     m_client.setToolbarsVisible(b);
228 }
229
230 bool Chrome::toolbarsVisible() const
231 {
232     return m_client.toolbarsVisible();
233 }
234
235 void Chrome::setStatusbarVisible(bool b) const
236 {
237     m_client.setStatusbarVisible(b);
238 }
239
240 bool Chrome::statusbarVisible() const
241 {
242     return m_client.statusbarVisible();
243 }
244
245 void Chrome::setScrollbarsVisible(bool b) const
246 {
247     m_client.setScrollbarsVisible(b);
248 }
249
250 bool Chrome::scrollbarsVisible() const
251 {
252     return m_client.scrollbarsVisible();
253 }
254
255 void Chrome::setMenubarVisible(bool b) const
256 {
257     m_client.setMenubarVisible(b);
258 }
259
260 bool Chrome::menubarVisible() const
261 {
262     return m_client.menubarVisible();
263 }
264
265 void Chrome::setResizable(bool b) const
266 {
267     m_client.setResizable(b);
268 }
269
270 bool Chrome::canRunBeforeUnloadConfirmPanel()
271 {
272     return m_client.canRunBeforeUnloadConfirmPanel();
273 }
274
275 bool Chrome::runBeforeUnloadConfirmPanel(const String& message, Frame& frame)
276 {
277     // Defer loads in case the client method runs a new event loop that would
278     // otherwise cause the load to continue while we're in the middle of executing JavaScript.
279     PageGroupLoadDeferrer deferrer(m_page, true);
280
281     return m_client.runBeforeUnloadConfirmPanel(message, frame);
282 }
283
284 void Chrome::closeWindowSoon()
285 {
286     m_client.closeWindowSoon();
287 }
288
289 void Chrome::runJavaScriptAlert(Frame& frame, const String& message)
290 {
291     // Defer loads in case the client method runs a new event loop that would
292     // otherwise cause the load to continue while we're in the middle of executing JavaScript.
293     PageGroupLoadDeferrer deferrer(m_page, true);
294
295     notifyPopupOpeningObservers();
296     String displayMessage = frame.displayStringModifiedByEncoding(message);
297
298     m_client.runJavaScriptAlert(frame, displayMessage);
299 }
300
301 bool Chrome::runJavaScriptConfirm(Frame& frame, const String& message)
302 {
303     // Defer loads in case the client method runs a new event loop that would
304     // otherwise cause the load to continue while we're in the middle of executing JavaScript.
305     PageGroupLoadDeferrer deferrer(m_page, true);
306
307     notifyPopupOpeningObservers();
308     return m_client.runJavaScriptConfirm(frame, frame.displayStringModifiedByEncoding(message));
309 }
310
311 bool Chrome::runJavaScriptPrompt(Frame& frame, const String& prompt, const String& defaultValue, String& result)
312 {
313     // Defer loads in case the client method runs a new event loop that would
314     // otherwise cause the load to continue while we're in the middle of executing JavaScript.
315     PageGroupLoadDeferrer deferrer(m_page, true);
316
317     notifyPopupOpeningObservers();
318     String displayPrompt = frame.displayStringModifiedByEncoding(prompt);
319
320     bool ok = m_client.runJavaScriptPrompt(frame, displayPrompt, frame.displayStringModifiedByEncoding(defaultValue), result);
321     if (ok)
322         result = frame.displayStringModifiedByEncoding(result);
323
324     return ok;
325 }
326
327 void Chrome::setStatusbarText(Frame& frame, const String& status)
328 {
329     m_client.setStatusbarText(frame.displayStringModifiedByEncoding(status));
330 }
331
332 void Chrome::mouseDidMoveOverElement(const HitTestResult& result, unsigned modifierFlags)
333 {
334     if (result.innerNode() && result.innerNode()->document().isDNSPrefetchEnabled())
335         m_page.mainFrame().loader().client().prefetchDNS(result.absoluteLinkURL().host().toString());
336     m_client.mouseDidMoveOverElement(result, modifierFlags);
337
338     InspectorInstrumentation::mouseDidMoveOverElement(m_page, result, modifierFlags);
339 }
340
341 void Chrome::setToolTip(const HitTestResult& result)
342 {
343     // First priority is a potential toolTip representing a spelling or grammar error
344     TextDirection toolTipDirection;
345     String toolTip = result.spellingToolTip(toolTipDirection);
346
347     // Next priority is a toolTip from a URL beneath the mouse (if preference is set to show those).
348     if (toolTip.isEmpty() && m_page.settings().showsURLsInToolTips()) {
349         if (Element* element = result.innerNonSharedElement()) {
350             // Get tooltip representing form action, if relevant
351             if (is<HTMLInputElement>(*element)) {
352                 HTMLInputElement& input = downcast<HTMLInputElement>(*element);
353                 if (input.isSubmitButton()) {
354                     if (HTMLFormElement* form = input.form()) {
355                         toolTip = form->action();
356                         if (form->renderer())
357                             toolTipDirection = form->renderer()->style().direction();
358                         else
359                             toolTipDirection = LTR;
360                     }
361                 }
362             }
363         }
364
365         // Get tooltip representing link's URL
366         if (toolTip.isEmpty()) {
367             // FIXME: Need to pass this URL through userVisibleString once that's in WebCore
368             toolTip = result.absoluteLinkURL().string();
369             // URL always display as LTR.
370             toolTipDirection = LTR;
371         }
372     }
373
374     // Next we'll consider a tooltip for element with "title" attribute
375     if (toolTip.isEmpty())
376         toolTip = result.title(toolTipDirection);
377
378     if (toolTip.isEmpty() && m_page.settings().showsToolTipOverTruncatedText())
379         toolTip = result.innerTextIfTruncated(toolTipDirection);
380
381     // Lastly, for <input type="file"> that allow multiple files, we'll consider a tooltip for the selected filenames
382     if (toolTip.isEmpty()) {
383         if (Element* element = result.innerNonSharedElement()) {
384             if (is<HTMLInputElement>(*element)) {
385                 toolTip = downcast<HTMLInputElement>(*element).defaultToolTip();
386
387                 // FIXME: We should obtain text direction of tooltip from
388                 // ChromeClient or platform. As of October 2011, all client
389                 // implementations don't use text direction information for
390                 // ChromeClient::setToolTip. We'll work on tooltip text
391                 // direction during bidi cleanup in form inputs.
392                 toolTipDirection = LTR;
393             }
394         }
395     }
396
397     m_client.setToolTip(toolTip, toolTipDirection);
398 }
399
400 bool Chrome::print(Frame& frame)
401 {
402     // FIXME: This should have PageGroupLoadDeferrer, like runModal() or runJavaScriptAlert(), because it's no different from those.
403
404     if (frame.document()->isSandboxed(SandboxModals)) {
405         frame.document()->domWindow()->printErrorMessage("Use of window.print is not allowed in a sandboxed frame when the allow-modals flag is not set.");
406         return false;
407     }
408
409     m_client.print(frame);
410     return true;
411 }
412
413 void Chrome::enableSuddenTermination()
414 {
415     m_client.enableSuddenTermination();
416 }
417
418 void Chrome::disableSuddenTermination()
419 {
420     m_client.disableSuddenTermination();
421 }
422
423 #if ENABLE(INPUT_TYPE_COLOR)
424
425 std::unique_ptr<ColorChooser> Chrome::createColorChooser(ColorChooserClient& client, const Color& initialColor)
426 {
427 #if PLATFORM(IOS)
428     return nullptr;
429 #endif
430     notifyPopupOpeningObservers();
431     return m_client.createColorChooser(client, initialColor);
432 }
433
434 #endif
435
436 #if ENABLE(DATALIST_ELEMENT)
437
438 std::unique_ptr<DataListSuggestionPicker> Chrome::createDataListSuggestionPicker(DataListSuggestionsClient& client)
439 {
440     notifyPopupOpeningObservers();
441     return m_client.createDataListSuggestionPicker(client);
442 }
443
444 #endif
445
446 void Chrome::runOpenPanel(Frame& frame, FileChooser& fileChooser)
447 {
448     notifyPopupOpeningObservers();
449     m_client.runOpenPanel(frame, fileChooser);
450 }
451
452 void Chrome::loadIconForFiles(const Vector<String>& filenames, FileIconLoader& loader)
453 {
454     m_client.loadIconForFiles(filenames, loader);
455 }
456
457 FloatSize Chrome::screenSize() const
458 {
459     return m_client.screenSize();
460 }
461
462 FloatSize Chrome::availableScreenSize() const
463 {
464     return m_client.availableScreenSize();
465 }
466
467 FloatSize Chrome::overrideScreenSize() const
468 {
469     return m_client.overrideScreenSize();
470 }
471
472 void Chrome::dispatchDisabledAdaptationsDidChange(const OptionSet<DisabledAdaptations>& disabledAdaptations) const
473 {
474     m_client.dispatchDisabledAdaptationsDidChange(disabledAdaptations);
475 }
476
477 void Chrome::dispatchViewportPropertiesDidChange(const ViewportArguments& arguments) const
478 {
479 #if PLATFORM(IOS)
480     if (m_isDispatchViewportDataDidChangeSuppressed)
481         return;
482 #endif
483     m_client.dispatchViewportPropertiesDidChange(arguments);
484 }
485
486 void Chrome::setCursor(const Cursor& cursor)
487 {
488 #if ENABLE(CURSOR_SUPPORT)
489     m_client.setCursor(cursor);
490 #else
491     UNUSED_PARAM(cursor);
492 #endif
493 }
494
495 void Chrome::setCursorHiddenUntilMouseMoves(bool hiddenUntilMouseMoves)
496 {
497 #if ENABLE(CURSOR_SUPPORT)
498     m_client.setCursorHiddenUntilMouseMoves(hiddenUntilMouseMoves);
499 #else
500     UNUSED_PARAM(hiddenUntilMouseMoves);
501 #endif
502 }
503
504 PlatformDisplayID Chrome::displayID() const
505 {
506     return m_displayID;
507 }
508
509 void Chrome::windowScreenDidChange(PlatformDisplayID displayID)
510 {
511     if (displayID == m_displayID)
512         return;
513
514     m_displayID = displayID;
515
516     for (Frame* frame = &m_page.mainFrame(); frame; frame = frame->tree().traverseNext()) {
517         if (frame->document())
518             frame->document()->windowScreenDidChange(displayID);
519     }
520
521 #if PLATFORM(MAC) && ENABLE(GRAPHICS_CONTEXT_3D)
522     GraphicsContext3DManager::sharedManager().screenDidChange(displayID, this);
523 #endif
524
525 }
526
527 // --------
528
529 #if ENABLE(DASHBOARD_SUPPORT)
530 void ChromeClient::annotatedRegionsChanged()
531 {
532 }
533 #endif
534
535 bool ChromeClient::shouldReplaceWithGeneratedFileForUpload(const String&, String&)
536 {
537     return false;
538 }
539
540 String ChromeClient::generateReplacementFile(const String&)
541 {
542     ASSERT_NOT_REACHED();
543     return String();
544 }
545
546 bool Chrome::selectItemWritingDirectionIsNatural()
547 {
548     return m_client.selectItemWritingDirectionIsNatural();
549 }
550
551 bool Chrome::selectItemAlignmentFollowsMenuWritingDirection()
552 {
553     return m_client.selectItemAlignmentFollowsMenuWritingDirection();
554 }
555
556 RefPtr<PopupMenu> Chrome::createPopupMenu(PopupMenuClient& client) const
557 {
558     notifyPopupOpeningObservers();
559     return m_client.createPopupMenu(client);
560 }
561
562 RefPtr<SearchPopupMenu> Chrome::createSearchPopupMenu(PopupMenuClient& client) const
563 {
564     notifyPopupOpeningObservers();
565     return m_client.createSearchPopupMenu(client);
566 }
567
568 bool Chrome::requiresFullscreenForVideoPlayback()
569 {
570     return m_client.requiresFullscreenForVideoPlayback();
571 }
572
573 void Chrome::didReceiveDocType(Frame& frame)
574 {
575 #if !PLATFORM(IOS)
576     UNUSED_PARAM(frame);
577 #else
578     if (!frame.isMainFrame())
579         return;
580
581     auto* doctype = frame.document()->doctype();
582     m_client.didReceiveMobileDocType(doctype && doctype->publicId().containsIgnoringASCIICase("xhtml mobile"));
583 #endif
584 }
585
586 void Chrome::registerPopupOpeningObserver(PopupOpeningObserver& observer)
587 {
588     m_popupOpeningObservers.append(&observer);
589 }
590
591 void Chrome::unregisterPopupOpeningObserver(PopupOpeningObserver& observer)
592 {
593     bool removed = m_popupOpeningObservers.removeFirst(&observer);
594     ASSERT_UNUSED(removed, removed);
595 }
596
597 void Chrome::notifyPopupOpeningObservers() const
598 {
599     const Vector<PopupOpeningObserver*> observers(m_popupOpeningObservers);
600     for (auto& observer : observers)
601         observer->willOpenPopup();
602 }
603
604 } // namespace WebCore