260 lines
		
	
	
	
		
			9.5 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			260 lines
		
	
	
	
		
			9.5 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
// Copyright (c) 2017 GitHub, Inc.
 | 
						|
// Use of this source code is governed by the MIT license that can be
 | 
						|
// found in the LICENSE file.
 | 
						|
 | 
						|
#include "atom/browser/ui/webui/pdf_viewer_ui.h"
 | 
						|
 | 
						|
#include <map>
 | 
						|
#include <memory>
 | 
						|
#include <utility>
 | 
						|
 | 
						|
#include "atom/browser/atom_browser_context.h"
 | 
						|
#include "atom/browser/loader/layered_resource_handler.h"
 | 
						|
#include "atom/browser/ui/webui/pdf_viewer_handler.h"
 | 
						|
#include "atom/common/api/api_messages.h"
 | 
						|
#include "atom/common/atom_constants.h"
 | 
						|
#include "base/sequenced_task_runner_helpers.h"
 | 
						|
#include "base/task/post_task.h"
 | 
						|
#include "content/browser/loader/resource_dispatcher_host_impl.h"
 | 
						|
#include "content/browser/loader/resource_request_info_impl.h"
 | 
						|
#include "content/browser/loader/stream_resource_handler.h"
 | 
						|
#include "content/browser/resource_context_impl.h"
 | 
						|
#include "content/browser/streams/stream.h"
 | 
						|
#include "content/browser/streams/stream_context.h"
 | 
						|
#include "content/public/browser/browser_task_traits.h"
 | 
						|
#include "content/public/browser/browser_thread.h"
 | 
						|
#include "content/public/browser/render_frame_host.h"
 | 
						|
#include "content/public/browser/render_process_host.h"
 | 
						|
#include "content/public/browser/render_view_host.h"
 | 
						|
#include "content/public/browser/resource_context.h"
 | 
						|
#include "content/public/browser/stream_handle.h"
 | 
						|
#include "content/public/browser/stream_info.h"
 | 
						|
#include "content/public/browser/url_data_source.h"
 | 
						|
#include "content/public/browser/web_contents.h"
 | 
						|
#include "grit/pdf_viewer_resources_map.h"
 | 
						|
#include "net/base/load_flags.h"
 | 
						|
#include "net/base/mime_util.h"
 | 
						|
#include "net/url_request/url_request.h"
 | 
						|
#include "net/url_request/url_request_context.h"
 | 
						|
#include "services/network/public/cpp/resource_response.h"
 | 
						|
#include "ui/base/resource/resource_bundle.h"
 | 
						|
 | 
						|
using content::BrowserThread;
 | 
						|
 | 
						|
