blob: cce0a7c38d428a31ae0b57b45ffa10c8213d86e6 [file] [log] [blame]
/*
* Copyright (C) 2022 Apple Inc. All rights reserved.
*
* 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``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 INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* 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.
*/
#include "config.h"
#include "PDFDocument.h"
#if ENABLE(PDFJS)
#include "AddEventListenerOptions.h"
#include "DOMWindow.h"
#include "DocumentLoader.h"
#include "EventListener.h"
#include "EventNames.h"
#include "Frame.h"
#include "FrameDestructionObserverInlines.h"
#include "HTMLAnchorElement.h"
#include "HTMLBodyElement.h"
#include "HTMLHeadElement.h"
#include "HTMLHtmlElement.h"
#include "HTMLIFrameElement.h"
#include "HTMLLinkElement.h"
#include "HTMLNames.h"
#include "HTMLScriptElement.h"
#include "RawDataDocumentParser.h"
#include "ScriptController.h"
#include "Settings.h"
#include <wtf/IsoMallocInlines.h>
#include <wtf/text/StringConcatenateNumbers.h>
namespace WebCore {
WTF_MAKE_ISO_ALLOCATED_IMPL(PDFDocument);
using namespace HTMLNames;
/* PDFDocumentParser: this receives the PDF bytes */
class PDFDocumentParser final : public RawDataDocumentParser {
public:
static Ref<PDFDocumentParser> create(PDFDocument& document)
{
return adoptRef(*new PDFDocumentParser(document));
}
private:
PDFDocumentParser(PDFDocument& document)
: RawDataDocumentParser(document)
{
}
PDFDocument& document() const;
void appendBytes(DocumentWriter&, const uint8_t*, size_t) override;
void finish() override;
};
inline PDFDocument& PDFDocumentParser::document() const
{
// Only used during parsing, so document is guaranteed to be non-null.
ASSERT(RawDataDocumentParser::document());
return downcast<PDFDocument>(*RawDataDocumentParser::document());
}
void PDFDocumentParser::appendBytes(DocumentWriter&, const uint8_t*, size_t)
{
document().updateDuringParsing();
}
void PDFDocumentParser::finish()
{
document().finishedParsing();
}
/* PDFDocumentEventListener: event listener for the PDFDocument iframe */
class PDFDocumentEventListener final : public EventListener {
public:
static Ref<PDFDocumentEventListener> create(PDFDocument& document) { return adoptRef(*new PDFDocumentEventListener(document)); }
private:
explicit PDFDocumentEventListener(PDFDocument& document)
: EventListener(PDFDocumentEventListenerType)
, m_document(document)
{
}
bool operator==(const EventListener&) const override;
void handleEvent(ScriptExecutionContext&, Event&) override;
WeakPtr<PDFDocument> m_document;
};
void PDFDocumentEventListener::handleEvent(ScriptExecutionContext&, Event& event)
{
if (is<HTMLIFrameElement>(event.target()) && event.type() == eventNames().loadEvent) {
m_document->injectStyleAndContentScript();
} else if (is<HTMLScriptElement>(event.target()) && event.type() == eventNames().loadEvent) {
m_document->setContentScriptLoaded(true);
if (m_document->isFinishedParsing())
m_document->sendPDFArrayBuffer();
} else
ASSERT_NOT_REACHED();
}
bool PDFDocumentEventListener::operator==(const EventListener& other) const
{
// All PDFDocumentEventListenerType objects compare as equal; OK since there is only one per document.
return other.type() == PDFDocumentEventListenerType;
}
/* PDFDocument */
PDFDocument::PDFDocument(Frame& frame, const URL& url)
: HTMLDocument(&frame, frame.settings(), url, { }, { DocumentClass::PDF })
{
}
Ref<DocumentParser> PDFDocument::createParser()
{
return PDFDocumentParser::create(*this);
}
void PDFDocument::createDocumentStructure()
{
// Description of parameters:
// - `#pagemode=none` prevents the sidebar from showing on load.
// - Empty `?file=` parameter prevents default pdf from loading.
auto viewerURL = "webkit-pdfjs-viewer://pdfjs/web/viewer.html?file=#pagemode=none"_s;
auto rootElement = HTMLHtmlElement::create(*this);
appendChild(rootElement);
rootElement->insertedByParser();
frame()->injectUserScripts(UserScriptInjectionTime::DocumentStart);
auto body = HTMLBodyElement::create(*this);
body->setAttribute(styleAttr, "margin: 0px;height: 100vh;"_s);
rootElement->appendChild(body);
m_iframe = HTMLIFrameElement::create(HTMLNames::iframeTag, *this);
m_iframe->setAttribute(srcAttr, AtomString(viewerURL));
m_iframe->setAttribute(styleAttr, "width: 100%; height: 100%; border: 0; display: block;"_s);
m_listener = PDFDocumentEventListener::create(*this);
m_iframe->addEventListener(eventNames().loadEvent, *m_listener, false);
body->appendChild(*m_iframe);
}
void PDFDocument::updateDuringParsing()
{
if (!m_iframe)
createDocumentStructure();
}
void PDFDocument::finishedParsing()
{
ASSERT(m_iframe);
m_isFinishedParsing = true;
if (m_isContentScriptLoaded)
sendPDFArrayBuffer();
}
void PDFDocument::sendPDFArrayBuffer()
{
using namespace JSC;
auto* frame = m_iframe->contentFrame();
// FIXME: https://bugs.webkit.org/show_bug.cgi?id=236668 - Use postMessage
auto openFunction = frame->script().executeScriptIgnoringException("PDFJSContentScript.open"_s).getObject();
auto globalObject = this->globalObject();
auto& vm = globalObject->vm();
JSLockHolder lock(vm);
auto callData = JSC::getCallData(openFunction);
ASSERT(callData.type != CallData::Type::None);
MarkedArgumentBuffer arguments;
auto arrayBuffer = loader()->mainResourceData()->tryCreateArrayBuffer();
if (!arrayBuffer) {
ASSERT_NOT_REACHED();
return;
}
auto sharingMode = arrayBuffer->sharingMode();
arguments.append(JSArrayBuffer::create(vm, globalObject->arrayBufferStructure(sharingMode), WTFMove(arrayBuffer)));
ASSERT(!arguments.hasOverflowed());
call(globalObject, openFunction, callData, globalObject, arguments);
}
void PDFDocument::injectStyleAndContentScript()
{
auto* contentDocument = m_iframe->contentDocument();
ASSERT(contentDocument->head());
auto link = HTMLLinkElement::create(HTMLNames::linkTag, *contentDocument, false);
link->setAttribute(relAttr, "stylesheet"_s);
#if PLATFORM(COCOA)
link->setAttribute(hrefAttr, "webkit-pdfjs-viewer://pdfjs/extras/cocoa/style.css"_s);
#elif PLATFORM(GTK) || PLATFORM(WPE)
link->setAttribute(hrefAttr, "webkit-pdfjs-viewer://pdfjs/extras/adwaita/style.css"_s);
#endif
contentDocument->head()->appendChild(link);
ASSERT(contentDocument->body());
auto script = HTMLScriptElement::create(scriptTag, *contentDocument, false);
script->addEventListener(eventNames().loadEvent, m_listener.releaseNonNull(), false);
script->setAttribute(srcAttr, "webkit-pdfjs-viewer://pdfjs/extras/content-script.js"_s);
contentDocument->body()->appendChild(script);
}
} // namepsace WebCore
#endif // ENABLE(PDFJS)