Chromium Code Reviews| Index: chrome/renderer/print_web_view_helper.cc | 
| diff --git a/chrome/renderer/print_web_view_helper.cc b/chrome/renderer/print_web_view_helper.cc | 
| index 1edc892a397aef33dd43754b5eb070d0da326bc5..f8ef3dd98727b55899bda4d5243e73c60ac91e17 100644 | 
| --- a/chrome/renderer/print_web_view_helper.cc | 
| +++ b/chrome/renderer/print_web_view_helper.cc | 
| @@ -4,12 +4,17 @@ | 
| #include "chrome/renderer/print_web_view_helper.h" | 
| +#include <algorithm> | 
| +#include <cmath> | 
| #include <string> | 
| #include "base/command_line.h" | 
| +#include "base/i18n/time_formatting.h" | 
| #include "base/logging.h" | 
| #include "base/metrics/histogram.h" | 
| #include "base/process_util.h" | 
| +#include "base/string_number_conversions.h" | 
| +#include "base/time.h" | 
| #include "base/utf_string_conversions.h" | 
| #include "chrome/common/chrome_switches.h" | 
| #include "chrome/common/print_messages.h" | 
| @@ -21,6 +26,9 @@ | 
| #include "printing/metafile_impl.h" | 
| #include "printing/print_job_constants.h" | 
| #include "printing/units.h" | 
| +#include "skia/ext/vector_canvas.h" | 
| +#include "skia/ext/vector_platform_device_skia.h" | 
| +#include "third_party/skia/include/core/SkTypeface.h" | 
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebConsoleMessage.h" | 
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebDataSource.h" | 
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h" | 
| @@ -32,13 +40,16 @@ | 
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebURLResponse.h" | 
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h" | 
| #include "ui/base/l10n/l10n_util.h" | 
| +#include "ui/base/text/text_elider.h" | 
| #if defined(OS_POSIX) | 
| #include "content/common/view_messages.h" | 
| #endif | 
| +using base::Time; | 
| using printing::ConvertPixelsToPoint; | 
| using printing::ConvertPixelsToPointDouble; | 
| +using printing::ConvertPointsToPixelDouble; | 
| using printing::ConvertUnit; | 
| using printing::ConvertUnitDouble; | 
| using WebKit::WebConsoleMessage; | 
| @@ -88,12 +99,210 @@ bool PrintMsg_Print_Params_IsEqual( | 
| oldParams.params.supports_alpha_blend == | 
| newParams.params.supports_alpha_blend && | 
| oldParams.pages.size() == newParams.pages.size() && | 
| + oldParams.params.display_header_footer == | 
| + newParams.params.display_header_footer && | 
| std::equal(oldParams.pages.begin(), oldParams.pages.end(), | 
| newParams.pages.begin()); | 
| } | 
| +// Splits the horizontal width equally into segments with an interstice | 
| +// between each segment. Returns the width of this segment. | 
| +SkScalar GetSegmentWidth(const PageSizeMargins& page_layout) { | 
| + return (page_layout.margin_left + | 
| + page_layout.content_width + | 
| 
 
vandebo (ex-Chrome)
2011/07/26 06:11:37
nit: indent
 
Aayush Kumar
2011/07/26 07:37:33
Done.
 
 | 
| + page_layout.margin_right - | 
| + // Interstice is left at both ends of the page as well as between each | 
| + // region, so 1 is added. | 
| + (printing::kSettingHeaderFooterHorizontalRegions + 1) * | 
| + printing::kSettingHeaderFooterInterstice) / | 
| + printing::kSettingHeaderFooterHorizontalRegions; | 
| +} | 
| + | 
| +// Given a text, the positions, and the paint object, this method gets the | 
| +// coordinates and prints the text at those coordinates on the canvas. | 
| +void PrintHeaderFooterText( | 
| + string16 text, | 
| + SkPaint paint, | 
| + skia::VectorCanvas& canvas, | 
| + printing::HorizontalHeaderFooterPosition horizontal_position, | 
| + printing::VerticalHeaderFooterPosition vertical_position, | 
| + float webkit_scale_factor, | 
| + const PageSizeMargins& page_layout, | 
| + SkScalar max_text_width, | 
| + gfx::Font font, | 
| + SkScalar height_to_baseline) { | 
| + text = ui::ElideText(text, font, ConvertPointsToPixelDouble(max_text_width), | 
| + false); | 
| + | 
| + SkPoint text_location; | 
| + size_t text_byte_length = text.length() * sizeof(char16); | 
| + // Get the (x, y) coordinate from where printing of the current text should | 
| + // start depending on the horizontal alignment (LEFT, RIGHT, CENTER) and | 
| + // vertical alignment (TOP, BOTTOM). | 
| + { | 
| 
 
vandebo (ex-Chrome)
2011/07/26 06:11:37
Don't use an extra scope.
 
Aayush Kumar
2011/07/26 07:37:33
Done.
 
 | 
| + SkScalar text_width_in_points = paint.measureText(text.c_str(), | 
| + text_byte_length); | 
| + | 
| + SkScalar x = 0; | 
| + switch (horizontal_position) { | 
| + case printing::LEFT: { | 
| + x = -page_layout.margin_left + | 
| + printing::kSettingHeaderFooterInterstice; | 
| + break; | 
| + } | 
| + case printing::RIGHT: { | 
| + x = (page_layout.content_width + page_layout.margin_right - | 
| + (printing::kSettingHeaderFooterInterstice + text_width_in_points)); | 
| + break; | 
| + } | 
| + case printing::CENTER: { | 
| + SkScalar available_width = GetSegmentWidth(page_layout); | 
| + x = available_width - page_layout.margin_left + | 
| + (available_width - text_width_in_points) / 2; | 
| + break; | 
| + } | 
| + default: { | 
| + NOTREACHED(); | 
| + } | 
| + } | 
| + | 
| + SkScalar y = 0; | 
| + switch (vertical_position) { | 
| + case printing::TOP: { | 
| + y = -page_layout.margin_top - height_to_baseline + | 
| + printing::kSettingHeaderFooterInterstice; | 
| + break; | 
| + } | 
| + case printing::BOTTOM: { | 
| + y = page_layout.margin_bottom + page_layout.content_height - | 
| + printing::kSettingHeaderFooterInterstice - height_to_baseline; | 
| + break; | 
| + } | 
| + default: { | 
| + NOTREACHED(); | 
| + } | 
| + } | 
| + | 
| + text_location.set(x / webkit_scale_factor, y / webkit_scale_factor); | 
| + } | 
| + | 
| + paint.setTextSize(paint.getTextSize() / webkit_scale_factor); | 
| + canvas.drawText(text.c_str(), text_byte_length, text_location.x(), | 
| + text_location.y(), paint); | 
| +} | 
| + | 
| } // namespace | 
| +void PrintHeaderAndFooter(skia::VectorPlatformDeviceSkia* device, | 
| + skia::VectorCanvas& canvas, | 
| + int page_number, int total_pages, | 
| + float webkit_scale_factor, | 
| + const PageSizeMargins& page_layout, | 
| + const DictionaryValue* header_footer_info) { | 
| + // Set the drawing area to draw in the margins. | 
| + device->setDrawingArea(SkPDFDevice::kMargin_DrawingArea); | 
| + | 
| + // Setting up styles for the headers and footers text. | 
| + SkPaint paint; | 
| + paint.setColor(SK_ColorBLACK); | 
| + paint.setTextEncoding(SkPaint::kUTF16_TextEncoding); | 
| + paint.setTextSize(printing::kSettingHeaderFooterFontSize); | 
| + gfx::Font font(UTF8ToUTF16(printing::kSettingHeaderFooterFontName), | 
| + ceil(ConvertPointsToPixelDouble( | 
| + printing::kSettingHeaderFooterFontSize))); | 
| + paint.setTypeface(SkTypeface::CreateFromName( | 
| + UTF16ToUTF8(font.GetFontName()).c_str(), SkTypeface::kNormal)); | 
| + | 
| + // Ensure we have enough space to print above and below the page to print | 
| + // headers. | 
| + string16 date = base::TimeFormatShortDateNumeric(Time::Now()); | 
| + string16 title; | 
| + if (!header_footer_info->GetString(printing::kSettingHeaderFooterTitle, | 
| + &title)) { | 
| + NOTREACHED(); | 
| + } | 
| + | 
| + SkRect header_bounds; | 
| + string16 header_text = date + title; | 
| + paint.measureText(header_text.c_str(), header_text.length() * sizeof(char16), | 
| + &header_bounds, 0); | 
| + SkScalar text_height = printing::kSettingHeaderFooterInterstice + | 
| 
 
vandebo (ex-Chrome)
2011/07/26 06:11:37
nit: = <newline>
 
Aayush Kumar
2011/07/26 07:37:33
Done.
 
 | 
| + header_bounds.height(); | 
| + | 
| + if (text_height > page_layout.margin_top) | 
| + return; | 
| + | 
| + // Ensure we have enough space to print above and below the page to print | 
| + // footers. | 
| + string16 page_on_page_total = base::IntToString16(page_number) + | 
| 
 
vandebo (ex-Chrome)
2011/07/26 06:11:37
page_of_total_pages ?
 
Aayush Kumar
2011/07/26 07:37:33
Done.
 
 | 
| + UTF8ToUTF16("/") + | 
| + base::IntToString16(total_pages); | 
| + std::string url; | 
| + if (!header_footer_info->GetString(printing::kSettingHeaderFooterURL, | 
| + &url)) { | 
| + NOTREACHED(); | 
| + } | 
| + | 
| + SkRect footer_bounds; | 
| + string16 footer_text = page_on_page_total + UTF8ToUTF16(url); | 
| + paint.measureText(footer_text.c_str(), footer_text.length() * sizeof(char16), | 
| + &footer_bounds, 0); | 
| + text_height = printing::kSettingHeaderFooterInterstice + | 
| 
 
vandebo (ex-Chrome)
2011/07/26 06:11:37
nit: = <newline>
 
Aayush Kumar
2011/07/26 07:37:33
Done.
 
 | 
| + footer_bounds.height(); | 
| + | 
| + if (text_height > page_layout.margin_bottom) | 
| + return; | 
| + | 
| + SkScalar segment_width = GetSegmentWidth(page_layout); | 
| + // Print the Date. | 
| + PrintHeaderFooterText(date, paint, canvas, printing::LEFT, printing::TOP, | 
| + webkit_scale_factor, page_layout, segment_width, | 
| + font, header_bounds.top()); | 
| + | 
| + // Print the title. | 
| + SkScalar date_width = paint.measureText(date.c_str(), | 
| + date.length() * sizeof(char16)); | 
| + // Calculate the available title width. If the date string is not long | 
| + // enough, then, we increase the available space we have for the title. | 
| + // Assumes there is no header text to RIGHT of title. | 
| + SkScalar max_title_width = std::min(2 * segment_width, | 
| + 2 * (segment_width - date_width) + | 
| + segment_width); | 
| + PrintHeaderFooterText(title, paint, canvas, printing::CENTER, | 
| + printing::TOP, webkit_scale_factor, page_layout, | 
| + max_title_width, font, header_bounds.top()); | 
| + | 
| + // Print the page numbers at the bottom right corner of page. | 
| + PrintHeaderFooterText(page_on_page_total, paint, canvas, printing::RIGHT, | 
| + printing::BOTTOM, webkit_scale_factor, | 
| + page_layout, segment_width, font, | 
| + footer_bounds.bottom()); | 
| + | 
| + // Print the URL. | 
| + GURL gurl(url); | 
| + SkScalar page_width = paint.measureText( | 
| + page_on_page_total.c_str(), | 
| + page_on_page_total.length() * sizeof(char16)); | 
| + | 
| + // Calculate the available URL width. We increase the available URL width | 
| + // if the |page_on_page_total| string isn't long enough. | 
| + // Assumes no footer text being printed in the CENTER. | 
| + SkScalar max_url_width = 3 * (page_layout.content_width + | 
| 
 
vandebo (ex-Chrome)
2011/07/26 06:11:37
nit: newline
foo = 
    (foo + bar +
     baz) * 3
 
Aayush Kumar
2011/07/26 07:37:33
Done.
 
 | 
| + page_layout.margin_left + page_layout.margin_right) / 4; | 
| + max_url_width = std::min(max_url_width, | 
| + 2 * (segment_width - page_width) + segment_width); | 
| + string16 url_elided = ui::ElideUrl(gurl, font, | 
| + ConvertPointsToPixelDouble(max_url_width), | 
| + std::string()); | 
| + PrintHeaderFooterText(url_elided, paint, canvas, printing::LEFT, | 
| + printing::BOTTOM, webkit_scale_factor, | 
| + page_layout, max_url_width, font, | 
| + footer_bounds.bottom()); | 
| + | 
| + // Restore the drawing area to draw in the content area. | 
| + device->setDrawingArea(SkPDFDevice::kContent_DrawingArea); | 
| +} | 
| + | 
| PrepareFrameAndViewForPrint::PrepareFrameAndViewForPrint( | 
| const PrintMsg_Print_Params& print_params, | 
| WebFrame* frame, | 
| @@ -156,6 +365,7 @@ PrintWebViewHelper::PrintWebViewHelper(RenderView* render_view) | 
| user_cancelled_scripted_print_count_(0), | 
| notify_browser_of_print_failure_(true) { | 
| is_preview_ = switches::IsPrintPreviewEnabled(); | 
| + header_footer_info_ = new DictionaryValue(); | 
| 
 
vandebo (ex-Chrome)
2011/07/26 06:11:37
put in init list
 
Aayush Kumar
2011/07/26 07:37:33
Done.
 
 | 
| } | 
| PrintWebViewHelper::~PrintWebViewHelper() {} | 
| @@ -734,6 +944,7 @@ bool PrintWebViewHelper::UpdatePrintSettingsCloud( | 
| settings.params.desired_dpi = 72; | 
| settings.params.selection_only = false; | 
| settings.params.supports_alpha_blend = false; | 
| + settings.params.display_header_footer = false; | 
| if (!UpdatePrintSettingsRequestId(job_settings, &(settings.params))) | 
| return false; | 
| // TODO(abodenha@chromium.org) Parse page ranges from the job_settings. | 
| @@ -755,6 +966,22 @@ bool PrintWebViewHelper::UpdatePrintSettingsLocal( | 
| return false; | 
| print_pages_params_.reset(new PrintMsg_PrintPages_Params(settings)); | 
| + // Getting Header and Footer settings. | 
| 
 
vandebo (ex-Chrome)
2011/07/26 06:11:37
Should this go into UpdatePrintSettings?
 
Aayush Kumar
2011/07/26 07:37:33
I just rebased and it seems that UpdatePrintSettin
 
 | 
| + bool display_header_footer; | 
| + if (!job_settings.GetBoolean(printing::kSettingHeaderFooter, | 
| + &display_header_footer)) { | 
| + NOTREACHED(); | 
| + } | 
| + print_pages_params_->params.display_header_footer = display_header_footer; | 
| + std::string url; | 
| + if (!job_settings.GetString(printing::kSettingHeaderFooterURL, &url)) | 
| + NOTREACHED(); | 
| + header_footer_info_->SetString(printing::kSettingHeaderFooterURL, url); | 
| + string16 title; | 
| + if (!job_settings.GetString(printing::kSettingHeaderFooterTitle, &title)) | 
| + NOTREACHED(); | 
| + header_footer_info_->SetString(printing::kSettingHeaderFooterTitle, title); | 
| + | 
| Send(new PrintHostMsg_DidGetDocumentCookie(routing_id(), | 
| settings.params.document_cookie)); | 
| return true; |