| /* |
| * Copyright (C) 2005, 2006, 2007, 2008, 2009 Apple Inc. All rights reserved. |
| * Copyright (C) 2006 David Smith (catfish.man@gmail.com) |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * 3. Neither the name of Apple Inc. ("Apple") nor the names of |
| * its contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
| * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #import "WebViewData.h" |
| |
| #import "WebKitLogging.h" |
| #import "WebPreferenceKeysPrivate.h" |
| #import "WebSelectionServiceController.h" |
| #import "WebViewGroup.h" |
| #import "WebViewInternal.h" |
| #import <JavaScriptCore/InitializeThreading.h> |
| #import <WebCore/AlternativeTextUIController.h> |
| #import <WebCore/HistoryItem.h> |
| #import <WebCore/RunLoopObserver.h> |
| #import <WebCore/TextIndicatorWindow.h> |
| #import <WebCore/ValidationBubble.h> |
| #import <WebCore/WebCoreJITOperations.h> |
| #import <wtf/MainThread.h> |
| #import <wtf/RunLoop.h> |
| #import <wtf/SetForScope.h> |
| |
| #if PLATFORM(IOS_FAMILY) |
| #import "WebGeolocationProviderIOS.h" |
| #import <WebCore/RuntimeApplicationChecks.h> |
| #import <WebCore/WebCoreThread.h> |
| #import <WebCore/WebCoreThreadInternal.h> |
| #endif |
| |
| #if ENABLE(WIRELESS_PLAYBACK_TARGET) && !PLATFORM(IOS_FAMILY) |
| #import "WebMediaPlaybackTargetPicker.h" |
| #endif |
| |
| #if PLATFORM(MAC) && ENABLE(VIDEO_PRESENTATION_MODE) |
| #import <WebCore/PlaybackSessionInterfaceMac.h> |
| #import <WebCore/PlaybackSessionModelMediaElement.h> |
| #endif |
| |
| BOOL applicationIsTerminating = NO; |
| int pluginDatabaseClientCount = 0; |
| |
| static CFRunLoopRef currentRunLoop() |
| { |
| #if PLATFORM(IOS_FAMILY) |
| // A race condition during WebView deallocation can lead to a crash if the layer sync run loop |
| // observer is added to the main run loop <rdar://problem/9798550>. However, for responsiveness, |
| // we still allow this, see <rdar://problem/7403328>. Since the race condition and subsequent |
| // crash are especially troublesome for iBooks, we never allow the observer to be added to the |
| // main run loop in iBooks. |
| if (WebCore::CocoaApplication::isIBooks()) |
| return WebThreadRunLoop(); |
| #endif |
| return CFRunLoopGetCurrent(); |
| } |
| |
| void LayerFlushController::scheduleLayerFlush() |
| { |
| m_layerFlushScheduler.schedule(); |
| } |
| |
| void LayerFlushController::invalidate() |
| { |
| m_layerFlushScheduler.invalidate(); |
| m_webView = nullptr; |
| } |
| |
| LayerFlushController::LayerFlushController(WebView* webView) |
| : m_webView(webView) |
| , m_layerFlushScheduler(this) |
| { |
| ASSERT_ARG(webView, webView); |
| } |
| |
| WebViewLayerFlushScheduler::WebViewLayerFlushScheduler(LayerFlushController* flushController) |
| : m_flushController(flushController) |
| { |
| m_runLoopObserver = makeUnique<WebCore::RunLoopObserver>(static_cast<CFIndex>(WebCore::RunLoopObserver::WellKnownRunLoopOrders::LayerFlush), [this]() { |
| this->layerFlushCallback(); |
| }); |
| } |
| |
| WebViewLayerFlushScheduler::~WebViewLayerFlushScheduler() |
| { |
| } |
| |
| void WebViewLayerFlushScheduler::schedule() |
| { |
| if (m_insideCallback) |
| m_rescheduledInsideCallback = true; |
| |
| m_runLoopObserver->schedule(currentRunLoop()); |
| } |
| |
| void WebViewLayerFlushScheduler::invalidate() |
| { |
| m_runLoopObserver->invalidate(); |
| } |
| |
| void WebViewLayerFlushScheduler::layerFlushCallback() |
| { |
| #if PLATFORM(IOS_FAMILY) |
| // Normally the layer flush callback happens before the web lock auto-unlock observer runs. |
| // However if the flush is rescheduled from the callback it may get pushed past it, to the next cycle. |
| WebThreadLock(); |
| #endif |
| |
| @autoreleasepool { |
| RefPtr<LayerFlushController> protector = m_flushController; |
| |
| SetForScope insideCallbackScope(m_insideCallback, true); |
| m_rescheduledInsideCallback = false; |
| |
| if (m_flushController->flushLayers() && !m_rescheduledInsideCallback) |
| invalidate(); |
| } |
| } |
| |
| #if PLATFORM(MAC) |
| |
| @implementation WebWindowVisibilityObserver |
| |
| - (instancetype)initWithView:(WebView *)view |
| { |
| self = [super init]; |
| if (!self) |
| return nil; |
| |
| _view = view; |
| return self; |
| } |
| |
| - (void)startObserving:(NSWindow *)window |
| { |
| // An NSView derived object such as WebView cannot observe these notifications, because NSView itself observes them. |
| [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(_windowVisibilityChanged:) |
| name:@"NSWindowDidOrderOffScreenNotification" object:window]; |
| [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(_windowVisibilityChanged:) |
| name:@"_NSWindowDidBecomeVisible" object:window]; |
| } |
| |
| - (void)stopObserving:(NSWindow *)window |
| { |
| [[NSNotificationCenter defaultCenter] removeObserver:self name:@"NSWindowDidOrderOffScreenNotification" object:window]; |
| [[NSNotificationCenter defaultCenter] removeObserver:self name:@"_NSWindowDidBecomeVisible" object:window]; |
| } |
| |
| - (void)_windowVisibilityChanged:(NSNotification *)notification |
| { |
| [_view _windowVisibilityChanged:notification]; |
| } |
| |
| @end |
| |
| #endif // PLATFORM(MAC) |
| |
| @implementation WebViewPrivate |
| |
| + (void)initialize |
| { |
| #if !PLATFORM(IOS_FAMILY) |
| JSC::initialize(); |
| WTF::initializeMainThread(); |
| WebCore::populateJITOperations(); |
| #endif |
| } |
| |
| - (id)init |
| { |
| self = [super init]; |
| if (!self) |
| return nil; |
| |
| allowsUndo = YES; |
| usesPageCache = YES; |
| shouldUpdateWhileOffscreen = YES; |
| |
| #if !PLATFORM(IOS_FAMILY) |
| windowOcclusionDetectionEnabled = YES; |
| #endif |
| |
| zoomMultiplier = 1; |
| zoomsTextOnly = NO; |
| |
| #if PLATFORM(MAC) |
| interactiveFormValidationEnabled = YES; |
| #else |
| interactiveFormValidationEnabled = NO; |
| #endif |
| // The default value should be synchronized with WebCore/page/Settings.cpp. |
| validationMessageTimerMagnification = 50; |
| |
| #if PLATFORM(IOS_FAMILY) |
| isStopping = NO; |
| _geolocationProvider = [WebGeolocationProviderIOS sharedGeolocationProvider]; |
| #endif |
| |
| shouldCloseWithWindow = false; |
| |
| pluginDatabaseClientCount++; |
| |
| m_alternativeTextUIController = makeUnique<WebCore::AlternativeTextUIController>(); |
| |
| return self; |
| } |
| |
| - (void)dealloc |
| { |
| ASSERT(applicationIsTerminating || !page); |
| ASSERT(applicationIsTerminating || !preferences); |
| #if !PLATFORM(IOS_FAMILY) |
| ASSERT(!insertionPasteboard); |
| #endif |
| #if ENABLE(VIDEO) |
| ASSERT(!fullscreenController); |
| #endif |
| |
| [super dealloc]; |
| } |
| |
| @end |