diff options
Diffstat (limited to 'chrome/renderer/print_web_view_helper_mac.mm')
-rw-r--r-- | chrome/renderer/print_web_view_helper_mac.mm | 238 |
1 files changed, 238 insertions, 0 deletions
diff --git a/chrome/renderer/print_web_view_helper_mac.mm b/chrome/renderer/print_web_view_helper_mac.mm new file mode 100644 index 0000000..4ac3a47 --- /dev/null +++ b/chrome/renderer/print_web_view_helper_mac.mm @@ -0,0 +1,238 @@ +// Copyright (c) 2009 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 "app/l10n_util.h" +#include "base/logging.h" +#include "base/process_util.h" +#include "base/scoped_cftyperef.h" +#include "chrome/common/render_messages.h" +#include "chrome/renderer/render_view.h" +#include "grit/generated_resources.h" +#include "printing/native_metafile.h" +#include "webkit/api/public/WebFrame.h" +#include "webkit/api/public/WebCanvas.h" +#include "webkit/api/public/WebConsoleMessage.h" + +using WebKit::WebFrame; +using WebKit::WebCanvas; +using WebKit::WebConsoleMessage; +using WebKit::WebString; + +// TODO(stuartmorgan): There's a fair amount of code here that is duplicated +// from _win that should instead be shared. Once Linux has a real print settings +// implementation, it's likely that this whole method can just be moved to the +// cross-platform file, and the slight divergences resolved/ifdef'd. +void PrintWebViewHelper::Print(WebFrame* frame, bool script_initiated) { + const int kMinSecondsToIgnoreJavascriptInitiatedPrint = 2; + const int kMaxSecondsToIgnoreJavascriptInitiatedPrint = 2 * 60; // 2 Minutes. + + // If still not finished with earlier print request simply ignore. + if (IsPrinting()) + return; + + // TODO(maruel): Move this out of platform specific code. + // Check if there is script repeatedly trying to print and ignore it if too + // frequent. We use exponential wait time so for a page that calls print() in + // a loop the user will need to cancel the print dialog after 2 seconds, 4 + // seconds, 8, ... up to the maximum of 2 minutes. + // This gives the user time to navigate from the page. + if (script_initiated && (user_cancelled_scripted_print_count_ > 0)) { + base::TimeDelta diff = base::Time::Now() - last_cancelled_script_print_; + int min_wait_seconds = std::min( + kMinSecondsToIgnoreJavascriptInitiatedPrint << + (user_cancelled_scripted_print_count_ - 1), + kMaxSecondsToIgnoreJavascriptInitiatedPrint); + if (diff.InSeconds() < min_wait_seconds) { + WebString message(WebString::fromUTF8( + "Ignoring too frequent calls to print().")); + frame->addMessageToConsole(WebConsoleMessage( + WebConsoleMessage::LevelWarning, + message)); + return; + } + } + + // Retrieve the default print settings to calculate the expected number of + // pages. + ViewMsg_Print_Params default_settings; + bool user_cancelled_print = false; + + IPC::SyncMessage* msg = + new ViewHostMsg_GetDefaultPrintSettings(routing_id(), &default_settings); + if (Send(msg)) { + msg = NULL; + if (default_settings.IsEmpty()) { + NOTREACHED() << "Couldn't get default print settings"; + return; + } + + // Continue only if the settings are valid. + if (default_settings.dpi && default_settings.document_cookie) { + int expected_pages_count = 0; + + // Prepare once to calculate the estimated page count. This must be in + // a scope for itself (see comments on PrepareFrameAndViewForPrint). + { + PrepareFrameAndViewForPrint prep_frame_view(default_settings, + frame, + frame->view()); + expected_pages_count = prep_frame_view.GetExpectedPageCount(); + DCHECK(expected_pages_count); + } + + // Ask the browser to show UI to retrieve the final print settings. + ViewMsg_PrintPages_Params print_settings; + + ViewHostMsg_ScriptedPrint_Params params; + + // The routing id is sent across as it is needed to look up the + // corresponding RenderViewHost instance to signal and reset the + // pump messages event. + params.routing_id = routing_id(); + // host_window_ may be NULL at this point if the current window is a popup + // and the print() command has been issued from the parent. The receiver + // of this message has to deal with this. + params.host_window_id = render_view_->host_window(); + params.cookie = default_settings.document_cookie; + params.has_selection = frame->hasSelection(); + params.expected_pages_count = expected_pages_count; + + msg = new ViewHostMsg_ScriptedPrint(params, &print_settings); + // TODO(stuartmorgan): This should use SendAndRunNestedMessageLoop, as on + // Windows, to prevent getting a "hung renderer" dialog, but if we do we + // never actually break out of the nested loop and continue with printing. + // Once that's fixed, switch back to SendAndRunNestedMessageLoop. + //if (render_view_->SendAndRunNestedMessageLoop(msg)) { + if (render_view_->Send(msg)) { + msg = NULL; + + // If the settings are invalid, early quit. + if (print_settings.params.dpi && + print_settings.params.document_cookie) { + if (print_settings.params.selection_only) { + CopyAndPrint(print_settings, frame); + } else { + // TODO: Always copy before printing. + PrintPages(print_settings, frame); + } + + // Reset cancel counter on first successful print. + user_cancelled_scripted_print_count_ = 0; + return; // All went well. + } else { + user_cancelled_print = true; + } + } else { + // Send() failed. + NOTREACHED(); + } + } else { + // Failed to get default settings. + NOTREACHED(); + } + } else { + // Send() failed. + NOTREACHED(); + } + if (script_initiated && user_cancelled_print) { + ++user_cancelled_scripted_print_count_; + last_cancelled_script_print_ = base::Time::Now(); + } + // When |user_cancelled_print| is true, we treat it as success so that + // DidFinishPrinting() won't show any error alert. + // If |user_cancelled_print| is false and we reach here, there must be + // something wrong and hence is not success, DidFinishPrinting() should show + // an error alert. + // In both cases, we have to call DidFinishPrinting() here to release + // printing resources, since we don't need them anymore. + DidFinishPrinting(user_cancelled_print); +} + +void PrintWebViewHelper::PrintPages(const ViewMsg_PrintPages_Params& params, + WebKit::WebFrame* frame) { + PrepareFrameAndViewForPrint prep_frame_view(params.params, + frame, + frame->view()); + int page_count = prep_frame_view.GetExpectedPageCount(); + + Send(new ViewHostMsg_DidGetPrintedPagesCount(routing_id(), + params.params.document_cookie, + page_count)); + if (!page_count) + return; + + const gfx::Size& canvas_size = prep_frame_view.GetPrintCanvasSize(); + ViewMsg_PrintPage_Params page_params; + page_params.params = params.params; + if (params.pages.empty()) { + for (int i = 0; i < page_count; ++i) { + page_params.page_number = i; + PrintPage(page_params, canvas_size, frame); + } + } else { + for (size_t i = 0; i < params.pages.size(); ++i) { + if (params.pages[i] >= page_count) + break; + page_params.page_number = params.pages[i]; + PrintPage(page_params, canvas_size, frame); + } + } +} + +void PrintWebViewHelper::PrintPage(const ViewMsg_PrintPage_Params& params, + const gfx::Size& canvas_size, + WebFrame* frame) { + printing::NativeMetafile metafile; + CGContextRef context = metafile.Init(); + + float scale_factor = frame->getPrintPageShrink(params.page_number); + metafile.StartPage(canvas_size.width(), canvas_size.height(), scale_factor); + + // printPage can create autoreleased references to |canvas|. PDF contexts + // don't write all their data until they are destroyed, so we need to make + // certain that there are no lingering references. + NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init]; + frame->printPage(params.page_number, context); + [pool release]; + + metafile.FinishPage(); + metafile.Close(); + + // Get the size of the compiled metafile. + ViewHostMsg_DidPrintPage_Params page_params; + page_params.data_size = 0; + page_params.page_number = params.page_number; + page_params.document_cookie = params.params.document_cookie; + page_params.actual_shrink = scale_factor; + base::SharedMemory shared_buf; + + // Ask the browser to create the shared memory for us. + unsigned int buf_size = metafile.GetDataSize(); + base::SharedMemoryHandle shared_mem_handle; + if (Send(new ViewHostMsg_AllocatePDFTransport(routing_id(), buf_size, + &shared_mem_handle))) { + if (base::SharedMemory::IsHandleValid(shared_mem_handle)) { + base::SharedMemory shared_buf(shared_mem_handle, false); + if (shared_buf.Map(buf_size)) { + metafile.GetData(shared_buf.memory(), buf_size); + page_params.data_size = buf_size; + shared_buf.GiveToProcess(base::GetCurrentProcessHandle(), + &(page_params.metafile_data_handle)); + } else { + NOTREACHED() << "Map failed"; + } + } else { + NOTREACHED() << "Browser failed to allocate shared memory"; + } + } else { + NOTREACHED() << "Browser allocation request message failed"; + } + + Send(new ViewHostMsg_DidPrintPage(routing_id(), page_params)); +} + |