| /* |
| * 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 "AddEventListenerOptionsInlines.h" |
| #include "DocumentLoader.h" |
| #include "EventListener.h" |
| #include "EventNames.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 "LocalDOMWindow.h" |
| #include "LocalFrame.h" |
| #include "RawDataDocumentParser.h" |
| #include "ScriptController.h" |
| #include "Settings.h" |
| #include "UserScriptTypes.h" |
| #include "WindowPostMessageOptions.h" |
| #include <JavaScriptCore/ObjectConstructor.h> |
| #include <wtf/TZoneMallocInlines.h> |
| |
| namespace WebCore { |
| |
| WTF_MAKE_TZONE_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: |
| explicit PDFDocumentParser(PDFDocument& document) |
| : RawDataDocumentParser(document) |
| { |
| } |
| |
| PDFDocument& document() const; |
| |
| void appendBytes(DocumentWriter&, std::span<const uint8_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&, std::span<const uint8_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, WeakPtrImplWithEventTargetData> 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->finishLoadingPDF(); |
| } 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(LocalFrame& frame, const URL& url) |
| : HTMLDocument(&frame, frame.settings(), url, { }, { DocumentClass::PDF }) |
| { |
| } |
| |
| PDFDocument::~PDFDocument() = default; |
| |
| Ref<DocumentParser> PDFDocument::createParser() |
| { |
| return PDFDocumentParser::create(*this); |
| } |
| |
| void PDFDocument::createDocumentStructure() |
| { |
| // Description of parameters: |
| // - Empty `?file=` parameter prevents default pdf from loading. |
| auto viewerURL = "webkit-pdfjs-viewer://pdfjs/web/viewer.html?file="_s; |
| Ref rootElement = HTMLHtmlElement::create(*this); |
| appendChild(rootElement); |
| |
| frame()->injectUserScripts(UserScriptInjectionTime::DocumentStart); |
| |
| Ref 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) |
| finishLoadingPDF(); |
| } |
| |
| void PDFDocument::postMessageToIframe(const String& name, JSC::JSObject* data) |
| { |
| auto globalObject = this->globalObject(); |
| auto& vm = globalObject->vm(); |
| JSC::JSLockHolder lock(vm); |
| |
| JSC::JSObject* message = constructEmptyObject(globalObject); |
| message->putDirect(vm, vm.propertyNames->message, JSC::jsNontrivialString(vm, name)); |
| if (data) |
| message->putDirect(vm, JSC::Identifier::fromString(vm, "data"_s), data); |
| |
| auto* contentFrame = dynamicDowncast<LocalFrame>(m_iframe->contentFrame()); |
| if (!contentFrame) |
| return; |
| auto* contentWindow = contentFrame->window(); |
| auto* contentWindowGlobalObject = m_iframe->contentDocument()->globalObject(); |
| |
| WindowPostMessageOptions options; |
| if (data) |
| options = WindowPostMessageOptions { "/"_s, Vector { JSC::Strong<JSC::JSObject> { vm, data } } }; |
| auto returnValue = contentWindow->postMessage(*contentWindowGlobalObject, *contentWindow, message, WTF::move(options)); |
| if (returnValue.hasException()) |
| returnValue.releaseException(); |
| } |
| |
| void PDFDocument::sendPDFArrayBuffer() |
| { |
| auto* documentLoader = loader(); |
| ASSERT(documentLoader); |
| if (auto mainResourceData = documentLoader->mainResourceData()) { |
| if (auto arrayBuffer = mainResourceData->tryCreateArrayBuffer()) { |
| auto& vm = globalObject()->vm(); |
| JSC::JSLockHolder lock(vm); |
| auto* dataObject = JSC::JSArrayBuffer::create(vm, globalObject()->arrayBufferStructure(arrayBuffer->sharingMode()), WTF::move(arrayBuffer)); |
| postMessageToIframe("open-pdf"_s, dataObject); |
| } |
| } |
| } |
| |
| void PDFDocument::finishLoadingPDF() |
| { |
| sendPDFArrayBuffer(); |
| |
| if (m_script) { |
| m_script->removeEventListener(eventNames().loadEvent, *m_listener, { }); |
| m_script = nullptr; |
| } |
| |
| m_listener = nullptr; |
| } |
| |
| void PDFDocument::injectStyleAndContentScript() |
| { |
| if (m_injectedStyleAndScript) |
| return; |
| |
| auto* contentDocument = m_iframe->contentDocument(); |
| ASSERT(contentDocument->head()); |
| Ref 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()); |
| m_script = HTMLScriptElement::create(scriptTag, *contentDocument, false); |
| ASSERT(m_listener); |
| m_script->addEventListener(eventNames().loadEvent, *m_listener, false); |
| m_script->setAttribute(srcAttr, "webkit-pdfjs-viewer://pdfjs/extras/content-script.js"_s); |
| contentDocument->body()->appendChild(*m_script); |
| |
| m_injectedStyleAndScript = true; |
| } |
| |
| } // namepsace WebCore |
| |
| #endif // ENABLE(PDFJS) |