blob: f2ba13cb21d66c2067264086926341e8490b1b84 [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/printing/print_job_worker.h"
#include <memory>
#include <string>
#include <utility>
#include "base/bind.h"
#include "base/callback.h"
#include "base/callback_helpers.h"
#include "base/compiler_specific.h"
#include "base/location.h"
#include "base/numerics/safe_conversions.h"
#include "base/single_thread_task_runner.h"
#include "base/threading/thread_task_runner_handle.h"
#include "base/values.h"
#include "build/build_config.h"
#include "build/chromeos_buildflags.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/chrome_notification_types.h"
#include "chrome/browser/printing/print_job.h"
#include "chrome/grit/generated_resources.h"
#include "components/crash/core/common/crash_keys.h"
#include "content/public/browser/browser_task_traits.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/notification_service.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/web_contents.h"
#include "printing/backend/print_backend.h"
#include "printing/print_job_constants.h"
#include "printing/printed_document.h"
#include "printing/printing_utils.h"
#include "ui/base/l10n/l10n_util.h"
#if defined(OS_ANDROID)
#include "chrome/browser/android/tab_android.h"
#include "chrome/browser/android/tab_printer.h"
#include "printing/printing_context_android.h"
#endif
#if defined(OS_WIN)
#include "base/threading/thread_restrictions.h"
#include "printing/printed_page_win.h"
#include "printing/printing_features.h"
#endif
#if (defined(OS_LINUX) || BUILDFLAG(IS_CHROMEOS_LACROS)) && defined(USE_CUPS)
#include "printing/mojom/print.mojom.h"
#endif
using content::BrowserThread;
namespace printing {
namespace {
class PrintingContextDelegate : public PrintingContext::Delegate {
public:
PrintingContextDelegate(int render_process_id, int render_frame_id);
~PrintingContextDelegate() override;
gfx::NativeView GetParentView() override;
std::string GetAppLocale() override;
// Not exposed to PrintingContext::Delegate because of dependency issues.
content::WebContents* GetWebContents();
int render_process_id() const { return render_process_id_; }
int render_frame_id() const { return render_frame_id_; }
private:
const int render_process_id_;
const int render_frame_id_;
DISALLOW_COPY_AND_ASSIGN(PrintingContextDelegate);
};
PrintingContextDelegate::PrintingContextDelegate(int render_process_id,
int render_frame_id)
: render_process_id_(render_process_id),
render_frame_id_(render_frame_id) {}
PrintingContextDelegate::~PrintingContextDelegate() {
}
gfx::NativeView PrintingContextDelegate::GetParentView() {
content::WebContents* wc = GetWebContents();
return wc ? wc->GetNativeView() : nullptr;
}
content::WebContents* PrintingContextDelegate::GetWebContents() {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
auto* rfh =
content::RenderFrameHost::FromID(render_process_id_, render_frame_id_);
return rfh ? content::WebContents::FromRenderFrameHost(rfh) : nullptr;
}
std::string PrintingContextDelegate::GetAppLocale() {
return g_browser_process->GetApplicationLocale();
}
void NotificationCallback(PrintJob* print_job,
JobEventDetails::Type detail_type,
int job_id,
PrintedDocument* document) {
auto details =
base::MakeRefCounted<JobEventDetails>(detail_type, job_id, document);
content::NotificationService::current()->Notify(
chrome::NOTIFICATION_PRINT_JOB_EVENT,
content::Source<PrintJob>(print_job),
content::Details<JobEventDetails>(details.get()));
}
#if defined(OS_WIN)
void PageNotificationCallback(PrintJob* print_job,
JobEventDetails::Type detail_type,
int job_id,
PrintedDocument* document,
PrintedPage* page) {
auto details = base::MakeRefCounted<JobEventDetails>(detail_type, job_id,
document, page);
content::NotificationService::current()->Notify(
chrome::NOTIFICATION_PRINT_JOB_EVENT,
content::Source<PrintJob>(print_job),
content::Details<JobEventDetails>(details.get()));
}
#endif
} // namespace
PrintJobWorker::PrintJobWorker(int render_process_id, int render_frame_id)
: printing_context_delegate_(
std::make_unique<PrintingContextDelegate>(render_process_id,
render_frame_id)),
printing_context_(
PrintingContext::Create(printing_context_delegate_.get())),
thread_("Printing_Worker") {
DCHECK_CURRENTLY_ON(content::BrowserThread::IO);
}
PrintJobWorker::~PrintJobWorker() {
// The object is normally deleted by PrintJob in the UI thread, but when the
// user cancels printing or in the case of print preview, the worker is
// destroyed with the PrinterQuery, which is on the I/O thread.
if (print_job_) {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
} else {
DCHECK_CURRENTLY_ON(content::BrowserThread::IO);
}
Stop();
}
void PrintJobWorker::SetPrintJob(PrintJob* print_job) {
DCHECK_EQ(page_number_, PageNumber::npos());
print_job_ = print_job;
}
void PrintJobWorker::GetSettings(bool ask_user_for_settings,
uint32_t document_page_count,
bool has_selection,
mojom::MarginType margin_type,
bool is_scripted,
bool is_modifiable,
SettingsCallback callback) {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
DCHECK_EQ(page_number_, PageNumber::npos());
printing_context_->set_margin_type(margin_type);
printing_context_->set_is_modifiable(is_modifiable);
// When we delegate to a destination, we don't ask the user for settings.
// TODO(mad): Ask the destination for settings.
if (ask_user_for_settings) {
content::GetUIThreadTaskRunner({})->PostTask(
FROM_HERE,
base::BindOnce(&PrintJobWorker::GetSettingsWithUI,
base::Unretained(this), document_page_count,
has_selection, is_scripted, std::move(callback)));
} else {
content::GetUIThreadTaskRunner({})->PostTask(
FROM_HERE, base::BindOnce(&PrintJobWorker::UseDefaultSettings,
base::Unretained(this), std::move(callback)));
}
}
void PrintJobWorker::SetSettings(base::Value new_settings,
SettingsCallback callback) {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
content::GetUIThreadTaskRunner({})->PostTask(
FROM_HERE, base::BindOnce(&PrintJobWorker::UpdatePrintSettings,
base::Unretained(this), std::move(new_settings),
std::move(callback)));
}
#if BUILDFLAG(IS_CHROMEOS_ASH)
void PrintJobWorker::SetSettingsFromPOD(
std::unique_ptr<PrintSettings> new_settings,
SettingsCallback callback) {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
content::GetUIThreadTaskRunner({})->PostTask(
FROM_HERE, base::BindOnce(&PrintJobWorker::UpdatePrintSettingsFromPOD,
base::Unretained(this), std::move(new_settings),
std::move(callback)));
}
#endif
void PrintJobWorker::UpdatePrintSettings(base::Value new_settings,
SettingsCallback callback) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
std::unique_ptr<crash_keys::ScopedPrinterInfo> crash_key;
PrinterType type = static_cast<PrinterType>(
new_settings.FindIntKey(kSettingPrinterType).value());
if (type == PrinterType::kLocal) {
#if defined(OS_WIN)
// Blocking is needed here because Windows printer drivers are oftentimes
// not thread-safe and have to be accessed on the UI thread.
base::ScopedAllowBlocking allow_blocking;
#endif
scoped_refptr<PrintBackend> print_backend =
PrintBackend::CreateInstance(g_browser_process->GetApplicationLocale());
std::string printer_name = *new_settings.FindStringKey(kSettingDeviceName);
crash_key = std::make_unique<crash_keys::ScopedPrinterInfo>(
print_backend->GetPrinterDriverInfo(printer_name));
#if defined(OS_LINUX) && defined(USE_CUPS)
PrinterBasicInfo basic_info;
if (print_backend->GetPrinterBasicInfo(printer_name, &basic_info) ==
mojom::ResultCode::kSuccess) {
base::Value advanced_settings(base::Value::Type::DICTIONARY);
for (const auto& pair : basic_info.options)
advanced_settings.SetStringKey(pair.first, pair.second);
new_settings.SetKey(kSettingAdvancedSettings,
std::move(advanced_settings));
}
#endif // defined(OS_LINUX) && defined(USE_CUPS)
}
PrintingContext::Result result;
{
#if defined(OS_WIN)
// Blocking is needed here because Windows printer drivers are oftentimes
// not thread-safe and have to be accessed on the UI thread.
base::ScopedAllowBlocking allow_blocking;
#endif
result = printing_context_->UpdatePrintSettings(std::move(new_settings));
}
GetSettingsDone(std::move(callback), result);
}
#if BUILDFLAG(IS_CHROMEOS_ASH)
void PrintJobWorker::UpdatePrintSettingsFromPOD(
std::unique_ptr<PrintSettings> new_settings,
SettingsCallback callback) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
PrintingContext::Result result =
printing_context_->UpdatePrintSettingsFromPOD(std::move(new_settings));
GetSettingsDone(std::move(callback), result);
}
#endif
void PrintJobWorker::GetSettingsDone(SettingsCallback callback,
PrintingContext::Result result) {
std::move(callback).Run(printing_context_->TakeAndResetSettings(), result);
}
void PrintJobWorker::GetSettingsWithUI(uint32_t document_page_count,
bool has_selection,
bool is_scripted,
SettingsCallback callback) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
if (document_page_count > kMaxPageCount) {
GetSettingsDone(std::move(callback), PrintingContext::Result::FAILED);
return;
}
content::WebContents* web_contents = GetWebContents();
#if defined(OS_ANDROID)
if (is_scripted) {
TabAndroid* tab =
web_contents ? TabAndroid::FromWebContents(web_contents) : nullptr;
// Regardless of whether the following call fails or not, the javascript
// call will return since startPendingPrint will make it return immediately
// in case of error.
if (tab) {
auto* printing_context_delegate = static_cast<PrintingContextDelegate*>(
printing_context_delegate_.get());
PrintingContextAndroid::SetPendingPrint(
web_contents->GetTopLevelNativeWindow(),
GetPrintableForTab(tab->GetJavaObject()),
printing_context_delegate->render_process_id(),
printing_context_delegate->render_frame_id());
}
}
#endif
// Running a dialog causes an exit to webpage-initiated fullscreen.
// http://crbug.com/728276
if (web_contents && web_contents->IsFullscreen())
web_contents->ExitFullscreen(true);
printing_context_->AskUserForSettings(
base::checked_cast<int>(document_page_count), has_selection, is_scripted,
base::BindOnce(&PrintJobWorker::GetSettingsDone,
weak_factory_.GetWeakPtr(), std::move(callback)));
}
void PrintJobWorker::UseDefaultSettings(SettingsCallback callback) {
PrintingContext::Result result;
{
#if defined(OS_WIN)
// Blocking is needed here because Windows printer drivers are oftentimes
// not thread-safe and have to be accessed on the UI thread.
base::ScopedAllowBlocking allow_blocking;
#endif
result = printing_context_->UseDefaultSettings();
}
GetSettingsDone(std::move(callback), result);
}
void PrintJobWorker::StartPrinting(PrintedDocument* new_document) {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
if (page_number_ != PageNumber::npos()) {
NOTREACHED();
return;
}
if (!document_) {
NOTREACHED();
return;
}
if (document_.get() != new_document) {
NOTREACHED();
return;
}
std::u16string document_name = SimplifyDocumentTitle(document_->name());
if (document_name.empty()) {
document_name = SimplifyDocumentTitle(
l10n_util::GetStringUTF16(IDS_DEFAULT_PRINT_DOCUMENT_TITLE));
}
PrintingContext::Result result =
printing_context_->NewDocument(document_name);
if (result != PrintingContext::OK) {
OnFailure();
return;
}
// This will start a loop to wait for the page data.
OnNewPage();
// Don't touch this anymore since the instance could be destroyed. It happens
// if all the pages are printed a one sweep and the client doesn't have a
// handle to us anymore. There's a timing issue involved between the worker
// thread and the UI thread. Take no chance.
}
void PrintJobWorker::OnDocumentChanged(PrintedDocument* new_document) {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
if (page_number_ != PageNumber::npos()) {
NOTREACHED();
return;
}
document_ = new_document;
}
void PrintJobWorker::PostWaitForPage() {
// We need to wait for the page to be available.
base::ThreadTaskRunnerHandle::Get()->PostDelayedTask(
FROM_HERE,
base::BindOnce(&PrintJobWorker::OnNewPage, weak_factory_.GetWeakPtr()),
base::TimeDelta::FromMilliseconds(500));
}
void PrintJobWorker::OnNewPage() {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
if (!document_)
return;
bool do_spool_job = true;
#if defined(OS_WIN)
const bool source_is_pdf =
!print_job_->document()->settings().is_modifiable();
if (!features::ShouldPrintUsingXps(source_is_pdf)) {
// Using the Windows GDI print API.
if (!OnNewPageHelperGdi())
return;
do_spool_job = false;
}
#endif // defined(OS_WIN)
if (do_spool_job) {
if (!document_->GetMetafile()) {
PostWaitForPage();
return;
}
SpoolJob();
}
OnDocumentDone();
// Don't touch |this| anymore since the instance could be destroyed.
}
#if defined(OS_WIN)
bool PrintJobWorker::OnNewPageHelperGdi() {
if (page_number_ == PageNumber::npos()) {
// Find first page to print.
int page_count = document_->page_count();
if (!page_count) {
// We still don't know how many pages the document contains.
return false;
}
// We have enough information to initialize |page_number_|.
page_number_.Init(document_->settings(), page_count);
}
while (true) {
scoped_refptr<PrintedPage> page = document_->GetPage(page_number_.ToUint());
if (!page) {
PostWaitForPage();
return false;
}
// The page is there, print it.
SpoolPage(page.get());
++page_number_;
if (page_number_ == PageNumber::npos())
break;
}
return true;
}
#endif // defined(OS_WIN)
void PrintJobWorker::Cancel() {
// This is the only function that can be called from any thread.
printing_context_->Cancel();
// Cannot touch any member variable since we don't know in which thread
// context we run.
}
bool PrintJobWorker::IsRunning() const {
return thread_.IsRunning();
}
bool PrintJobWorker::PostTask(const base::Location& from_here,
base::OnceClosure task) {
return task_runner_ && task_runner_->PostTask(from_here, std::move(task));
}
void PrintJobWorker::StopSoon() {
thread_.StopSoon();
}
void PrintJobWorker::Stop() {
thread_.Stop();
}
bool PrintJobWorker::Start() {
bool result = thread_.Start();
task_runner_ = thread_.task_runner();
return result;
}
void PrintJobWorker::OnDocumentDone() {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
DCHECK_EQ(page_number_, PageNumber::npos());
DCHECK(document_);
// PrintJob must own this, because only PrintJob can send notifications.
DCHECK(print_job_);
int job_id = printing_context_->job_id();
if (printing_context_->DocumentDone() != PrintingContext::OK) {
OnFailure();
return;
}
print_job_->PostTask(
FROM_HERE,
base::BindOnce(&NotificationCallback, base::RetainedRef(print_job_),
JobEventDetails::DOC_DONE, job_id,
base::RetainedRef(document_)));
// Makes sure the variables are reinitialized.
document_ = nullptr;
}
#if defined(OS_WIN)
void PrintJobWorker::SpoolPage(PrintedPage* page) {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
DCHECK_NE(page_number_, PageNumber::npos());
// Preprocess.
if (printing_context_->NewPage() != PrintingContext::OK) {
OnFailure();
return;
}
// Actual printing.
document_->RenderPrintedPage(*page, printing_context_->context());
// Postprocess.
if (printing_context_->PageDone() != PrintingContext::OK) {
OnFailure();
return;
}
// Signal everyone that the page is printed.
DCHECK(print_job_);
print_job_->PostTask(
FROM_HERE,
base::BindOnce(&PageNotificationCallback, base::RetainedRef(print_job_),
JobEventDetails::PAGE_DONE, printing_context_->job_id(),
base::RetainedRef(document_), base::RetainedRef(page)));
}
#endif // defined(OS_WIN)
void PrintJobWorker::SpoolJob() {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
if (!document_->RenderPrintedDocument(printing_context_.get()))
OnFailure();
}
void PrintJobWorker::OnFailure() {
DCHECK(task_runner_->RunsTasksInCurrentSequence());
DCHECK(print_job_);
// We may loose our last reference by broadcasting the FAILED event.
scoped_refptr<PrintJob> handle(print_job_);
print_job_->PostTask(
FROM_HERE,
base::BindOnce(&NotificationCallback, base::RetainedRef(print_job_),
JobEventDetails::FAILED, 0, base::RetainedRef(document_)));
Cancel();
// Makes sure the variables are reinitialized.
document_ = nullptr;
page_number_ = PageNumber::npos();
}
content::WebContents* PrintJobWorker::GetWebContents() {
PrintingContextDelegate* printing_context_delegate =
static_cast<PrintingContextDelegate*>(printing_context_delegate_.get());
return printing_context_delegate->GetWebContents();
}
} // namespace printing