blob: 6ff16c92de1855860bf41f138f79ed9d81621de6 [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/renderer/print_web_view_helper.h"
#import <AppKit/AppKit.h>
#include "base/logging.h"
#include "base/mac/scoped_nsautorelease_pool.h"
#include "base/metrics/histogram.h"
#include "chrome/common/print_messages.h"
#include "printing/metafile.h"
#include "printing/metafile_impl.h"
#include "printing/page_size_margins.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h"
#if defined(USE_SKIA)
#include "printing/metafile_skia_wrapper.h"
#include "skia/ext/platform_device.h"
#include "skia/ext/vector_canvas.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebCanvas.h"
#endif
using WebKit::WebFrame;
void PrintWebViewHelper::PrintPageInternal(
const PrintMsg_PrintPage_Params& params,
const gfx::Size& canvas_size,
WebFrame* frame) {
printing::NativeMetafile metafile;
if (!metafile.Init())
return;
int page_number = params.page_number;
gfx::Size page_size_in_dpi;
gfx::Rect content_area_in_dpi;
RenderPage(print_pages_params_->params, page_number, frame, false, &metafile,
&page_size_in_dpi, &content_area_in_dpi);
metafile.FinishDocument();
PrintHostMsg_DidPrintPage_Params page_params;
page_params.data_size = metafile.GetDataSize();
page_params.page_number = page_number;
page_params.document_cookie = params.params.document_cookie;
page_params.page_size = page_size_in_dpi;
page_params.content_area = content_area_in_dpi;
// Ask the browser to create the shared memory for us.
if (!CopyMetafileDataToSharedMem(&metafile,
&(page_params.metafile_data_handle))) {
page_params.data_size = 0;
}
Send(new PrintHostMsg_DidPrintPage(routing_id(), page_params));
}
bool PrintWebViewHelper::RenderPreviewPage(int page_number) {
PrintMsg_Print_Params printParams = print_preview_context_.print_params();
scoped_ptr<printing::Metafile> draft_metafile;
printing::Metafile* initial_render_metafile =
print_preview_context_.metafile();
#if defined(USE_SKIA)
bool render_to_draft = print_preview_context_.IsModifiable() &&
is_print_ready_metafile_sent_;
#else
// If the page needs to be in both draft metafile and print ready metafile,
// we should always render to the draft metafile first and then copy that
// into the print ready metafile because CG does not allow us to do it in
// the other order.
bool render_to_draft = print_preview_context_.IsModifiable() &&
print_preview_context_.generate_draft_pages();
#endif
if (render_to_draft) {
draft_metafile.reset(new printing::PreviewMetafile());
if (!draft_metafile->Init()) {
print_preview_context_.set_error(
PREVIEW_ERROR_MAC_DRAFT_METAFILE_INIT_FAILED);
LOG(ERROR) << "Draft PreviewMetafile Init failed";
return false;
}
initial_render_metafile = draft_metafile.get();
}
base::TimeTicks begin_time = base::TimeTicks::Now();
gfx::Size page_size;
RenderPage(printParams, page_number, print_preview_context_.frame(), true,
initial_render_metafile, &page_size, NULL);
print_preview_context_.RenderedPreviewPage(
base::TimeTicks::Now() - begin_time);
if (draft_metafile.get()) {
draft_metafile->FinishDocument();
#if !defined(USE_SKIA)
if (!is_print_ready_metafile_sent_) {
// With CG, we rendered into a new metafile so we could get it as a draft
// document. Now we need to add it to print ready document. But the
// document has already been scaled and adjusted for margins, so do a 1:1
// drawing.
printing::Metafile* print_ready_metafile =
print_preview_context_.metafile();
bool success = print_ready_metafile->StartPage(page_size,
gfx::Rect(page_size), 1.0);
DCHECK(success);
// StartPage unconditionally flips the content over, flip it back since it
// was already flipped in |draft_metafile|.
CGContextTranslateCTM(print_ready_metafile->context(), 0,
page_size.height());
CGContextScaleCTM(print_ready_metafile->context(), 1.0, -1.0);
draft_metafile->RenderPage(1,
print_ready_metafile->context(),
draft_metafile->GetPageBounds(1).ToCGRect(),
false /* shrink_to_fit */,
false /* stretch_to_fit */,
true /* center_horizontally */,
true /* center_vertically */);
print_ready_metafile->FinishPage();
}
#endif
} else {
#if defined(USE_SKIA)
if (print_preview_context_.IsModifiable() &&
print_preview_context_.generate_draft_pages()) {
DCHECK(!draft_metafile.get());
draft_metafile.reset(
print_preview_context_.metafile()->GetMetafileForCurrentPage());
}
#endif
}
return PreviewPageRendered(page_number, draft_metafile.get());
}
void PrintWebViewHelper::RenderPage(
const PrintMsg_Print_Params& params, int page_number, WebFrame* frame,
bool is_preview, printing::Metafile* metafile, gfx::Size* page_size,
gfx::Rect* content_rect) {
double scale_factor = 1.0f;
double webkit_shrink_factor = frame->getPrintPageShrink(page_number);
printing::PageSizeMargins page_layout_in_points;
gfx::Rect content_area;
ComputePageLayoutInPointsForCss(frame, page_number, params,
ignore_css_margins_, &scale_factor,
&page_layout_in_points);
GetPageSizeAndContentAreaFromPageLayout(page_layout_in_points, page_size,
&content_area);
if (content_rect)
*content_rect = content_area;
scale_factor *= webkit_shrink_factor;
{
#if defined(USE_SKIA)
SkDevice* device = metafile->StartPageForVectorCanvas(
*page_size, content_area, scale_factor);
if (!device)
return;
SkRefPtr<skia::VectorCanvas> canvas = new skia::VectorCanvas(device);
canvas->unref(); // SkRefPtr and new both took a reference.
WebKit::WebCanvas* canvas_ptr = canvas.get();
printing::MetafileSkiaWrapper::SetMetafileOnCanvas(*canvas, metafile);
skia::SetIsDraftMode(*canvas, is_print_ready_metafile_sent_);
skia::SetIsPreviewMetafile(*canvas, is_preview);
#else
bool success = metafile->StartPage(*page_size, content_area, scale_factor);
DCHECK(success);
// printPage can create autoreleased references to |context|. PDF contexts
// don't write all their data until they are destroyed, so we need to make
// certain that there are no lingering references.
base::mac::ScopedNSAutoreleasePool pool;
CGContextRef cgContext = metafile->context();
CGContextRef canvas_ptr = cgContext;
// For CoreGraphics, print in the margins before printing in the content
// area so that we don't spill over. Webkit draws a white background in the
// content area and this acts as a clip.
// TODO(aayushkumar): Combine the calls to PrintHeaderAndFooter once we
// can draw in the margins safely in Skia in any order.
if (print_pages_params_->params.display_header_footer) {
PrintHeaderAndFooter(canvas_ptr, page_number + 1,
print_preview_context_.total_page_count(),
scale_factor, page_layout_in_points,
*header_footer_info_);
}
#endif // !USE_SKIA
frame->printPage(page_number, canvas_ptr);
#if defined(USE_SKIA)
if (print_pages_params_->params.display_header_footer) {
// |page_number| is 0-based, so 1 is added.
PrintHeaderAndFooter(canvas_ptr, page_number + 1,
print_preview_context_.total_page_count(),
scale_factor, page_layout_in_points,
*header_footer_info_);
}
#endif // defined(USE_SKIA)
}
// Done printing. Close the device context to retrieve the compiled metafile.
metafile->FinishPage();
}