blob: 87ffdf6f44029700ef615b16366e3a9805e7d3ff [file] [log] [blame]
// Copyright 2011 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "printing/printing_context.h"
#include <utility>
#include "base/check_op.h"
#include "base/notreached.h"
#include "build/build_config.h"
#include "printing/buildflags/buildflags.h"
#include "printing/mojom/print.mojom.h"
#include "printing/page_setup.h"
#include "printing/print_job_constants.h"
#include "printing/print_settings_conversion.h"
#include "printing/printing_context_factory_for_test.h"
#include "printing/units.h"
#if BUILDFLAG(ENABLE_OOP_PRINTING)
#include "printing/printing_features.h"
#endif
namespace printing {
namespace {
PrintingContextFactoryForTest* g_printing_context_factory_for_test = nullptr;
} // namespace
PrintingContext::PrintingContext(Delegate* delegate,
ProcessBehavior process_behavior)
: settings_(std::make_unique<PrintSettings>()),
delegate_(delegate),
in_print_job_(false),
abort_printing_(false),
process_behavior_(process_behavior) {
DCHECK(delegate_);
}
PrintingContext::~PrintingContext() = default;
// static
std::unique_ptr<PrintingContext> PrintingContext::Create(
Delegate* delegate,
ProcessBehavior process_behavior) {
return g_printing_context_factory_for_test
? g_printing_context_factory_for_test->CreatePrintingContext(
delegate, process_behavior)
: PrintingContext::CreateImpl(delegate, process_behavior);
}
// static
void PrintingContext::SetPrintingContextFactoryForTest(
PrintingContextFactoryForTest* factory) {
g_printing_context_factory_for_test = factory;
}
void PrintingContext::set_margin_type(mojom::MarginType type) {
DCHECK(type != mojom::MarginType::kCustomMargins);
settings_->set_margin_type(type);
}
void PrintingContext::set_is_modifiable(bool is_modifiable) {
settings_->set_is_modifiable(is_modifiable);
}
const PrintSettings& PrintingContext::settings() const {
DCHECK(!in_print_job_);
return *settings_;
}
void PrintingContext::ResetSettings() {
ReleaseContext();
settings_->Clear();
in_print_job_ = false;
abort_printing_ = false;
}
std::unique_ptr<PrintSettings> PrintingContext::TakeAndResetSettings() {
std::unique_ptr<PrintSettings> result = std::move(settings_);
settings_ = std::make_unique<PrintSettings>();
return result;
}
#if BUILDFLAG(ENABLE_OOP_PRINTING)
void PrintingContext::SetJobId(int job_id) {
// Should only use this method to update the browser `PrintingContext` with
// the value provided by the PrintBackend service.
CHECK_EQ(process_behavior_, ProcessBehavior::kOopEnabledSkipSystemCalls);
job_id_ = job_id;
}
#endif
mojom::ResultCode PrintingContext::OnError() {
mojom::ResultCode result = abort_printing_ ? mojom::ResultCode::kCanceled
: mojom::ResultCode::kFailed;
ResetSettings();
return result;
}
void PrintingContext::SetDefaultPrintableAreaForVirtualPrinters() {
gfx::Size paper_size(GetPdfPaperSizeDeviceUnits());
if (!settings_->requested_media().size_microns.IsEmpty()) {
float device_microns_per_device_unit = static_cast<float>(kMicronsPerInch) /
settings_->device_units_per_inch();
paper_size = gfx::Size(settings_->requested_media().size_microns.width() /
device_microns_per_device_unit,
settings_->requested_media().size_microns.height() /
device_microns_per_device_unit);
}
gfx::Rect paper_rect(0, 0, paper_size.width(), paper_size.height());
settings_->SetPrinterPrintableArea(paper_size, paper_rect,
/*landscape_needs_flip=*/true);
}
void PrintingContext::UsePdfSettings() {
base::Value::Dict pdf_settings;
pdf_settings.Set(kSettingHeaderFooterEnabled, false);
pdf_settings.Set(kSettingShouldPrintBackgrounds, false);
pdf_settings.Set(kSettingShouldPrintSelectionOnly, false);
pdf_settings.Set(kSettingMarginsType,
static_cast<int>(mojom::MarginType::kNoMargins));
pdf_settings.Set(kSettingCollate, true);
pdf_settings.Set(kSettingCopies, 1);
pdf_settings.Set(kSettingColor, static_cast<int>(mojom::ColorModel::kColor));
// DPI value should match GetPdfCapabilities().
pdf_settings.Set(kSettingDpiHorizontal, kDefaultPdfDpi);
pdf_settings.Set(kSettingDpiVertical, kDefaultPdfDpi);
pdf_settings.Set(kSettingDuplexMode,
static_cast<int>(printing::mojom::DuplexMode::kSimplex));
pdf_settings.Set(kSettingLandscape, false);
pdf_settings.Set(kSettingDeviceName, "");
pdf_settings.Set(kSettingPrinterType,
static_cast<int>(mojom::PrinterType::kPdf));
pdf_settings.Set(kSettingScaleFactor, 100);
pdf_settings.Set(kSettingRasterizePdf, false);
pdf_settings.Set(kSettingPagesPerSheet, 1);
mojom::ResultCode result = UpdatePrintSettings(std::move(pdf_settings));
// TODO(thestig): Downgrade these to DCHECKs after shipping these CHECKs to
// production without any failures.
CHECK_EQ(result, mojom::ResultCode::kSuccess);
// UsePdfSettings() should never fail and the returned DPI should always be a
// well-known value that is safe to use as a divisor.
#if BUILDFLAG(IS_MAC)
CHECK_EQ(settings_->device_units_per_inch(), kPointsPerInch);
#else
CHECK_EQ(settings_->device_units_per_inch(), kDefaultPdfDpi);
#endif
}
mojom::ResultCode PrintingContext::UpdatePrintSettings(
base::Value::Dict job_settings) {
ResetSettings();
{
std::unique_ptr<PrintSettings> settings =
PrintSettingsFromJobSettings(job_settings);
if (!settings) {
DUMP_WILL_BE_NOTREACHED_NORETURN();
return OnError();
}
settings_ = std::move(settings);
}
mojom::PrinterType printer_type = static_cast<mojom::PrinterType>(
job_settings.FindInt(kSettingPrinterType).value());
if (printer_type == mojom::PrinterType::kPrivetDeprecated ||
printer_type == mojom::PrinterType::kCloudDeprecated) {
NOTREACHED();
return OnError();
}
bool open_in_external_preview =
job_settings.contains(kSettingOpenPDFInPreview);
if (!open_in_external_preview &&
(printer_type == mojom::PrinterType::kPdf ||
printer_type == mojom::PrinterType::kExtension)) {
if (settings_->page_setup_device_units().printable_area().IsEmpty())
SetDefaultPrintableAreaForVirtualPrinters();
return mojom::ResultCode::kSuccess;
}
// The `open_in_external_preview` case does not care about the printable area.
// Local printers set their printable area within UpdatePrinterSettings().
DCHECK(open_in_external_preview ||
printer_type == mojom::PrinterType::kLocal);
PrinterSettings printer_settings {
#if BUILDFLAG(IS_MAC)
.external_preview = open_in_external_preview,
#endif
.show_system_dialog =
job_settings.FindBool(kSettingShowSystemDialog).value_or(false),
#if BUILDFLAG(IS_WIN)
.page_count = job_settings.FindInt(kSettingPreviewPageCount).value_or(0)
#endif
};
return UpdatePrinterSettings(printer_settings);
}
#if BUILDFLAG(IS_CHROMEOS)
mojom::ResultCode PrintingContext::UpdatePrintSettingsFromPOD(
std::unique_ptr<PrintSettings> job_settings) {
ResetSettings();
settings_ = std::move(job_settings);
return UpdatePrinterSettings({.show_system_dialog = false});
}
#endif
void PrintingContext::SetPrintSettings(const PrintSettings& settings) {
*settings_ = settings;
}
} // namespace printing