diff --git a/atom/browser/api/atom_api_web_contents.cc b/atom/browser/api/atom_api_web_contents.cc index ba648e5ad32..d47a4da2fec 100644 --- a/atom/browser/api/atom_api_web_contents.cc +++ b/atom/browser/api/atom_api_web_contents.cc @@ -44,6 +44,7 @@ #include "base/process/process_handle.h" #include "base/strings/utf_string_conversions.h" #include "base/threading/thread_task_runner_handle.h" +#include "base/values.h" #include "brightray/browser/inspectable_web_contents.h" #include "brightray/browser/inspectable_web_contents_view.h" #include "chrome/browser/printing/print_preview_message_handler.h" @@ -74,6 +75,7 @@ #include "content/public/browser/storage_partition.h" #include "content/public/browser/web_contents.h" #include "content/public/common/context_menu_params.h" +#include "native_mate/converter.h" #include "native_mate/dictionary.h" #include "native_mate/object_template_builder.h" #include "net/url_request/url_request_context.h" @@ -93,6 +95,7 @@ namespace { struct PrintSettings { bool silent; bool print_background; + base::string16 device_name; }; } // namespace @@ -130,10 +133,25 @@ struct Converter { return false; dict.Get("silent", &(out->silent)); dict.Get("printBackground", &(out->print_background)); + dict.Get("deviceName", &(out->device_name)); return true; } }; +template<> +struct Converter { + static v8::Local ToV8(v8::Isolate* isolate, + const printing::PrinterBasicInfo& val) { + mate::Dictionary dict(isolate, v8::Object::New(isolate)); + dict.Set("name", val.printer_name); + dict.Set("description", val.printer_description); + dict.Set("status", val.printer_status); + dict.Set("isDefault", val.is_default ? true : false); + dict.Set("options", val.options); + return dict.GetHandle(); + } +}; + template<> struct Converter { static v8::Local ToV8(v8::Isolate* isolate, @@ -1291,7 +1309,7 @@ bool WebContents::IsAudioMuted() { } void WebContents::Print(mate::Arguments* args) { - PrintSettings settings = { false, false }; + PrintSettings settings = { false, false, base::string16() }; if (args->Length() == 1 && !args->GetNext(&settings)) { args->ThrowError(); return; @@ -1300,7 +1318,15 @@ void WebContents::Print(mate::Arguments* args) { printing::PrintViewManagerBasic::FromWebContents(web_contents())-> PrintNow(web_contents()->GetMainFrame(), settings.silent, - settings.print_background); + settings.print_background, + settings.device_name); +} + +std::vector WebContents::GetPrinterList() { + std::vector printers; + auto print_backend = printing::PrintBackend::CreateInstance(nullptr); + print_backend->EnumeratePrinters(&printers); + return printers; } void WebContents::PrintToPDF(const base::DictionaryValue& setting, @@ -1844,6 +1870,7 @@ void WebContents::BuildPrototype(v8::Isolate* isolate, &WebContents::UnregisterServiceWorker) .SetMethod("inspectServiceWorker", &WebContents::InspectServiceWorker) .SetMethod("print", &WebContents::Print) + .SetMethod("getPrinters", &WebContents::GetPrinterList) .SetMethod("_printToPDF", &WebContents::PrintToPDF) .SetMethod("addWorkSpace", &WebContents::AddWorkSpace) .SetMethod("removeWorkSpace", &WebContents::RemoveWorkSpace) diff --git a/atom/browser/api/atom_api_web_contents.h b/atom/browser/api/atom_api_web_contents.h index ce35da99476..6a04d6f518c 100644 --- a/atom/browser/api/atom_api_web_contents.h +++ b/atom/browser/api/atom_api_web_contents.h @@ -18,6 +18,7 @@ #include "content/public/browser/web_contents_observer.h" #include "content/public/common/favicon_url.h" #include "native_mate/handle.h" +#include "printing/backend/print_backend.h" #include "ui/gfx/image/image.h" namespace blink { @@ -121,6 +122,7 @@ class WebContents : public mate::TrackableObject, void SetAudioMuted(bool muted); bool IsAudioMuted(); void Print(mate::Arguments* args); + std::vector GetPrinterList(); void SetEmbedder(const WebContents* embedder); // Print current page as PDF. diff --git a/chromium_src/chrome/browser/printing/print_job_worker.cc b/chromium_src/chrome/browser/printing/print_job_worker.cc index 0c56c200d08..7965e9d5e59 100644 --- a/chromium_src/chrome/browser/printing/print_job_worker.cc +++ b/chromium_src/chrome/browser/printing/print_job_worker.cc @@ -30,6 +30,20 @@ #include "printing/printing_utils.h" #include "ui/base/l10n/l10n_util.h" +#include +#include +#include +#include +#include + +#include "base/strings/string_number_conversions.h" +#include "base/strings/utf_string_conversions.h" +#include "base/time/time.h" +#include "printing/page_size_margins.h" +#include "printing/print_job_constants.h" +#include "printing/print_settings.h" +#include "printing/units.h" + using content::BrowserThread; namespace printing { @@ -42,6 +56,81 @@ void HoldRefCallback(const scoped_refptr& owner, callback.Run(); } +void SetCustomMarginsToJobSettings(const PageSizeMargins& page_size_margins, + base::DictionaryValue* settings) { + std::unique_ptr custom_margins(new base::DictionaryValue()); + custom_margins->SetDouble(kSettingMarginTop, page_size_margins.margin_top); + custom_margins->SetDouble(kSettingMarginBottom, page_size_margins.margin_bottom); + custom_margins->SetDouble(kSettingMarginLeft, page_size_margins.margin_left); + custom_margins->SetDouble(kSettingMarginRight, page_size_margins.margin_right); + settings->Set(kSettingMarginsCustom, std::move(custom_margins)); +} + +void PrintSettingsToJobSettings(const PrintSettings& settings, + base::DictionaryValue* job_settings) { + // header footer + job_settings->SetBoolean(kSettingHeaderFooterEnabled, + settings.display_header_footer()); + job_settings->SetString(kSettingHeaderFooterTitle, settings.title()); + job_settings->SetString(kSettingHeaderFooterURL, settings.url()); + + // bg + job_settings->SetBoolean(kSettingShouldPrintBackgrounds, + settings.should_print_backgrounds()); + job_settings->SetBoolean(kSettingShouldPrintSelectionOnly, + settings.selection_only()); + + // margin + auto margin_type = settings.margin_type(); + job_settings->SetInteger(kSettingMarginsType, settings.margin_type()); + if (margin_type == CUSTOM_MARGINS) { + const auto& margins_in_points = settings.requested_custom_margins_in_points(); + + PageSizeMargins page_size_margins; + + page_size_margins.margin_top = margins_in_points.top; + page_size_margins.margin_bottom = margins_in_points.bottom; + page_size_margins.margin_left = margins_in_points.left; + page_size_margins.margin_right = margins_in_points.right; + SetCustomMarginsToJobSettings(page_size_margins, job_settings); + } + job_settings->SetInteger(kSettingPreviewPageCount, 1); + + // range + + if (!settings.ranges().empty()) { + base::ListValue* page_range_array = new base::ListValue; + job_settings->Set(kSettingPageRange, page_range_array); + for (size_t i = 0; i < settings.ranges().size(); ++i) { + std::unique_ptr dict(new base::DictionaryValue); + dict->SetInteger(kSettingPageRangeFrom, settings.ranges()[i].from + 1); + dict->SetInteger(kSettingPageRangeTo, settings.ranges()[i].to + 1); + page_range_array->Append(std::move(dict)); + } + } + + job_settings->SetBoolean(kSettingCollate, settings.collate()); + job_settings->SetInteger(kSettingCopies, 1); + job_settings->SetInteger(kSettingColor, settings.color()); + job_settings->SetInteger(kSettingDuplexMode, settings.duplex_mode()); + job_settings->SetBoolean(kSettingLandscape, settings.landscape()); + job_settings->SetString(kSettingDeviceName, settings.device_name()); + job_settings->SetInteger(kSettingScaleFactor, 100); + job_settings->SetBoolean("rasterizePDF", false); + + job_settings->SetInteger("desiredDpi", settings.desired_dpi()); + job_settings->SetInteger("dpi", settings.dpi()); + + job_settings->SetBoolean(kSettingPrintToPDF, false); + job_settings->SetBoolean(kSettingCloudPrintDialog, false); + job_settings->SetBoolean(kSettingPrintWithPrivet, false); + job_settings->SetBoolean(kSettingPrintWithExtension, false); + + job_settings->SetBoolean(kSettingShowSystemDialog, false); + job_settings->SetInteger(kSettingPreviewPageCount, 1); +} + + class PrintingContextDelegate : public PrintingContext::Delegate { public: PrintingContextDelegate(int render_process_id, int render_frame_id); @@ -133,7 +222,8 @@ void PrintJobWorker::GetSettings(bool ask_user_for_settings, bool has_selection, MarginType margin_type, bool is_scripted, - bool is_modifiable) { + bool is_modifiable, + const base::string16& device_name) { DCHECK(task_runner_->RunsTasksOnCurrentThread()); DCHECK_EQ(page_number_, PageNumber::npos()); @@ -157,6 +247,13 @@ void PrintJobWorker::GetSettings(bool ask_user_for_settings, document_page_count, has_selection, is_scripted))); + } else if (!device_name.empty()) { + BrowserThread::PostTask( + BrowserThread::UI, FROM_HERE, + base::Bind(&HoldRefCallback, make_scoped_refptr(owner_), + base::Bind(&PrintJobWorker::InitWithDeviceName, + base::Unretained(this), + device_name))); } else { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, @@ -225,6 +322,14 @@ void PrintJobWorker::UseDefaultSettings() { GetSettingsDone(result); } +void PrintJobWorker::InitWithDeviceName(const base::string16& device_name) { + const auto& settings = printing_context_->settings(); + std::unique_ptr dic(new base::DictionaryValue); + PrintSettingsToJobSettings(settings, dic.get()); + dic->SetString(kSettingDeviceName, device_name); + UpdatePrintSettings(std::move(dic)); +} + void PrintJobWorker::StartPrinting(PrintedDocument* new_document) { DCHECK(task_runner_->RunsTasksOnCurrentThread()); DCHECK_EQ(page_number_, PageNumber::npos()); diff --git a/chromium_src/chrome/browser/printing/print_job_worker.h b/chromium_src/chrome/browser/printing/print_job_worker.h index e001a86b840..7d146bc34d6 100644 --- a/chromium_src/chrome/browser/printing/print_job_worker.h +++ b/chromium_src/chrome/browser/printing/print_job_worker.h @@ -51,7 +51,8 @@ class PrintJobWorker { bool has_selection, MarginType margin_type, bool is_scripted, - bool is_modifiable); + bool is_modifiable, + const base::string16& device_name); // Set the new print settings. void SetSettings(std::unique_ptr new_settings); @@ -128,6 +129,9 @@ class PrintJobWorker { // systems. void UseDefaultSettings(); + // set the printer name + void InitWithDeviceName(const base::string16& device_name); + // Printing context delegate. std::unique_ptr printing_context_delegate_; diff --git a/chromium_src/chrome/browser/printing/print_view_manager_base.cc b/chromium_src/chrome/browser/printing/print_view_manager_base.cc index 07ec2992f33..bdaa3ab0b0e 100644 --- a/chromium_src/chrome/browser/printing/print_view_manager_base.cc +++ b/chromium_src/chrome/browser/printing/print_view_manager_base.cc @@ -67,11 +67,12 @@ PrintViewManagerBase::~PrintViewManagerBase() { #if !defined(DISABLE_BASIC_PRINTING) bool PrintViewManagerBase::PrintNow(content::RenderFrameHost* rfh, - bool silent, bool print_background) { + bool silent, bool print_background, + const base::string16& device_name) { int32_t id = rfh->GetRoutingID(); return PrintNowInternal( rfh, - base::MakeUnique(id, silent, print_background)); + base::MakeUnique(id, silent, print_background, device_name)); } #endif // !DISABLE_BASIC_PRINTING diff --git a/chromium_src/chrome/browser/printing/print_view_manager_base.h b/chromium_src/chrome/browser/printing/print_view_manager_base.h index 46765d09100..61f42382bd4 100644 --- a/chromium_src/chrome/browser/printing/print_view_manager_base.h +++ b/chromium_src/chrome/browser/printing/print_view_manager_base.h @@ -40,7 +40,8 @@ class PrintViewManagerBase : public content::NotificationObserver, // asynchronous, the actual printing will not be completed on the return of // this function. Returns false if printing is impossible at the moment. virtual bool PrintNow(content::RenderFrameHost* rfh, - bool silent, bool print_background); + bool silent, bool print_background, + const base::string16& device_name); #endif // !DISABLE_BASIC_PRINTING // PrintedPagesSource implementation. diff --git a/chromium_src/chrome/browser/printing/printer_query.cc b/chromium_src/chrome/browser/printing/printer_query.cc index bfaeb574db3..acfbec320b3 100644 --- a/chromium_src/chrome/browser/printing/printer_query.cc +++ b/chromium_src/chrome/browser/printing/printer_query.cc @@ -86,7 +86,27 @@ void PrinterQuery::GetSettings(GetSettingsAskParam ask_user_for_settings, FROM_HERE, base::Bind(&PrintJobWorker::GetSettings, base::Unretained(worker_.get()), is_print_dialog_box_shown_, expected_page_count, has_selection, - margin_type, is_scripted, is_modifiable)); + margin_type, is_scripted, is_modifiable, base::string16())); +} + +void PrinterQuery::GetSettings( + GetSettingsAskParam ask_user_for_settings, + int expected_page_count, + bool has_selection, + MarginType margin_type, + bool is_scripted, + bool is_modifiable, + const base::string16& device_name, + const base::Closure& callback) { + DCHECK(RunsTasksOnCurrentThread()); + DCHECK(!is_print_dialog_box_shown_); + StartWorker(callback); + + is_print_dialog_box_shown_ = false; + worker_->PostTask(FROM_HERE, + base::Bind(&PrintJobWorker::GetSettings, base::Unretained(worker_.get()), + is_print_dialog_box_shown_, expected_page_count, has_selection, + margin_type, is_scripted, is_modifiable, device_name)); } void PrinterQuery::SetSettings( diff --git a/chromium_src/chrome/browser/printing/printer_query.h b/chromium_src/chrome/browser/printing/printer_query.h index 9008e19de81..f150f1b278a 100644 --- a/chromium_src/chrome/browser/printing/printer_query.h +++ b/chromium_src/chrome/browser/printing/printer_query.h @@ -53,7 +53,17 @@ class PrinterQuery : public PrintJobWorkerOwner { bool is_modifiable, const base::Closure& callback); - // Updates the current settings with |new_settings| dictionary values. + void GetSettings( + GetSettingsAskParam ask_user_for_settings, + int expected_page_count, + bool has_selection, + MarginType margin_type, + bool is_scripted, + bool is_modifiable, + const base::string16& device_name, + const base::Closure& callback); + + // Updates the current settings with |new_settings| dictionary values. void SetSettings(std::unique_ptr new_settings, const base::Closure& callback); diff --git a/chromium_src/chrome/browser/printing/printing_message_filter.cc b/chromium_src/chrome/browser/printing/printing_message_filter.cc index b0d3824fc87..1c9e9daadef 100644 --- a/chromium_src/chrome/browser/printing/printing_message_filter.cc +++ b/chromium_src/chrome/browser/printing/printing_message_filter.cc @@ -116,6 +116,8 @@ bool PrintingMessageFilter::OnMessageReceived(const IPC::Message& message) { #endif IPC_MESSAGE_HANDLER_DELAY_REPLY(PrintHostMsg_GetDefaultPrintSettings, OnGetDefaultPrintSettings) + IPC_MESSAGE_HANDLER_DELAY_REPLY(PrintHostMsg_InitSettingWithDeviceName, + OnInitSettingWithDeviceName) IPC_MESSAGE_HANDLER_DELAY_REPLY(PrintHostMsg_ScriptedPrint, OnScriptedPrint) IPC_MESSAGE_HANDLER_DELAY_REPLY(PrintHostMsg_UpdatePrintSettings, OnUpdatePrintSettings) @@ -174,6 +176,32 @@ void PrintingMessageFilter::OnGetDefaultPrintSettings(IPC::Message* reply_msg) { printer_query, reply_msg)); } +void PrintingMessageFilter::OnInitSettingWithDeviceName(const base::string16& device_name, + IPC::Message* reply_msg) { + DCHECK_CURRENTLY_ON(BrowserThread::IO); + scoped_refptr printer_query; + printer_query = queue_->PopPrinterQuery(0); + if (!printer_query.get()) { + printer_query = + queue_->CreatePrinterQuery(render_process_id_, reply_msg->routing_id()); + } + + // Loads default settings. This is asynchronous, only the IPC message sender + // will hang until the settings are retrieved. + printer_query->GetSettings( + PrinterQuery::GetSettingsAskParam::DEFAULTS, + 0, + false, + DEFAULT_MARGINS, + true, + true, + device_name, + base::Bind(&PrintingMessageFilter::OnGetDefaultPrintSettingsReply, + this, + printer_query, + reply_msg)); +} + void PrintingMessageFilter::OnGetDefaultPrintSettingsReply( scoped_refptr printer_query, IPC::Message* reply_msg) { diff --git a/chromium_src/chrome/browser/printing/printing_message_filter.h b/chromium_src/chrome/browser/printing/printing_message_filter.h index d023af1b34f..fb7dced52e4 100644 --- a/chromium_src/chrome/browser/printing/printing_message_filter.h +++ b/chromium_src/chrome/browser/printing/printing_message_filter.h @@ -61,6 +61,11 @@ class PrintingMessageFilter : public content::BrowserMessageFilter { // Get the default print setting. void OnGetDefaultPrintSettings(IPC::Message* reply_msg); + + // Set deviceName + void OnInitSettingWithDeviceName(const base::string16& device_name, + IPC::Message* reply_msg); + void OnGetDefaultPrintSettingsReply(scoped_refptr printer_query, IPC::Message* reply_msg); diff --git a/chromium_src/chrome/common/print_messages.cc b/chromium_src/chrome/common/print_messages.cc index 8c68e432bba..f9eacfc5e6c 100644 --- a/chromium_src/chrome/common/print_messages.cc +++ b/chromium_src/chrome/common/print_messages.cc @@ -27,7 +27,9 @@ PrintMsg_Print_Params::PrintMsg_Print_Params() display_header_footer(false), title(), url(), - should_print_backgrounds(false) {} + should_print_backgrounds(false), + device_name() { +} PrintMsg_Print_Params::~PrintMsg_Print_Params() {} @@ -52,6 +54,7 @@ void PrintMsg_Print_Params::Reset() { title = base::string16(); url = base::string16(); should_print_backgrounds = false; + device_name.clear(); } PrintMsg_PrintPages_Params::PrintMsg_PrintPages_Params() diff --git a/chromium_src/chrome/common/print_messages.h b/chromium_src/chrome/common/print_messages.h index 196d8be508c..2a88b0c3b76 100644 --- a/chromium_src/chrome/common/print_messages.h +++ b/chromium_src/chrome/common/print_messages.h @@ -55,6 +55,7 @@ struct PrintMsg_Print_Params { base::string16 title; base::string16 url; bool should_print_backgrounds; + base::string16 device_name; }; struct PrintMsg_PrintPages_Params { @@ -233,9 +234,10 @@ IPC_STRUCT_END() // Tells the render view to switch the CSS to print media type, renders every // requested pages and switch back the CSS to display media type. -IPC_MESSAGE_ROUTED2(PrintMsg_PrintPages, +IPC_MESSAGE_ROUTED3(PrintMsg_PrintPages, bool /* silent print */, - bool /* print page's background */) + bool /* print page's background */, + base::string16 /* device name*/) // Tells the render view that printing is done so it can clean up. IPC_MESSAGE_ROUTED1(PrintMsg_PrintingDone, @@ -280,6 +282,11 @@ IPC_MESSAGE_ROUTED1(PrintHostMsg_DidPrintPage, IPC_SYNC_MESSAGE_ROUTED0_1(PrintHostMsg_GetDefaultPrintSettings, PrintMsg_Print_Params /* default_settings */) +// you can set the printer +IPC_SYNC_MESSAGE_ROUTED1_1(PrintHostMsg_InitSettingWithDeviceName, + base::string16, /* device name */ + PrintMsg_Print_Params /* default_settings */) + // The renderer wants to update the current print settings with new // |job_settings|. IPC_SYNC_MESSAGE_ROUTED2_2(PrintHostMsg_UpdatePrintSettings, diff --git a/chromium_src/chrome/renderer/printing/print_web_view_helper.cc b/chromium_src/chrome/renderer/printing/print_web_view_helper.cc index adac07cb7ce..4014b3fa0ed 100644 --- a/chromium_src/chrome/renderer/printing/print_web_view_helper.cc +++ b/chromium_src/chrome/renderer/printing/print_web_view_helper.cc @@ -659,12 +659,13 @@ void PrintWebViewHelper::OnDestruct() { } #if !defined(DISABLE_BASIC_PRINTING) -void PrintWebViewHelper::OnPrintPages(bool silent, bool print_background) { +void PrintWebViewHelper::OnPrintPages(bool silent, bool print_background, + const base::string16& device_name) { if (ipc_nesting_level_> 1) return; blink::WebLocalFrame* frame = render_frame()->GetWebFrame(); - Print(frame, blink::WebNode(), silent, print_background); + Print(frame, blink::WebNode(), silent, print_background, device_name); } #endif // !DISABLE_BASIC_PRINTING @@ -847,7 +848,8 @@ void PrintWebViewHelper::PrintNode(const blink::WebNode& node) { void PrintWebViewHelper::Print(blink::WebLocalFrame* frame, const blink::WebNode& node, bool silent, - bool print_background) { + bool print_background, + const base::string16& device_name) { // If still not finished with earlier print request simply ignore. if (prep_frame_view_) return; @@ -855,7 +857,7 @@ void PrintWebViewHelper::Print(blink::WebLocalFrame* frame, FrameReference frame_ref(frame); int expected_page_count = 0; - if (!CalculateNumberOfPages(frame, node, &expected_page_count)) { + if (!CalculateNumberOfPages(frame, node, &expected_page_count, device_name)) { DidFinishPrinting(FAIL_PRINT_INIT); return; // Failed to init print page settings. } @@ -991,10 +993,16 @@ void PrintWebViewHelper::ComputePageLayoutInPointsForCss( CalculatePageLayoutFromPrintParams(params, page_layout_in_points); } -bool PrintWebViewHelper::InitPrintSettings(bool fit_to_paper_size) { +bool PrintWebViewHelper::InitPrintSettings(bool fit_to_paper_size, + const base::string16& device_name) { PrintMsg_PrintPages_Params settings; - Send(new PrintHostMsg_GetDefaultPrintSettings(routing_id(), - &settings.params)); + if (device_name.empty()) { + Send(new PrintHostMsg_GetDefaultPrintSettings(routing_id(), + &settings.params)); + } else { + Send(new PrintHostMsg_InitSettingWithDeviceName(routing_id(), device_name, + &settings.params)); + } // Check if the printer returned any settings, if the settings is empty, we // can safely assume there are no printer drivers configured. So we safely // terminate. @@ -1019,10 +1027,11 @@ bool PrintWebViewHelper::InitPrintSettings(bool fit_to_paper_size) { bool PrintWebViewHelper::CalculateNumberOfPages(blink::WebLocalFrame* frame, const blink::WebNode& node, - int* number_of_pages) { + int* number_of_pages, + const base::string16& device_name) { DCHECK(frame); bool fit_to_paper_size = !(PrintingNodeOrPdfFrame(frame, node)); - if (!InitPrintSettings(fit_to_paper_size)) { + if (!InitPrintSettings(fit_to_paper_size, device_name)) { notify_browser_of_print_failure_ = false; Send(new PrintHostMsg_ShowInvalidPrinterSettingsError(routing_id())); return false; diff --git a/chromium_src/chrome/renderer/printing/print_web_view_helper.h b/chromium_src/chrome/renderer/printing/print_web_view_helper.h index 4929477a1ba..acda59b694c 100644 --- a/chromium_src/chrome/renderer/printing/print_web_view_helper.h +++ b/chromium_src/chrome/renderer/printing/print_web_view_helper.h @@ -99,7 +99,9 @@ class PrintWebViewHelper // Message handlers --------------------------------------------------------- #if !defined(DISABLE_BASIC_PRINTING) - void OnPrintPages(bool silent, bool print_background); + void OnPrintPages(bool silent, + bool print_background, + const base::string16& device_name); void OnPrintingDone(bool success); #endif // !DISABLE_BASIC_PRINTING void OnPrintPreview(const base::DictionaryValue& settings); @@ -138,7 +140,8 @@ class PrintWebViewHelper void Print(blink::WebLocalFrame* frame, const blink::WebNode& node, bool silent = false, - bool print_background = false); + bool print_background = false, + const base::string16& device_name = base::string16()); // Notification when printing is done - signal tear-down/free resources. void DidFinishPrinting(PrintingResult result); @@ -147,12 +150,14 @@ class PrintWebViewHelper // Initialize print page settings with default settings. // Used only for native printing workflow. - bool InitPrintSettings(bool fit_to_paper_size); + bool InitPrintSettings(bool fit_to_paper_size, + const base::string16& device_name = base::string16()); // Calculate number of pages in source document. bool CalculateNumberOfPages(blink::WebLocalFrame* frame, const blink::WebNode& node, - int* number_of_pages); + int* number_of_pages, + const base::string16& device_name = base::string16()); // Update the current print settings with new |passed_job_settings|. // |passed_job_settings| dictionary contains print job details such as printer diff --git a/docs/api/structures/printer-info.md b/docs/api/structures/printer-info.md new file mode 100644 index 00000000000..2d8cab49eef --- /dev/null +++ b/docs/api/structures/printer-info.md @@ -0,0 +1,44 @@ +# PrinterInfo Object + +* `name` String +* `description` String +* `status` Number +* `isDefault` Boolean +* `options` Object - Additional fields + +## Example + +Below is an example of some of the additional options that may be set which +may be different on each platform. + +```javascript +{ + name: 'Zebra_LP2844', + description: 'Zebra LP2844', + status: 3, + isDefault: false, + options: { + copies: '1', + 'device-uri': 'usb://Zebra/LP2844?location=14200000', + finishings: '3', + 'job-cancel-after': '10800', + 'job-hold-until': 'no-hold', + 'job-priority': '50', + 'job-sheets': 'none,none', + 'marker-change-time': '0', + 'number-up': '1', + 'printer-commands': 'none', + 'printer-info': 'Zebra LP2844', + 'printer-is-accepting-jobs': 'true', + 'printer-is-shared': 'true', + 'printer-location': '', + 'printer-make-and-model': 'Zebra EPL2 Label Printer', + 'printer-state': '3', + 'printer-state-change-time': '1484872644', + 'printer-state-reasons': 'offline-report', + 'printer-type': '36932', + 'printer-uri-supported': 'ipp://localhost/printers/Zebra_LP2844', + system_driverinfo: 'Z' + } +} +``` diff --git a/docs/api/web-contents.md b/docs/api/web-contents.md index a4b779de295..99d562d1b46 100644 --- a/docs/api/web-contents.md +++ b/docs/api/web-contents.md @@ -916,18 +916,26 @@ Unregisters any ServiceWorker if present and returns a boolean as response to `callback` when the JS promise is fulfilled or false when the JS promise is rejected. +#### `contents.getPrinters()` + +Get the system printer list. + +Returns [PrinterInfo[]](structures/printer-info.md) + #### `contents.print([options])` * `options` Object (optional) * `silent` Boolean - Don't ask user for print settings. Default is `false`. * `printBackground` Boolean - Also prints the background color and image of the web page. Default is `false`. + * `deviceName` String - Set the printer device name to use. Default is `''`. Prints window's web page. When `silent` is set to `true`, Electron will pick -up system's default printer and default settings for printing. +the system's default printer if `deviceName` is empty and the default settings +for printing. Calling `window.print()` in web page is equivalent to calling -`webContents.print({silent: false, printBackground: false})`. +`webContents.print({silent: false, printBackground: false, deviceName: ''})`. Use `page-break-before: always; ` CSS style to force to print to a new page. diff --git a/spec/api-browser-window-spec.js b/spec/api-browser-window-spec.js index fb37cfd2980..6c54731aa70 100644 --- a/spec/api-browser-window-spec.js +++ b/spec/api-browser-window-spec.js @@ -1114,6 +1114,23 @@ describe('BrowserWindow module', function () { }) }) + it('can get printer list', function (done) { + w.destroy() + w = new BrowserWindow({ + show: false, + webPreferences: { + sandbox: true, + preload: preload + } + }) + w.loadURL('data:text/html,%3Ch1%3EHello%2C%20World!%3C%2Fh1%3E') + w.webContents.once('did-finish-load', function () { + const printers = w.webContents.getPrinters() + assert.equal(Array.isArray(printers), true) + done() + }) + }) + it('can print to PDF', function (done) { w.destroy() w = new BrowserWindow({