namespace atom {
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
// Extracts the path value from the URL without the leading '/',
 | 
						|
// which follows the mapping of names in pdf_viewer_resources_map.
 | 
						|
std::string PathWithoutParams(const std::string& path) {
 | 
						|
  return GURL(kPdfViewerUIOrigin + path).path().substr(1);
 | 
						|
}
 | 
						|
 | 
						|
class BundledDataSource : public content::URLDataSource {
 | 
						|
 public:
 | 
						|
  BundledDataSource() {
 | 
						|
    for (size_t i = 0; i < kPdfViewerResourcesSize; ++i) {
 | 
						|
      std::string resource_path = kPdfViewerResources[i].name;
 | 
						|
      DCHECK(path_to_resource_id_.find(resource_path) ==
 | 
						|
             path_to_resource_id_.end());
 | 
						|
      path_to_resource_id_[resource_path] = kPdfViewerResources[i].value;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // content::URLDataSource implementation.
 | 
						|
  std::string GetSource() const override { return kPdfViewerUIHost; }
 | 
						|
 | 
						|
  void StartDataRequest(
 | 
						|
      const std::string& path,
 | 
						|
      const content::ResourceRequestInfo::WebContentsGetter& wc_getter,
 | 
						|
      const GotDataCallback& callback) override {
 | 
						|
    std::string filename = PathWithoutParams(path);
 | 
						|
    auto entry = path_to_resource_id_.find(filename);
 | 
						|
 | 
						|
    if (entry != path_to_resource_id_.end()) {
 | 
						|
      int resource_id = entry->second;
 | 
						|
      const ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
 | 
						|
      callback.Run(rb.LoadDataResourceBytes(resource_id));
 | 
						|
    } else {
 | 
						|
      LOG(ERROR) << "Unable to find: " << path;
 | 
						|
      callback.Run(new base::RefCountedString());
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  std::string GetMimeType(const std::string& path) const override {
 | 
						|
    base::FilePath::StringType ext =
 | 
						|
        base::FilePath::FromUTF8Unsafe(PathWithoutParams(path)).Extension();
 | 
						|
    std::string mime_type;
 | 
						|
    if (!ext.empty() &&
 | 
						|
        net::GetWellKnownMimeTypeFromExtension(ext.substr(1), &mime_type))
 | 
						|
      return mime_type;
 | 
						|
    return "text/html";
 | 
						|
  }
 | 
						|
 | 
						|
  bool ShouldAddContentSecurityPolicy() const override { return false; }
 | 
						|
 | 
						|
  bool ShouldDenyXFrameOptions() const override { return false; }
 | 
						|
 | 
						|
  bool ShouldServeMimeTypeAsContentTypeHeader() const override { return true; }
 | 
						|
 | 
						|
 private:
 | 
						|
  ~BundledDataSource() override {}
 | 
						|
 | 
						|
  // A map from a resource path to the resource ID.
 | 
						|
  std::map<std::string, int> path_to_resource_id_;
 | 
						|
 | 
						|
  DISALLOW_COPY_AND_ASSIGN(BundledDataSource);
 | 
						|
};
 | 
						|
 | 
						|
// Helper to convert from OnceCallback to Callback.
 | 
						|
template <typename T>
 | 
						|
void CallMigrationCallback(T callback,
 | 
						|
                           std::unique_ptr<content::StreamInfo> stream_info) {
 | 
						|
  std::move(callback).Run(std::move(stream_info));
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace
 | 
						|
 | 
						|
class PdfViewerUI::ResourceRequester
 | 
						|
    : public base::RefCountedThreadSafe<ResourceRequester,
 | 
						|
                                        BrowserThread::DeleteOnIOThread>,
 | 
						|
      public atom::LayeredResourceHandler::Delegate {
 | 
						|
 public:
 | 
						|
  explicit ResourceRequester(StreamResponseCallback cb)
 | 
						|
      : stream_response_cb_(std::move(cb)) {}
 | 
						|
 | 
						|
  void StartRequest(const GURL& url,
 | 
						|
                    const GURL& origin,
 | 
						|
                    int render_process_id,
 | 
						|
                    int render_view_id,
 | 
						|
                    int render_frame_id,
 | 
						|
                    content::ResourceContext* resource_context) {
 | 
						|
    DCHECK_CURRENTLY_ON(BrowserThread::IO);
 | 
						|
 | 
						|
    const net::URLRequestContext* request_context =
 | 
						|
        resource_context->GetRequestContext();
 | 
						|
    std::unique_ptr<net::URLRequest> request(
 | 
						|
        request_context->CreateRequest(url, net::DEFAULT_PRIORITY, nullptr));
 | 
						|
    request->set_method("GET");
 | 
						|
 | 
						|
    content::ResourceDispatcherHostImpl::Get()->InitializeURLRequest(
 | 
						|
        request.get(), content::Referrer(url, blink::kWebReferrerPolicyDefault),
 | 
						|
        false,  // download.
 | 
						|
        render_process_id, render_view_id, render_frame_id,
 | 
						|
        content::PREVIEWS_OFF, resource_context);
 | 
						|
 | 
						|
    content::ResourceRequestInfoImpl* info =
 | 
						|
        content::ResourceRequestInfoImpl::ForRequest(request.get());
 | 
						|
    content::StreamContext* stream_context =
 | 
						|
        content::GetStreamContextForResourceContext(resource_context);
 | 
						|
 | 
						|
    std::unique_ptr<content::ResourceHandler> handler =
 | 
						|
        std::make_unique<content::StreamResourceHandler>(
 | 
						|
            request.get(), stream_context->registry(), origin, false);
 | 
						|
    info->set_is_stream(true);
 | 
						|
    stream_info_.reset(new content::StreamInfo);
 | 
						|
    stream_info_->handle =
 | 
						|
        static_cast<content::StreamResourceHandler*>(handler.get())
 | 
						|
            ->stream()
 | 
						|
            ->CreateHandle();
 | 
						|
    stream_info_->original_url = request->url();
 | 
						|
 | 
						|
    // Helper to fill stream response details.
 | 
						|
    handler.reset(new atom::LayeredResourceHandler(request.get(),
 | 
						|
                                                   std::move(handler), this));
 | 
						|
 | 
						|
    content::ResourceDispatcherHostImpl::Get()->BeginURLRequest(
 | 
						|
        std::move(request), std::move(handler),
 | 
						|
        false,  // download
 | 
						|
        false,  // content_initiated (download specific)
 | 
						|
        false,  // do_not_prompt_for_login (download specific)
 | 
						|
        resource_context);
 | 
						|
  }
 | 
						|
 | 
						|
 protected:
 | 
						|
  // atom::LayeredResourceHandler::Delegate:
 | 
						|
  void OnResponseStarted(network::ResourceResponse* response) override {
 | 
						|
    DCHECK_CURRENTLY_ON(BrowserThread::IO);
 | 
						|
 | 
						|
    auto resource_response_head = response->head;
 | 
						|
    auto headers = resource_response_head.headers;
 | 
						|
    auto mime_type = resource_response_head.mime_type;
 | 
						|
    if (headers.get())
 | 
						|
      stream_info_->response_headers =
 | 
						|
          new net::HttpResponseHeaders(headers->raw_headers());
 | 
						|
    stream_info_->mime_type = mime_type;
 | 
						|
 | 
						|
    base::PostTaskWithTraits(
 | 
						|
        FROM_HERE, {BrowserThread::UI},
 | 
						|
        base::Bind(&CallMigrationCallback<StreamResponseCallback>,
 | 
						|
                   base::Passed(&stream_response_cb_),
 | 
						|
                   base::Passed(&stream_info_)));
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  friend struct BrowserThread::DeleteOnThread<BrowserThread::IO>;
 | 
						|
  friend class base::DeleteHelper<ResourceRequester>;
 | 
						|
  ~ResourceRequester() override {}
 | 
						|
 | 
						|
  StreamResponseCallback stream_response_cb_;
 | 
						|
  std::unique_ptr<content::StreamInfo> stream_info_;
 | 
						|
 | 
						|
  DISALLOW_COPY_AND_ASSIGN(ResourceRequester);
 | 
						|
};
 | 
						|
 | 
						|
PdfViewerUI::PdfViewerUI(content::BrowserContext* browser_context,
 | 
						|
                         content::WebUI* web_ui,
 | 
						|
                         const std::string& src)
 | 
						|
    : content::WebUIController(web_ui),
 | 
						|
      content::WebContentsObserver(web_ui->GetWebContents()),
 | 
						|
      src_(src) {
 | 
						|
  pdf_handler_ = new PdfViewerHandler(src);
 | 
						|
  web_ui->AddMessageHandler(
 | 
						|
      std::unique_ptr<content::WebUIMessageHandler>(pdf_handler_));
 | 
						|
  content::URLDataSource::Add(browser_context, new BundledDataSource);
 | 
						|
}
 | 
						|
 | 
						|
PdfViewerUI::~PdfViewerUI() {}
 | 
						|
 | 
						|
bool PdfViewerUI::OnMessageReceived(
 | 
						|
    const IPC::Message& message,
 | 
						|
    content::RenderFrameHost* render_frame_host) {
 | 
						|
  bool handled = true;
 | 
						|
  IPC_BEGIN_MESSAGE_MAP(PdfViewerUI, message)
 | 
						|
    IPC_MESSAGE_HANDLER(AtomFrameHostMsg_PDFSaveURLAs, OnSaveURLAs)
 | 
						|
    IPC_MESSAGE_UNHANDLED(handled = false)
 | 
						|
  IPC_END_MESSAGE_MAP()
 | 
						|
  return handled;
 | 
						|
}
 | 
						|
 | 
						|
void PdfViewerUI::OnPdfStreamCreated(
 | 
						|
    std::unique_ptr<content::StreamInfo> stream) {
 | 
						|
  stream_ = std::move(stream);
 | 
						|
  if (pdf_handler_)
 | 
						|
    pdf_handler_->SetPdfResourceStream(stream_.get());
 | 
						|
  resource_requester_ = nullptr;
 | 
						|
}
 | 
						|
 | 
						|
void PdfViewerUI::RenderFrameCreated(content::RenderFrameHost* rfh) {
 | 
						|
  int render_process_id = rfh->GetProcess()->GetID();
 | 
						|
  int render_frame_id = rfh->GetRoutingID();
 | 
						|
  int render_view_id = rfh->GetRenderViewHost()->GetRoutingID();
 | 
						|
  auto resource_context =
 | 
						|
      web_contents()->GetBrowserContext()->GetResourceContext();
 | 
						|
  auto callback =
 | 
						|
      base::BindOnce(&PdfViewerUI::OnPdfStreamCreated, base::Unretained(this));
 | 
						|
  resource_requester_ = new ResourceRequester(std::move(callback));
 | 
						|
  base::PostTaskWithTraits(
 | 
						|
      FROM_HERE, {BrowserThread::IO},
 | 
						|
      base::Bind(&ResourceRequester::StartRequest, resource_requester_,
 | 
						|
                 GURL(src_), GURL(kPdfViewerUIOrigin), render_process_id,
 | 
						|
                 render_view_id, render_frame_id, resource_context));
 | 
						|
}
 | 
						|
 | 
						|
void PdfViewerUI::OnSaveURLAs(const GURL& url,
 | 
						|
                              const content::Referrer& referrer) {
 | 
						|
  web_contents()->SaveFrame(url, referrer);
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace atom
 |