#include "config.h"
#include "WKBundlePage.h"
#include "WKBundlePagePrivate.h"
#include "APIArray.h"
#include "APIString.h"
#include "APIURL.h"
#include "APIURLRequest.h"
#include "InjectedBundleBackForwardList.h"
#include "InjectedBundleNodeHandle.h"
#include "InjectedBundlePageFormClient.h"
#include "InjectedBundlePageUIClient.h"
#include "PageBanner.h"
#include "WKAPICast.h"
#include "WKArray.h"
#include "WKBundleAPICast.h"
#include "WKRetainPtr.h"
#include "WKString.h"
#include "WebContextMenu.h"
#include "WebContextMenuItem.h"
#include "WebFrame.h"
#include "WebFullScreenManager.h"
#include "WebImage.h"
#include "WebInspector.h"
#include "WebPage.h"
#include "WebPageGroupProxy.h"
#include "WebPageOverlay.h"
#include "WebRenderLayer.h"
#include "WebRenderObject.h"
#include <WebCore/AXObjectCache.h>
#include <WebCore/AccessibilityObject.h>
#include <WebCore/MainFrame.h>
#include <WebCore/Page.h>
#include <WebCore/PageOverlay.h>
#include <WebCore/PageOverlayController.h>
#include <WebCore/URL.h>
#include <WebCore/WheelEventTestTrigger.h>
#include <wtf/StdLibExtras.h>
using namespace WebKit;
WKTypeID WKBundlePageGetTypeID()
{
return toAPI(WebPage::APIType);
}
void WKBundlePageSetContextMenuClient(WKBundlePageRef pageRef, WKBundlePageContextMenuClientBase* wkClient)
{
#if ENABLE(CONTEXT_MENUS)
toImpl(pageRef)->setInjectedBundleContextMenuClient(std::make_unique<InjectedBundlePageContextMenuClient>(wkClient));
#else
UNUSED_PARAM(pageRef);
UNUSED_PARAM(wkClient);
#endif
}
void WKBundlePageSetEditorClient(WKBundlePageRef pageRef, WKBundlePageEditorClientBase* wkClient)
{
toImpl(pageRef)->initializeInjectedBundleEditorClient(wkClient);
}
void WKBundlePageSetFormClient(WKBundlePageRef pageRef, WKBundlePageFormClientBase* wkClient)
{
toImpl(pageRef)->setInjectedBundleFormClient(std::make_unique<InjectedBundlePageFormClient>(wkClient));
}
void WKBundlePageSetPageLoaderClient(WKBundlePageRef pageRef, WKBundlePageLoaderClientBase* wkClient)
{
toImpl(pageRef)->initializeInjectedBundleLoaderClient(wkClient);
}
void WKBundlePageSetResourceLoadClient(WKBundlePageRef pageRef, WKBundlePageResourceLoadClientBase* wkClient)
{
toImpl(pageRef)->initializeInjectedBundleResourceLoadClient(wkClient);
}
void WKBundlePageSetPolicyClient(WKBundlePageRef pageRef, WKBundlePagePolicyClientBase* wkClient)
{
toImpl(pageRef)->initializeInjectedBundlePolicyClient(wkClient);
}
void WKBundlePageSetUIClient(WKBundlePageRef pageRef, WKBundlePageUIClientBase* wkClient)
{
toImpl(pageRef)->setInjectedBundleUIClient(std::make_unique<InjectedBundlePageUIClient>(wkClient));
}
void WKBundlePageSetFullScreenClient(WKBundlePageRef pageRef, WKBundlePageFullScreenClientBase* wkClient)
{
#if defined(ENABLE_FULLSCREEN_API) && ENABLE_FULLSCREEN_API
toImpl(pageRef)->initializeInjectedBundleFullScreenClient(wkClient);
#else
UNUSED_PARAM(pageRef);
UNUSED_PARAM(wkClient);
#endif
}
void WKBundlePageWillEnterFullScreen(WKBundlePageRef pageRef)
{
#if defined(ENABLE_FULLSCREEN_API) && ENABLE_FULLSCREEN_API
toImpl(pageRef)->fullScreenManager()->willEnterFullScreen();
#else
UNUSED_PARAM(pageRef);
#endif
}
void WKBundlePageDidEnterFullScreen(WKBundlePageRef pageRef)
{
#if defined(ENABLE_FULLSCREEN_API) && ENABLE_FULLSCREEN_API
toImpl(pageRef)->fullScreenManager()->didEnterFullScreen();
#else
UNUSED_PARAM(pageRef);
#endif
}
void WKBundlePageWillExitFullScreen(WKBundlePageRef pageRef)
{
#if defined(ENABLE_FULLSCREEN_API) && ENABLE_FULLSCREEN_API
toImpl(pageRef)->fullScreenManager()->willExitFullScreen();
#else
UNUSED_PARAM(pageRef);
#endif
}
void WKBundlePageDidExitFullScreen(WKBundlePageRef pageRef)
{
#if defined(ENABLE_FULLSCREEN_API) && ENABLE_FULLSCREEN_API
toImpl(pageRef)->fullScreenManager()->didExitFullScreen();
#else
UNUSED_PARAM(pageRef);
#endif
}
void WKBundlePageSetDiagnosticLoggingClient(WKBundlePageRef pageRef, WKBundlePageDiagnosticLoggingClientBase* client)
{
toImpl(pageRef)->initializeInjectedBundleDiagnosticLoggingClient(client);
}
WKBundlePageGroupRef WKBundlePageGetPageGroup(WKBundlePageRef pageRef)
{
return toAPI(toImpl(pageRef)->pageGroup());
}
WKBundleFrameRef WKBundlePageGetMainFrame(WKBundlePageRef pageRef)
{
return toAPI(toImpl(pageRef)->mainWebFrame());
}
void WKBundlePageClickMenuItem(WKBundlePageRef pageRef, WKContextMenuItemRef item)
{
#if ENABLE(CONTEXT_MENUS)
toImpl(pageRef)->contextMenu()->itemSelected(*toImpl(item)->data());
#else
UNUSED_PARAM(pageRef);
UNUSED_PARAM(item);
#endif
}
#if ENABLE(CONTEXT_MENUS)
static PassRefPtr<API::Array> contextMenuItems(const WebContextMenu& contextMenu)
{
auto items = contextMenu.items();
Vector<RefPtr<API::Object>> menuItems;
menuItems.reserveInitialCapacity(items.size());
for (const auto& item : items)
menuItems.uncheckedAppend(WebContextMenuItem::create(item));
return API::Array::create(WTF::move(menuItems));
}
#endif
WKArrayRef WKBundlePageCopyContextMenuItems(WKBundlePageRef pageRef)
{
#if ENABLE(CONTEXT_MENUS)
WebContextMenu* contextMenu = toImpl(pageRef)->contextMenu();
return toAPI(contextMenuItems(*contextMenu).leakRef());
#else
UNUSED_PARAM(pageRef);
return nullptr;
#endif
}
WKArrayRef WKBundlePageCopyContextMenuAtPointInWindow(WKBundlePageRef pageRef, WKPoint point)
{
#if ENABLE(CONTEXT_MENUS)
WebContextMenu* contextMenu = toImpl(pageRef)->contextMenuAtPointInWindow(toIntPoint(point));
if (!contextMenu)
return nullptr;
return toAPI(contextMenuItems(*contextMenu).leakRef());
#else
UNUSED_PARAM(pageRef);
UNUSED_PARAM(point);
return nullptr;
#endif
}
void* WKAccessibilityRootObject(WKBundlePageRef pageRef)
{
#if HAVE(ACCESSIBILITY)
if (!pageRef)
return 0;
WebCore::Page* page = toImpl(pageRef)->corePage();
if (!page)
return 0;
WebCore::Frame& core = page->mainFrame();
if (!core.document())
return 0;
WebCore::AXObjectCache::enableAccessibility();
WebCore::AccessibilityObject* root = core.document()->axObjectCache()->rootObject();
if (!root)
return 0;
return root->wrapper();
#else
UNUSED_PARAM(pageRef);
return 0;
#endif
}
void* WKAccessibilityFocusedObject(WKBundlePageRef pageRef)
{
#if HAVE(ACCESSIBILITY)
if (!pageRef)
return 0;
WebCore::Page* page = toImpl(pageRef)->corePage();
if (!page)
return 0;
WebCore::AXObjectCache::enableAccessibility();
WebCore::AccessibilityObject* focusedObject = WebCore::AXObjectCache::focusedUIElementForPage(page);
if (!focusedObject)
return 0;
return focusedObject->wrapper();
#else
UNUSED_PARAM(pageRef);
return 0;
#endif
}
void WKAccessibilityEnableEnhancedAccessibility(bool enable)
{
#if HAVE(ACCESSIBILITY)
WebCore::AXObjectCache::setEnhancedUserInterfaceAccessibility(enable);
#endif
}
bool WKAccessibilityEnhancedAccessibilityEnabled()
{
#if HAVE(ACCESSIBILITY)
return WebCore::AXObjectCache::accessibilityEnhancedUserInterfaceEnabled();
#else
return false;
#endif
}
void WKBundlePageStopLoading(WKBundlePageRef pageRef)
{
toImpl(pageRef)->stopLoading();
}
void WKBundlePageSetDefersLoading(WKBundlePageRef pageRef, bool defersLoading)
{
toImpl(pageRef)->setDefersLoading(defersLoading);
}
WKStringRef WKBundlePageCopyRenderTreeExternalRepresentation(WKBundlePageRef pageRef)
{
return toCopiedAPI(toImpl(pageRef)->renderTreeExternalRepresentation());
}
WKStringRef WKBundlePageCopyRenderTreeExternalRepresentationForPrinting(WKBundlePageRef pageRef)
{
return toCopiedAPI(toImpl(pageRef)->renderTreeExternalRepresentationForPrinting());
}
void WKBundlePageExecuteEditingCommand(WKBundlePageRef pageRef, WKStringRef name, WKStringRef argument)
{
toImpl(pageRef)->executeEditingCommand(toWTFString(name), toWTFString(argument));
}
bool WKBundlePageIsEditingCommandEnabled(WKBundlePageRef pageRef, WKStringRef name)
{
return toImpl(pageRef)->isEditingCommandEnabled(toWTFString(name));
}
void WKBundlePageClearMainFrameName(WKBundlePageRef pageRef)
{
toImpl(pageRef)->clearMainFrameName();
}
void WKBundlePageClose(WKBundlePageRef pageRef)
{
toImpl(pageRef)->sendClose();
}
double WKBundlePageGetTextZoomFactor(WKBundlePageRef pageRef)
{
return toImpl(pageRef)->textZoomFactor();
}
void WKBundlePageSetTextZoomFactor(WKBundlePageRef pageRef, double zoomFactor)
{
toImpl(pageRef)->setTextZoomFactor(zoomFactor);
}
double WKBundlePageGetPageZoomFactor(WKBundlePageRef pageRef)
{
return toImpl(pageRef)->pageZoomFactor();
}
void WKBundlePageSetPageZoomFactor(WKBundlePageRef pageRef, double zoomFactor)
{
toImpl(pageRef)->setPageZoomFactor(zoomFactor);
}
void WKBundlePageSetScaleAtOrigin(WKBundlePageRef pageRef, double scale, WKPoint origin)
{
toImpl(pageRef)->scalePage(scale, toIntPoint(origin));
}
WKBundleBackForwardListRef WKBundlePageGetBackForwardList(WKBundlePageRef pageRef)
{
return toAPI(toImpl(pageRef)->backForwardList());
}
void WKBundlePageInstallPageOverlay(WKBundlePageRef pageRef, WKBundlePageOverlayRef pageOverlayRef)
{
toImpl(pageRef)->mainFrame()->pageOverlayController().installPageOverlay(toImpl(pageOverlayRef)->coreOverlay(), WebCore::PageOverlay::FadeMode::DoNotFade);
}
void WKBundlePageUninstallPageOverlay(WKBundlePageRef pageRef, WKBundlePageOverlayRef pageOverlayRef)
{
toImpl(pageRef)->mainFrame()->pageOverlayController().uninstallPageOverlay(toImpl(pageOverlayRef)->coreOverlay(), WebCore::PageOverlay::FadeMode::DoNotFade);
}
void WKBundlePageInstallPageOverlayWithAnimation(WKBundlePageRef pageRef, WKBundlePageOverlayRef pageOverlayRef)
{
toImpl(pageRef)->mainFrame()->pageOverlayController().installPageOverlay(toImpl(pageOverlayRef)->coreOverlay(), WebCore::PageOverlay::FadeMode::Fade);
}
void WKBundlePageUninstallPageOverlayWithAnimation(WKBundlePageRef pageRef, WKBundlePageOverlayRef pageOverlayRef)
{
toImpl(pageRef)->mainFrame()->pageOverlayController().uninstallPageOverlay(toImpl(pageOverlayRef)->coreOverlay(), WebCore::PageOverlay::FadeMode::Fade);
}
void WKBundlePageSetTopOverhangImage(WKBundlePageRef pageRef, WKImageRef imageRef)
{
#if PLATFORM(MAC)
toImpl(pageRef)->setTopOverhangImage(toImpl(imageRef));
#else
UNUSED_PARAM(pageRef);
UNUSED_PARAM(imageRef);
#endif
}
void WKBundlePageSetBottomOverhangImage(WKBundlePageRef pageRef, WKImageRef imageRef)
{
#if PLATFORM(MAC)
toImpl(pageRef)->setBottomOverhangImage(toImpl(imageRef));
#else
UNUSED_PARAM(pageRef);
UNUSED_PARAM(imageRef);
#endif
}
#if !PLATFORM(IOS)
void WKBundlePageSetHeaderBanner(WKBundlePageRef pageRef, WKBundlePageBannerRef bannerRef)
{
toImpl(pageRef)->setHeaderPageBanner(toImpl(bannerRef));
}
void WKBundlePageSetFooterBanner(WKBundlePageRef pageRef, WKBundlePageBannerRef bannerRef)
{
toImpl(pageRef)->setFooterPageBanner(toImpl(bannerRef));
}
#endif // !PLATFORM(IOS)
bool WKBundlePageHasLocalDataForURL(WKBundlePageRef pageRef, WKURLRef urlRef)
{
return toImpl(pageRef)->hasLocalDataForURL(WebCore::URL(WebCore::URL(), toWTFString(urlRef)));
}
bool WKBundlePageCanHandleRequest(WKURLRequestRef requestRef)
{
return WebPage::canHandleRequest(toImpl(requestRef)->resourceRequest());
}
bool WKBundlePageFindString(WKBundlePageRef pageRef, WKStringRef target, WKFindOptions findOptions)
{
return toImpl(pageRef)->findStringFromInjectedBundle(toWTFString(target), toFindOptions(findOptions));
}
WKImageRef WKBundlePageCreateSnapshotWithOptions(WKBundlePageRef pageRef, WKRect rect, WKSnapshotOptions options)
{
RefPtr<WebImage> webImage = toImpl(pageRef)->scaledSnapshotWithOptions(toIntRect(rect), 1, toSnapshotOptions(options));
return toAPI(webImage.release().leakRef());
}
WKImageRef WKBundlePageCreateSnapshotInViewCoordinates(WKBundlePageRef pageRef, WKRect rect, WKImageOptions options)
{
SnapshotOptions snapshotOptions = snapshotOptionsFromImageOptions(options);
snapshotOptions |= SnapshotOptionsInViewCoordinates;
RefPtr<WebImage> webImage = toImpl(pageRef)->scaledSnapshotWithOptions(toIntRect(rect), 1, snapshotOptions);
return toAPI(webImage.release().leakRef());
}
WKImageRef WKBundlePageCreateSnapshotInDocumentCoordinates(WKBundlePageRef pageRef, WKRect rect, WKImageOptions options)
{
RefPtr<WebImage> webImage = toImpl(pageRef)->scaledSnapshotWithOptions(toIntRect(rect), 1, snapshotOptionsFromImageOptions(options));
return toAPI(webImage.release().leakRef());
}
WKImageRef WKBundlePageCreateScaledSnapshotInDocumentCoordinates(WKBundlePageRef pageRef, WKRect rect, double scaleFactor, WKImageOptions options)
{
RefPtr<WebImage> webImage = toImpl(pageRef)->scaledSnapshotWithOptions(toIntRect(rect), scaleFactor, snapshotOptionsFromImageOptions(options));
return toAPI(webImage.release().leakRef());
}
double WKBundlePageGetBackingScaleFactor(WKBundlePageRef pageRef)
{
return toImpl(pageRef)->deviceScaleFactor();
}
void WKBundlePageListenForLayoutMilestones(WKBundlePageRef pageRef, WKLayoutMilestones milestones)
{
toImpl(pageRef)->listenForLayoutMilestones(toLayoutMilestones(milestones));
}
WKBundleInspectorRef WKBundlePageGetInspector(WKBundlePageRef pageRef)
{
return toAPI(toImpl(pageRef)->inspector());
}
void WKBundlePageForceRepaint(WKBundlePageRef page)
{
toImpl(page)->forceRepaintWithoutCallback();
}
void WKBundlePageSimulateMouseDown(WKBundlePageRef page, int button, WKPoint position, int clickCount, WKEventModifiers modifiers, double time)
{
toImpl(page)->simulateMouseDown(button, toIntPoint(position), clickCount, modifiers, time);
}
void WKBundlePageSimulateMouseUp(WKBundlePageRef page, int button, WKPoint position, int clickCount, WKEventModifiers modifiers, double time)
{
toImpl(page)->simulateMouseUp(button, toIntPoint(position), clickCount, modifiers, time);
}
void WKBundlePageSimulateMouseMotion(WKBundlePageRef page, WKPoint position, double time)
{
toImpl(page)->simulateMouseMotion(toIntPoint(position), time);
}
uint64_t WKBundlePageGetRenderTreeSize(WKBundlePageRef pageRef)
{
return toImpl(pageRef)->renderTreeSize();
}
WKRenderObjectRef WKBundlePageCopyRenderTree(WKBundlePageRef pageRef)
{
return toAPI(WebRenderObject::create(toImpl(pageRef)).leakRef());
}
WKRenderLayerRef WKBundlePageCopyRenderLayerTree(WKBundlePageRef pageRef)
{
return toAPI(WebRenderLayer::create(toImpl(pageRef)).leakRef());
}
void WKBundlePageSetPaintedObjectsCounterThreshold(WKBundlePageRef, uint64_t)
{
}
void WKBundlePageSetTracksRepaints(WKBundlePageRef pageRef, bool trackRepaints)
{
toImpl(pageRef)->setTracksRepaints(trackRepaints);
}
bool WKBundlePageIsTrackingRepaints(WKBundlePageRef pageRef)
{
return toImpl(pageRef)->isTrackingRepaints();
}
void WKBundlePageResetTrackedRepaints(WKBundlePageRef pageRef)
{
toImpl(pageRef)->resetTrackedRepaints();
}
WKArrayRef WKBundlePageCopyTrackedRepaintRects(WKBundlePageRef pageRef)
{
return toAPI(&toImpl(pageRef)->trackedRepaintRects().leakRef());
}
void WKBundlePageSetComposition(WKBundlePageRef pageRef, WKStringRef text, int from, int length)
{
toImpl(pageRef)->setCompositionForTesting(toWTFString(text), from, length);
}
bool WKBundlePageHasComposition(WKBundlePageRef pageRef)
{
return toImpl(pageRef)->hasCompositionForTesting();
}
void WKBundlePageConfirmComposition(WKBundlePageRef pageRef)
{
toImpl(pageRef)->confirmCompositionForTesting(String());
}
void WKBundlePageConfirmCompositionWithText(WKBundlePageRef pageRef, WKStringRef text)
{
toImpl(pageRef)->confirmCompositionForTesting(toWTFString(text));
}
bool WKBundlePageCanShowMIMEType(WKBundlePageRef pageRef, WKStringRef mimeTypeRef)
{
return toImpl(pageRef)->canShowMIMEType(toWTFString(mimeTypeRef));
}
WKRenderingSuppressionToken WKBundlePageExtendIncrementalRenderingSuppression(WKBundlePageRef pageRef)
{
return toImpl(pageRef)->extendIncrementalRenderingSuppression();
}
void WKBundlePageStopExtendingIncrementalRenderingSuppression(WKBundlePageRef pageRef, WKRenderingSuppressionToken token)
{
toImpl(pageRef)->stopExtendingIncrementalRenderingSuppression(token);
}
bool WKBundlePageIsUsingEphemeralSession(WKBundlePageRef pageRef)
{
return toImpl(pageRef)->usesEphemeralSession();
}
#if TARGET_OS_IPHONE
void WKBundlePageSetUseTestingViewportConfiguration(WKBundlePageRef pageRef, bool useTestingViewportConfiguration)
{
toImpl(pageRef)->setUseTestingViewportConfiguration(useTestingViewportConfiguration);
}
#endif
void WKBundlePageStartMonitoringScrollOperations(WKBundlePageRef pageRef)
{
WebKit::WebPage* webPage = toImpl(pageRef);
WebCore::Page* page = webPage ? webPage->corePage() : nullptr;
if (!page)
return;
page->ensureTestTrigger();
}
void WKBundlePageRegisterScrollOperationCompletionCallback(WKBundlePageRef pageRef, WKBundlePageTestNotificationCallback callback, void* context)
{
if (!callback)
return;
WebKit::WebPage* webPage = toImpl(pageRef);
WebCore::Page* page = webPage ? webPage->corePage() : nullptr;
if (!page || !page->expectsWheelEventTriggers())
return;
page->ensureTestTrigger().setTestCallbackAndStartNotificationTimer([=]() {
callback(context);
});
}
void WKBundlePagePostMessage(WKBundlePageRef pageRef, WKStringRef messageNameRef, WKTypeRef messageBodyRef)
{
toImpl(pageRef)->postMessage(toWTFString(messageNameRef), toImpl(messageBodyRef));
}
void WKBundlePagePostSynchronousMessage(WKBundlePageRef pageRef, WKStringRef messageNameRef, WKTypeRef messageBodyRef, WKTypeRef* returnDataRef)
{
RefPtr<API::Object> returnData;
toImpl(pageRef)->postSynchronousMessage(toWTFString(messageNameRef), toImpl(messageBodyRef), returnData);
if (returnDataRef)
*returnDataRef = toAPI(returnData.release().leakRef());
}