1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #include "chrome/browser/printing/print_dialog_cloud.h" 6 7 8 #include "base/base64.h" 9 #include "base/bind.h" 10 #include "base/bind_helpers.h" 11 #include "base/command_line.h" 12 #include "base/files/file_util.h" 13 #include "base/json/json_reader.h" 14 #include "base/prefs/pref_service.h" 15 #include "base/strings/utf_string_conversions.h" 16 #include "base/values.h" 17 #include "chrome/browser/browser_process.h" 18 #include "chrome/browser/devtools/devtools_window.h" 19 #include "chrome/browser/lifetime/application_lifetime.h" 20 #include "chrome/browser/printing/print_dialog_cloud_internal.h" 21 #include "chrome/browser/profiles/profile.h" 22 #include "chrome/browser/ui/browser.h" 23 #include "chrome/browser/ui/browser_dialogs.h" 24 #include "chrome/browser/ui/browser_window.h" 25 #include "chrome/common/chrome_switches.h" 26 #include "chrome/common/pref_names.h" 27 #include "chrome/common/print_messages.h" 28 #include "chrome/common/url_constants.h" 29 #include "components/cloud_devices/common/cloud_devices_urls.h" 30 #include "components/google/core/browser/google_util.h" 31 #include "components/pref_registry/pref_registry_syncable.h" 32 #include "components/signin/core/common/profile_management_switches.h" 33 #include "content/public/browser/browser_thread.h" 34 #include "content/public/browser/navigation_controller.h" 35 #include "content/public/browser/navigation_entry.h" 36 #include "content/public/browser/notification_registrar.h" 37 #include "content/public/browser/notification_source.h" 38 #include "content/public/browser/notification_types.h" 39 #include "content/public/browser/render_view_host.h" 40 #include "content/public/browser/web_contents.h" 41 #include "content/public/browser/web_contents_observer.h" 42 #include "content/public/browser/web_ui.h" 43 #include "content/public/common/frame_navigate_params.h" 44 #include "content/public/common/web_preferences.h" 45 46 #if defined(USE_AURA) 47 #include "ui/aura/window.h" 48 #include "ui/aura/window_tree_host.h" 49 #endif 50 51 #if defined(OS_WIN) 52 #include "ui/base/win/foreground_helper.h" 53 #endif 54 55 // This module implements the UI support in Chrome for cloud printing. 56 // This means hosting a dialog containing HTML/JavaScript and using 57 // the published cloud print user interface integration APIs to get 58 // page setup settings from the dialog contents and provide the 59 // generated print data to the dialog contents for uploading to the 60 // cloud print service. 61 62 // Currently, the flow between these classes is as follows: 63 64 // PrintDialogCloud::CreatePrintDialogForFile is called from 65 // resource_message_filter_gtk.cc once the renderer has informed the 66 // renderer host that print data generation into the renderer host provided 67 // temp file has been completed. That call is on the FILE thread. 68 // That, in turn, hops over to the UI thread to create an instance of 69 // PrintDialogCloud. 70 71 // The constructor for PrintDialogCloud creates a 72 // CloudPrintWebDialogDelegate and asks the current active browser to 73 // show an HTML dialog using that class as the delegate. That class 74 // hands in the kChromeUICloudPrintResourcesURL as the URL to visit. That is 75 // recognized by the GetWebUIFactoryFunction as a signal to create an 76 // ExternalWebDialogUI. 77 78 // CloudPrintWebDialogDelegate also temporarily owns a 79 // CloudPrintFlowHandler, a class which is responsible for the actual 80 // interactions with the dialog contents, including handing in the 81 // print data and getting any page setup parameters that the dialog 82 // contents provides. As part of bringing up the dialog, 83 // WebDialogUI::RenderViewCreated is called (an override of 84 // WebUI::RenderViewCreated). That routine, in turn, calls the 85 // delegate's GetWebUIMessageHandlers routine, at which point the 86 // ownership of the CloudPrintFlowHandler is handed over. A pointer 87 // to the flow handler is kept to facilitate communication back and 88 // forth between the two classes. 89 90 // The WebUI continues dialog bring-up, calling 91 // CloudPrintFlowHandler::RegisterMessages. This is where the 92 // additional object model capabilities are registered for the dialog 93 // contents to use. It is also at this time that capabilities for the 94 // dialog contents are adjusted to allow the dialog contents to close 95 // the window. In addition, the pending URL is redirected to the 96 // actual cloud print service URL. The flow controller also registers 97 // for notification of when the dialog contents finish loading, which 98 // is currently used to send the data to the dialog contents. 99 100 // In order to send the data to the dialog contents, the flow 101 // handler uses a CloudPrintDataSender. It creates one, letting it 102 // know the name of the temporary file containing the data, and 103 // posts the task of reading the file 104 // (CloudPrintDataSender::ReadPrintDataFile) to the file thread. That 105 // routine reads in the file, and then hops over to the IO thread to 106 // send that data to the dialog contents. 107 108 // When the dialog contents are finished (by either being cancelled or 109 // hitting the print button), the delegate is notified, and responds 110 // that the dialog should be closed, at which point things are torn 111 // down and released. 112 113 using content::BrowserThread; 114 using content::NavigationController; 115 using content::NavigationEntry; 116 using content::RenderViewHost; 117 using content::WebContents; 118 using content::WebPreferences; 119 using content::WebUIMessageHandler; 120 using ui::WebDialogDelegate; 121 122 namespace { 123 124 const int kDefaultWidth = 912; 125 const int kDefaultHeight = 633; 126 127 bool IsSimilarUrl(const GURL& url, const GURL& cloud_print_url) { 128 return url.host() == cloud_print_url.host() && 129 StartsWithASCII(url.path(), cloud_print_url.path(), false) && 130 url.scheme() == cloud_print_url.scheme(); 131 } 132 133 class SignInObserver : public content::WebContentsObserver { 134 public: 135 SignInObserver(content::WebContents* web_contents, 136 GURL cloud_print_url, 137 const base::Closure& callback) 138 : WebContentsObserver(web_contents), 139 cloud_print_url_(cloud_print_url), 140 callback_(callback), 141 weak_ptr_factory_(this) { 142 } 143 144 private: 145 // Overridden from content::WebContentsObserver: 146 virtual void DidNavigateMainFrame( 147 const content::LoadCommittedDetails& details, 148 const content::FrameNavigateParams& params) OVERRIDE { 149 if (IsSimilarUrl(params.url, cloud_print_url_)) { 150 base::MessageLoop::current()->PostTask( 151 FROM_HERE, 152 base::Bind(&SignInObserver::OnSignIn, 153 weak_ptr_factory_.GetWeakPtr())); 154 } 155 } 156 157 virtual void WebContentsDestroyed() OVERRIDE { 158 delete this; 159 } 160 161 void OnSignIn() { 162 callback_.Run(); 163 if (web_contents()) 164 web_contents()->Close(); 165 } 166 167 GURL cloud_print_url_; 168 base::Closure callback_; 169 base::WeakPtrFactory<SignInObserver> weak_ptr_factory_; 170 171 DISALLOW_COPY_AND_ASSIGN(SignInObserver); 172 }; 173 174 } // namespace 175 176 namespace internal_cloud_print_helpers { 177 178 // From the JSON parsed value, get the entries for the page setup 179 // parameters. 180 bool GetPageSetupParameters(const std::string& json, 181 PrintMsg_Print_Params& parameters) { 182 scoped_ptr<base::Value> parsed_value(base::JSONReader::Read(json)); 183 DLOG_IF(ERROR, (!parsed_value.get() || 184 !parsed_value->IsType(base::Value::TYPE_DICTIONARY))) 185 << "PageSetup call didn't have expected contents"; 186 if (!parsed_value.get() || 187 !parsed_value->IsType(base::Value::TYPE_DICTIONARY)) { 188 return false; 189 } 190 191 bool result = true; 192 base::DictionaryValue* params = 193 static_cast<base::DictionaryValue*>(parsed_value.get()); 194 result &= params->GetDouble("dpi", ¶meters.dpi); 195 result &= params->GetDouble("min_shrink", ¶meters.min_shrink); 196 result &= params->GetDouble("max_shrink", ¶meters.max_shrink); 197 result &= params->GetBoolean("selection_only", ¶meters.selection_only); 198 return result; 199 } 200 201 base::string16 GetSwitchValueString16(const CommandLine& command_line, 202 const char* switchName) { 203 #if defined(OS_WIN) 204 return command_line.GetSwitchValueNative(switchName); 205 #elif defined(OS_POSIX) 206 // POSIX Command line string types are different. 207 CommandLine::StringType native_switch_val; 208 native_switch_val = command_line.GetSwitchValueASCII(switchName); 209 // Convert the ASCII string to UTF16 to prepare to pass. 210 return base::ASCIIToUTF16(native_switch_val); 211 #endif 212 } 213 214 void CloudPrintDataSenderHelper::CallJavascriptFunction( 215 const std::string& function_name, 216 const base::Value& arg1, 217 const base::Value& arg2) { 218 web_ui_->CallJavascriptFunction(function_name, arg1, arg2); 219 } 220 221 // Clears out the pointer we're using to communicate. Either routine is 222 // potentially expensive enough that stopping whatever is in progress 223 // is worth it. 224 void CloudPrintDataSender::CancelPrintDataFile() { 225 base::AutoLock lock(lock_); 226 // We don't own helper, it was passed in to us, so no need to 227 // delete, just let it go. 228 helper_ = NULL; 229 } 230 231 CloudPrintDataSender::CloudPrintDataSender( 232 CloudPrintDataSenderHelper* helper, 233 const base::string16& print_job_title, 234 const base::string16& print_ticket, 235 const std::string& file_type, 236 const base::RefCountedMemory* data) 237 : helper_(helper), 238 print_job_title_(print_job_title), 239 print_ticket_(print_ticket), 240 file_type_(file_type), 241 data_(data) { 242 } 243 244 CloudPrintDataSender::~CloudPrintDataSender() {} 245 246 // We have the data in hand that needs to be pushed into the dialog 247 // contents; do so from the IO thread. 248 249 // TODO(scottbyer): If the print data ends up being larger than the 250 // upload limit (currently 10MB), what we need to do is upload that 251 // large data to google docs and set the URL in the printing 252 // JavaScript to that location, and make sure it gets deleted when not 253 // needed. - 4/1/2010 254 void CloudPrintDataSender::SendPrintData() { 255 DCHECK_CURRENTLY_ON(BrowserThread::IO); 256 if (!data_.get() || !data_->size()) 257 return; 258 259 std::string base64_data; 260 base::Base64Encode( 261 base::StringPiece(data_->front_as<char>(), data_->size()), 262 &base64_data); 263 std::string header("data:"); 264 header.append(file_type_); 265 header.append(";base64,"); 266 base64_data.insert(0, header); 267 268 base::AutoLock lock(lock_); 269 if (helper_) { 270 base::StringValue title(print_job_title_); 271 base::StringValue ticket(print_ticket_); 272 // TODO(abodenha): Change Javascript call to pass in print ticket 273 // after server side support is added. Add test for it. 274 275 // Send the print data to the dialog contents. The JavaScript 276 // function is a preliminary API for prototyping purposes and is 277 // subject to change. 278 helper_->CallJavascriptFunction( 279 "printApp._printDataUrl", base::StringValue(base64_data), title); 280 } 281 } 282 283 284 CloudPrintFlowHandler::CloudPrintFlowHandler( 285 const base::RefCountedMemory* data, 286 const base::string16& print_job_title, 287 const base::string16& print_ticket, 288 const std::string& file_type) 289 : dialog_delegate_(NULL), 290 data_(data), 291 print_job_title_(print_job_title), 292 print_ticket_(print_ticket), 293 file_type_(file_type) { 294 } 295 296 CloudPrintFlowHandler::~CloudPrintFlowHandler() { 297 // This will also cancel any task in flight. 298 CancelAnyRunningTask(); 299 } 300 301 302 void CloudPrintFlowHandler::SetDialogDelegate( 303 CloudPrintWebDialogDelegate* delegate) { 304 // Even if setting a new WebUI, it means any previous task needs 305 // to be canceled, its now invalid. 306 DCHECK_CURRENTLY_ON(BrowserThread::UI); 307 CancelAnyRunningTask(); 308 dialog_delegate_ = delegate; 309 } 310 311 // Cancels any print data sender we have in flight and removes our 312 // reference to it, so when the task that is calling it finishes and 313 // removes its reference, it goes away. 314 void CloudPrintFlowHandler::CancelAnyRunningTask() { 315 DCHECK_CURRENTLY_ON(BrowserThread::UI); 316 if (print_data_sender_.get()) { 317 print_data_sender_->CancelPrintDataFile(); 318 print_data_sender_ = NULL; 319 } 320 } 321 322 void CloudPrintFlowHandler::RegisterMessages() { 323 // TODO(scottbyer) - This is where we will register messages for the 324 // UI JS to use. Needed: Call to update page setup parameters. 325 web_ui()->RegisterMessageCallback("ShowDebugger", 326 base::Bind(&CloudPrintFlowHandler::HandleShowDebugger, 327 base::Unretained(this))); 328 web_ui()->RegisterMessageCallback("SendPrintData", 329 base::Bind(&CloudPrintFlowHandler::HandleSendPrintData, 330 base::Unretained(this))); 331 web_ui()->RegisterMessageCallback("SetPageParameters", 332 base::Bind(&CloudPrintFlowHandler::HandleSetPageParameters, 333 base::Unretained(this))); 334 335 // Register for appropriate notifications, and re-direct the URL 336 // to the real server URL, now that we've gotten an HTML dialog 337 // going. 338 NavigationController* controller = 339 &web_ui()->GetWebContents()->GetController(); 340 NavigationEntry* pending_entry = controller->GetPendingEntry(); 341 if (pending_entry) { 342 pending_entry->SetURL(google_util::AppendGoogleLocaleParam( 343 cloud_devices::GetCloudPrintRelativeURL("client/dialog.html"), 344 g_browser_process->GetApplicationLocale())); 345 } 346 registrar_.Add(this, content::NOTIFICATION_LOAD_STOP, 347 content::Source<NavigationController>(controller)); 348 registrar_.Add(this, content::NOTIFICATION_NAV_ENTRY_COMMITTED, 349 content::Source<NavigationController>(controller)); 350 } 351 352 void CloudPrintFlowHandler::Observe( 353 int type, 354 const content::NotificationSource& source, 355 const content::NotificationDetails& details) { 356 switch (type) { 357 case content::NOTIFICATION_LOAD_STOP: { 358 GURL url = web_ui()->GetWebContents()->GetURL(); 359 if (IsCloudPrintDialogUrl(url)) { 360 // Take the opportunity to set some (minimal) additional 361 // script permissions required for the web UI. 362 RenderViewHost* rvh = web_ui()->GetWebContents()->GetRenderViewHost(); 363 if (rvh) { 364 // TODO(chrishtr): this is wrong. allow_scripts_to_close_windows will 365 // be reset the next time a preference changes. 366 WebPreferences webkit_prefs = rvh->GetWebkitPreferences(); 367 webkit_prefs.allow_scripts_to_close_windows = true; 368 rvh->UpdateWebkitPreferences(webkit_prefs); 369 } else { 370 NOTREACHED(); 371 } 372 // Choose one or the other. If you need to debug, bring up the 373 // debugger. You can then use the various chrome.send() 374 // registrations above to kick of the various function calls, 375 // including chrome.send("SendPrintData") in the javaScript 376 // console and watch things happen with: 377 // HandleShowDebugger(NULL); 378 HandleSendPrintData(NULL); 379 } 380 break; 381 } 382 } 383 } 384 385 void CloudPrintFlowHandler::HandleShowDebugger(const base::ListValue* args) { 386 ShowDebugger(); 387 } 388 389 void CloudPrintFlowHandler::ShowDebugger() { 390 if (web_ui()) { 391 WebContents* web_contents = web_ui()->GetWebContents(); 392 if (web_contents) 393 DevToolsWindow::OpenDevToolsWindow(web_contents); 394 } 395 } 396 397 scoped_refptr<CloudPrintDataSender> 398 CloudPrintFlowHandler::CreateCloudPrintDataSender() { 399 DCHECK(web_ui()); 400 print_data_helper_.reset(new CloudPrintDataSenderHelper(web_ui())); 401 scoped_refptr<CloudPrintDataSender> sender( 402 new CloudPrintDataSender(print_data_helper_.get(), 403 print_job_title_, 404 print_ticket_, 405 file_type_, 406 data_.get())); 407 return sender; 408 } 409 410 void CloudPrintFlowHandler::HandleSendPrintData(const base::ListValue* args) { 411 DCHECK_CURRENTLY_ON(BrowserThread::UI); 412 // This will cancel any ReadPrintDataFile() or SendPrintDataFile() 413 // requests in flight (this is anticipation of when setting page 414 // setup parameters becomes asynchronous and may be set while some 415 // data is in flight). Then we can clear out the print data. 416 CancelAnyRunningTask(); 417 if (web_ui()) { 418 print_data_sender_ = CreateCloudPrintDataSender(); 419 BrowserThread::PostTask( 420 BrowserThread::IO, FROM_HERE, 421 base::Bind(&CloudPrintDataSender::SendPrintData, print_data_sender_)); 422 } 423 } 424 425 void CloudPrintFlowHandler::HandleSetPageParameters( 426 const base::ListValue* args) { 427 std::string json; 428 bool ret = args->GetString(0, &json); 429 if (!ret || json.empty()) { 430 NOTREACHED() << "Empty json string"; 431 return; 432 } 433 434 // These are backstop default values - 72 dpi to match the screen, 435 // 8.5x11 inch paper with margins subtracted (1/4 inch top, left, 436 // right and 0.56 bottom), and the min page shrink and max page 437 // shrink values appear all over the place with no explanation. 438 439 // TODO(scottbyer): Get a Linux/ChromeOS edge for PrintSettings 440 // working so that we can get the default values from there. Fix up 441 // PrintWebViewHelper to do the same. 442 const int kDPI = 72; 443 const int kWidth = static_cast<int>((8.5-0.25-0.25)*kDPI); 444 const int kHeight = static_cast<int>((11-0.25-0.56)*kDPI); 445 const double kMinPageShrink = 1.25; 446 const double kMaxPageShrink = 2.0; 447 448 PrintMsg_Print_Params default_settings; 449 default_settings.content_size = gfx::Size(kWidth, kHeight); 450 default_settings.printable_area = gfx::Rect(0, 0, kWidth, kHeight); 451 default_settings.dpi = kDPI; 452 default_settings.min_shrink = kMinPageShrink; 453 default_settings.max_shrink = kMaxPageShrink; 454 default_settings.desired_dpi = kDPI; 455 default_settings.document_cookie = 0; 456 default_settings.selection_only = false; 457 default_settings.preview_request_id = 0; 458 default_settings.is_first_request = true; 459 default_settings.print_to_pdf = false; 460 461 if (!GetPageSetupParameters(json, default_settings)) { 462 NOTREACHED(); 463 return; 464 } 465 466 // TODO(scottbyer) - Here is where we would kick the originating 467 // renderer thread with these new parameters in order to get it to 468 // re-generate the PDF data and hand it back to us. window.print() is 469 // currently synchronous, so there's a lot of work to do to get to 470 // that point. 471 } 472 473 void CloudPrintFlowHandler::StoreDialogClientSize() const { 474 if (web_ui() && web_ui()->GetWebContents()) { 475 gfx::Size size = web_ui()->GetWebContents()->GetContainerBounds().size(); 476 Profile* profile = Profile::FromWebUI(web_ui()); 477 profile->GetPrefs()->SetInteger(prefs::kCloudPrintDialogWidth, 478 size.width()); 479 profile->GetPrefs()->SetInteger(prefs::kCloudPrintDialogHeight, 480 size.height()); 481 } 482 } 483 484 bool CloudPrintFlowHandler::IsCloudPrintDialogUrl(const GURL& url) { 485 GURL cloud_print_url = cloud_devices::GetCloudPrintURL(); 486 return IsSimilarUrl(url, cloud_print_url); 487 } 488 489 CloudPrintWebDialogDelegate::CloudPrintWebDialogDelegate( 490 content::BrowserContext* browser_context, 491 gfx::NativeWindow modal_parent, 492 const base::RefCountedMemory* data, 493 const std::string& json_arguments, 494 const base::string16& print_job_title, 495 const base::string16& print_ticket, 496 const std::string& file_type) 497 : flow_handler_( 498 new CloudPrintFlowHandler(data, print_job_title, print_ticket, 499 file_type)), 500 modal_parent_(modal_parent), 501 owns_flow_handler_(true), 502 keep_alive_when_non_modal_(true) { 503 Init(browser_context, json_arguments); 504 } 505 506 // For unit testing. 507 CloudPrintWebDialogDelegate::CloudPrintWebDialogDelegate( 508 CloudPrintFlowHandler* flow_handler, 509 const std::string& json_arguments) 510 : flow_handler_(flow_handler), 511 modal_parent_(NULL), 512 owns_flow_handler_(true), 513 keep_alive_when_non_modal_(false) { 514 Init(NULL, json_arguments); 515 } 516 517 // Returns the persisted width/height for the print dialog. 518 void GetDialogWidthAndHeightFromPrefs(content::BrowserContext* browser_context, 519 int* width, 520 int* height) { 521 if (!browser_context) { 522 *width = kDefaultWidth; 523 *height = kDefaultHeight; 524 return; 525 } 526 527 PrefService* prefs = Profile::FromBrowserContext(browser_context)->GetPrefs(); 528 *width = prefs->GetInteger(prefs::kCloudPrintDialogWidth); 529 *height = prefs->GetInteger(prefs::kCloudPrintDialogHeight); 530 } 531 532 void CloudPrintWebDialogDelegate::Init(content::BrowserContext* browser_context, 533 const std::string& json_arguments) { 534 // This information is needed to show the dialog HTML content. 535 DCHECK_CURRENTLY_ON(BrowserThread::UI); 536 537 params_.url = GURL(chrome::kChromeUICloudPrintResourcesURL); 538 GetDialogWidthAndHeightFromPrefs(browser_context, 539 ¶ms_.width, 540 ¶ms_.height); 541 params_.json_input = json_arguments; 542 543 flow_handler_->SetDialogDelegate(this); 544 // If we're not modal we can show the dialog with no browser. 545 // We need this to keep Chrome alive while our dialog is up. 546 if (!modal_parent_ && keep_alive_when_non_modal_) 547 chrome::IncrementKeepAliveCount(); 548 } 549 550 CloudPrintWebDialogDelegate::~CloudPrintWebDialogDelegate() { 551 // If the flow_handler_ is about to outlive us because we don't own 552 // it anymore, we need to have it remove its reference to us. 553 DCHECK_CURRENTLY_ON(BrowserThread::UI); 554 flow_handler_->SetDialogDelegate(NULL); 555 if (owns_flow_handler_) { 556 delete flow_handler_; 557 } 558 } 559 560 ui::ModalType CloudPrintWebDialogDelegate::GetDialogModalType() const { 561 return modal_parent_ ? ui::MODAL_TYPE_WINDOW : ui::MODAL_TYPE_NONE; 562 } 563 564 base::string16 CloudPrintWebDialogDelegate::GetDialogTitle() const { 565 return base::string16(); 566 } 567 568 GURL CloudPrintWebDialogDelegate::GetDialogContentURL() const { 569 return params_.url; 570 } 571 572 void CloudPrintWebDialogDelegate::GetWebUIMessageHandlers( 573 std::vector<WebUIMessageHandler*>* handlers) const { 574 handlers->push_back(flow_handler_); 575 // We don't own flow_handler_ anymore, but it sticks around until at 576 // least right after OnDialogClosed() is called (and this object is 577 // destroyed). 578 owns_flow_handler_ = false; 579 } 580 581 void CloudPrintWebDialogDelegate::GetDialogSize(gfx::Size* size) const { 582 size->set_width(params_.width); 583 size->set_height(params_.height); 584 } 585 586 std::string CloudPrintWebDialogDelegate::GetDialogArgs() const { 587 return params_.json_input; 588 } 589 590 void CloudPrintWebDialogDelegate::OnDialogClosed( 591 const std::string& json_retval) { 592 // Get the final dialog size and store it. 593 flow_handler_->StoreDialogClientSize(); 594 595 // If we're modal we can show the dialog with no browser. 596 // End the keep-alive so that Chrome can exit. 597 if (!modal_parent_ && keep_alive_when_non_modal_) { 598 // Post to prevent recursive call tho this function. 599 base::MessageLoop::current()->PostTask( 600 FROM_HERE, base::Bind(&chrome::DecrementKeepAliveCount)); 601 } 602 delete this; 603 } 604 605 void CloudPrintWebDialogDelegate::OnCloseContents(WebContents* source, 606 bool* out_close_dialog) { 607 if (out_close_dialog) 608 *out_close_dialog = true; 609 } 610 611 bool CloudPrintWebDialogDelegate::ShouldShowDialogTitle() const { 612 return false; 613 } 614 615 bool CloudPrintWebDialogDelegate::HandleContextMenu( 616 const content::ContextMenuParams& params) { 617 return true; 618 } 619 620 // Called from the UI thread, starts up the dialog. 621 void CreateDialogImpl(content::BrowserContext* browser_context, 622 gfx::NativeWindow modal_parent, 623 const base::RefCountedMemory* data, 624 const base::string16& print_job_title, 625 const base::string16& print_ticket, 626 const std::string& file_type) { 627 DCHECK_CURRENTLY_ON(BrowserThread::UI); 628 WebDialogDelegate* dialog_delegate = 629 new internal_cloud_print_helpers::CloudPrintWebDialogDelegate( 630 browser_context, modal_parent, data, std::string(), print_job_title, 631 print_ticket, file_type); 632 #if defined(OS_WIN) 633 gfx::NativeWindow window = 634 #endif 635 chrome::ShowWebDialog(modal_parent, 636 Profile::FromBrowserContext(browser_context), 637 dialog_delegate); 638 #if defined(OS_WIN) 639 if (window) { 640 HWND dialog_handle; 641 #if defined(USE_AURA) 642 dialog_handle = window->GetHost()->GetAcceleratedWidget(); 643 #else 644 dialog_handle = window; 645 #endif 646 if (::GetForegroundWindow() != dialog_handle) { 647 ui::ForegroundHelper::SetForeground(dialog_handle); 648 } 649 } 650 #endif 651 } 652 653 void CreateDialogForFileImpl(content::BrowserContext* browser_context, 654 gfx::NativeWindow modal_parent, 655 const base::FilePath& path_to_file, 656 const base::string16& print_job_title, 657 const base::string16& print_ticket, 658 const std::string& file_type) { 659 DCHECK_CURRENTLY_ON(BrowserThread::FILE); 660 scoped_refptr<base::RefCountedMemory> data; 661 int64 file_size = 0; 662 if (base::GetFileSize(path_to_file, &file_size) && file_size != 0) { 663 std::string file_data; 664 if (file_size < kuint32max) { 665 file_data.reserve(static_cast<unsigned int>(file_size)); 666 } else { 667 DLOG(WARNING) << " print data file too large to reserve space"; 668 } 669 if (base::ReadFileToString(path_to_file, &file_data)) { 670 data = base::RefCountedString::TakeString(&file_data); 671 } 672 } 673 // Proceed even for empty data to simplify testing. 674 BrowserThread::PostTask( 675 BrowserThread::UI, FROM_HERE, 676 base::Bind(&print_dialog_cloud::CreatePrintDialogForBytes, 677 browser_context, modal_parent, data, print_job_title, 678 print_ticket, file_type)); 679 base::DeleteFile(path_to_file, false); 680 } 681 682 } // namespace internal_cloud_print_helpers 683 684 namespace print_dialog_cloud { 685 686 void RegisterProfilePrefs(user_prefs::PrefRegistrySyncable* registry) { 687 registry->RegisterIntegerPref( 688 prefs::kCloudPrintDialogWidth, 689 kDefaultWidth, 690 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); 691 registry->RegisterIntegerPref( 692 prefs::kCloudPrintDialogHeight, 693 kDefaultHeight, 694 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); 695 } 696 697 // Called on the FILE or UI thread. This is the main entry point into creating 698 // the dialog. 699 700 void CreatePrintDialogForFile(content::BrowserContext* browser_context, 701 gfx::NativeWindow modal_parent, 702 const base::FilePath& path_to_file, 703 const base::string16& print_job_title, 704 const base::string16& print_ticket, 705 const std::string& file_type) { 706 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE) || 707 BrowserThread::CurrentlyOn(BrowserThread::UI)); 708 BrowserThread::PostTask( 709 BrowserThread::FILE, FROM_HERE, 710 base::Bind(&internal_cloud_print_helpers::CreateDialogForFileImpl, 711 browser_context, modal_parent, path_to_file, print_job_title, 712 print_ticket, file_type)); 713 } 714 715 void CreateCloudPrintSigninTab(Browser* browser, 716 bool add_account, 717 const base::Closure& callback) { 718 DCHECK_CURRENTLY_ON(BrowserThread::UI); 719 if (switches::IsEnableAccountConsistency() && 720 !browser->profile()->IsOffTheRecord()) { 721 browser->window()->ShowAvatarBubbleFromAvatarButton( 722 add_account ? BrowserWindow::AVATAR_BUBBLE_MODE_ADD_ACCOUNT 723 : BrowserWindow::AVATAR_BUBBLE_MODE_SIGNIN, 724 signin::ManageAccountsParams()); 725 } else { 726 GURL url = add_account ? cloud_devices::GetCloudPrintAddAccountURL() 727 : cloud_devices::GetCloudPrintSigninURL(); 728 content::WebContents* web_contents = 729 browser->OpenURL(content::OpenURLParams( 730 google_util::AppendGoogleLocaleParam( 731 url, g_browser_process->GetApplicationLocale()), 732 content::Referrer(), 733 NEW_FOREGROUND_TAB, 734 ui::PAGE_TRANSITION_AUTO_BOOKMARK, 735 false)); 736 new SignInObserver(web_contents, cloud_devices::GetCloudPrintURL(), 737 callback); 738 } 739 } 740 741 void CreatePrintDialogForBytes(content::BrowserContext* browser_context, 742 gfx::NativeWindow modal_parent, 743 const base::RefCountedMemory* data, 744 const base::string16& print_job_title, 745 const base::string16& print_ticket, 746 const std::string& file_type) { 747 internal_cloud_print_helpers::CreateDialogImpl(browser_context, modal_parent, 748 data, print_job_title, 749 print_ticket, file_type); 750 } 751 752 bool CreatePrintDialogFromCommandLine(Profile* profile, 753 const CommandLine& command_line) { 754 DCHECK(command_line.HasSwitch(switches::kCloudPrintFile)); 755 if (!command_line.GetSwitchValuePath(switches::kCloudPrintFile).empty()) { 756 base::FilePath cloud_print_file; 757 cloud_print_file = 758 command_line.GetSwitchValuePath(switches::kCloudPrintFile); 759 if (!cloud_print_file.empty()) { 760 base::string16 print_job_title; 761 base::string16 print_job_print_ticket; 762 if (command_line.HasSwitch(switches::kCloudPrintJobTitle)) { 763 print_job_title = 764 internal_cloud_print_helpers::GetSwitchValueString16( 765 command_line, switches::kCloudPrintJobTitle); 766 } 767 if (command_line.HasSwitch(switches::kCloudPrintPrintTicket)) { 768 print_job_print_ticket = 769 internal_cloud_print_helpers::GetSwitchValueString16( 770 command_line, switches::kCloudPrintPrintTicket); 771 } 772 std::string file_type = "application/pdf"; 773 if (command_line.HasSwitch(switches::kCloudPrintFileType)) { 774 file_type = command_line.GetSwitchValueASCII( 775 switches::kCloudPrintFileType); 776 } 777 778 print_dialog_cloud::CreatePrintDialogForFile(profile, NULL, 779 cloud_print_file, print_job_title, print_job_print_ticket, file_type); 780 return true; 781 } 782 } 783 return false; 784 } 785 786 } // namespace print_dialog_cloud 787