| /* |
| * 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) |