blob: cce8dea599fa158203a85fcc9e0cf64084a08bb9 [file] [log] [blame]
[email protected]87ab41e72012-01-04 18:45:111// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]73852b8f2010-05-14 00:38:122// 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#include "chrome/browser/printing/print_dialog_cloud_internal.h"
7
[email protected]73852b8f2010-05-14 00:38:128#include "base/base64.h"
[email protected]ba4fc242011-10-04 18:56:569#include "base/bind.h"
10#include "base/bind_helpers.h"
[email protected]65c9d89a2011-04-13 21:02:3911#include "base/command_line.h"
[email protected]73852b8f2010-05-14 00:38:1212#include "base/file_util.h"
13#include "base/json/json_reader.h"
[email protected]3853a4c2013-02-11 17:15:5714#include "base/prefs/pref_service.h"
[email protected]4b8852a2011-06-10 17:24:4615#include "base/utf_string_conversions.h"
[email protected]73852b8f2010-05-14 00:38:1216#include "base/values.h"
[email protected]70019152012-12-19 11:44:1917#include "chrome/browser/devtools/devtools_window.h"
[email protected]2e6389f2012-05-18 19:41:2518#include "chrome/browser/lifetime/application_lifetime.h"
[email protected]2283eead2010-09-29 23:17:3019#include "chrome/browser/printing/cloud_print/cloud_print_url.h"
[email protected]8ecad5e2010-12-02 21:18:3320#include "chrome/browser/profiles/profile.h"
[email protected]e39027a2011-01-24 21:41:5421#include "chrome/browser/profiles/profile_manager.h"
[email protected]508326df2012-05-23 16:01:1922#include "chrome/browser/ui/browser_dialogs.h"
[email protected]65c9d89a2011-04-13 21:02:3923#include "chrome/common/chrome_switches.h"
[email protected]ea161da2010-11-02 21:57:3524#include "chrome/common/pref_names.h"
[email protected]1375e3ab2011-03-24 17:07:2225#include "chrome/common/print_messages.h"
[email protected]73852b8f2010-05-14 00:38:1226#include "chrome/common/url_constants.h"
[email protected]75fee372013-03-06 00:42:4427#include "components/user_prefs/pref_registry_syncable.h"
[email protected]c38831a12011-10-28 12:44:4928#include "content/public/browser/browser_thread.h"
[email protected]cdcb1dee2012-01-04 00:46:2029#include "content/public/browser/navigation_controller.h"
[email protected]022af742011-12-28 18:37:2530#include "content/public/browser/navigation_entry.h"
[email protected]6c2381d2011-10-19 02:52:5331#include "content/public/browser/notification_registrar.h"
32#include "content/public/browser/notification_source.h"
[email protected]0d6e9bd2011-10-18 04:29:1633#include "content/public/browser/notification_types.h"
[email protected]9c1662b2012-03-06 15:44:3334#include "content/public/browser/render_view_host.h"
[email protected]0ec4898e2011-12-30 21:09:2435#include "content/public/browser/web_contents.h"
[email protected]8643e6d2012-01-18 20:26:1036#include "content/public/browser/web_contents_view.h"
[email protected]01ec4ec2012-01-18 04:13:4737#include "content/public/browser/web_ui.h"
[email protected]fca6b9b2012-05-22 17:11:0538#include "grit/generated_resources.h"
[email protected]c051a1b2011-01-21 23:30:1739#include "ui/base/l10n/l10n_util.h"
[email protected]fca6b9b2012-05-22 17:11:0540#include "webkit/glue/webpreferences.h"
[email protected]520c2022012-03-15 00:13:1541
[email protected]88bfd25b2012-06-22 06:28:3342#if defined(USE_AURA)
43#include "ui/aura/root_window.h"
44#include "ui/aura/window.h"
45#endif
46
[email protected]520c2022012-03-15 00:13:1547#if defined(OS_WIN)
48#include "ui/base/win/foreground_helper.h"
49#endif
50
[email protected]73852b8f2010-05-14 00:38:1251// This module implements the UI support in Chrome for cloud printing.
52// This means hosting a dialog containing HTML/JavaScript and using
53// the published cloud print user interface integration APIs to get
54// page setup settings from the dialog contents and provide the
[email protected]a984bdf2011-03-15 20:17:1655// generated print data to the dialog contents for uploading to the
[email protected]73852b8f2010-05-14 00:38:1256// cloud print service.
57
58// Currently, the flow between these classes is as follows:
59
[email protected]a984bdf2011-03-15 20:17:1660// PrintDialogCloud::CreatePrintDialogForFile is called from
[email protected]73852b8f2010-05-14 00:38:1261// resource_message_filter_gtk.cc once the renderer has informed the
[email protected]a984bdf2011-03-15 20:17:1662// renderer host that print data generation into the renderer host provided
[email protected]032682b2011-01-12 22:05:0263// temp file has been completed. That call is on the FILE thread.
[email protected]73852b8f2010-05-14 00:38:1264// That, in turn, hops over to the UI thread to create an instance of
65// PrintDialogCloud.
66
67// The constructor for PrintDialogCloud creates a
[email protected]5835871a2012-04-25 21:56:5568// CloudPrintWebDialogDelegate and asks the current active browser to
[email protected]73852b8f2010-05-14 00:38:1269// show an HTML dialog using that class as the delegate. That class
[email protected]89f550b2011-06-08 18:34:0370// hands in the kChromeUICloudPrintResourcesURL as the URL to visit. That is
[email protected]80a8fad2011-01-29 04:02:3871// recognized by the GetWebUIFactoryFunction as a signal to create an
[email protected]02b5ccc2012-04-30 23:58:3172// ExternalWebDialogUI.
[email protected]73852b8f2010-05-14 00:38:1273
[email protected]5835871a2012-04-25 21:56:5574// CloudPrintWebDialogDelegate also temporarily owns a
[email protected]73852b8f2010-05-14 00:38:1275// CloudPrintFlowHandler, a class which is responsible for the actual
[email protected]a984bdf2011-03-15 20:17:1676// interactions with the dialog contents, including handing in the
[email protected]73852b8f2010-05-14 00:38:1277// print data and getting any page setup parameters that the dialog
78// contents provides. As part of bringing up the dialog,
[email protected]02b5ccc2012-04-30 23:58:3179// WebDialogUI::RenderViewCreated is called (an override of
[email protected]c39f9bf2011-02-12 00:43:5580// WebUI::RenderViewCreated). That routine, in turn, calls the
[email protected]36e12172011-02-08 23:46:0281// delegate's GetWebUIMessageHandlers routine, at which point the
[email protected]73852b8f2010-05-14 00:38:1282// ownership of the CloudPrintFlowHandler is handed over. A pointer
83// to the flow handler is kept to facilitate communication back and
84// forth between the two classes.
85
[email protected]c39f9bf2011-02-12 00:43:5586// The WebUI continues dialog bring-up, calling
[email protected]73852b8f2010-05-14 00:38:1287// CloudPrintFlowHandler::RegisterMessages. This is where the
88// additional object model capabilities are registered for the dialog
89// contents to use. It is also at this time that capabilities for the
90// dialog contents are adjusted to allow the dialog contents to close
91// the window. In addition, the pending URL is redirected to the
92// actual cloud print service URL. The flow controller also registers
93// for notification of when the dialog contents finish loading, which
[email protected]a984bdf2011-03-15 20:17:1694// is currently used to send the data to the dialog contents.
[email protected]73852b8f2010-05-14 00:38:1295
[email protected]a984bdf2011-03-15 20:17:1696// In order to send the data to the dialog contents, the flow
[email protected]73852b8f2010-05-14 00:38:1297// handler uses a CloudPrintDataSender. It creates one, letting it
[email protected]a984bdf2011-03-15 20:17:1698// know the name of the temporary file containing the data, and
[email protected]73852b8f2010-05-14 00:38:1299// posts the task of reading the file
100// (CloudPrintDataSender::ReadPrintDataFile) to the file thread. That
101// routine reads in the file, and then hops over to the IO thread to
102// send that data to the dialog contents.
103
104// When the dialog contents are finished (by either being cancelled or
105// hitting the print button), the delegate is notified, and responds
106// that the dialog should be closed, at which point things are torn
107// down and released.
108
[email protected]631bb742011-11-02 11:29:39109using content::BrowserThread;
[email protected]c5eed492012-01-04 17:07:50110using content::NavigationController;
[email protected]10f417c52011-12-28 21:04:23111using content::NavigationEntry;
[email protected]eaabba22012-03-07 15:02:11112using content::RenderViewHost;
[email protected]a81343d232011-12-27 07:39:20113using content::WebContents;
[email protected]26e2632a2011-12-31 04:02:55114using content::WebUIMessageHandler;
[email protected]20c07f8e2012-05-31 08:43:14115using ui::WebDialogDelegate;
[email protected]631bb742011-11-02 11:29:39116
[email protected]681958c2013-02-21 13:48:14117const int kDefaultWidth = 912;
118const int kDefaultHeight = 633;
119
[email protected]73852b8f2010-05-14 00:38:12120namespace internal_cloud_print_helpers {
121
[email protected]73852b8f2010-05-14 00:38:12122// From the JSON parsed value, get the entries for the page setup
123// parameters.
124bool GetPageSetupParameters(const std::string& json,
[email protected]1375e3ab2011-03-24 17:07:22125 PrintMsg_Print_Params& parameters) {
[email protected]cd5785752012-04-11 00:15:41126 scoped_ptr<Value> parsed_value(base::JSONReader::Read(json));
[email protected]73852b8f2010-05-14 00:38:12127 DLOG_IF(ERROR, (!parsed_value.get() ||
128 !parsed_value->IsType(Value::TYPE_DICTIONARY)))
129 << "PageSetup call didn't have expected contents";
130 if (!parsed_value.get() || !parsed_value->IsType(Value::TYPE_DICTIONARY))
131 return false;
132
133 bool result = true;
134 DictionaryValue* params = static_cast<DictionaryValue*>(parsed_value.get());
[email protected]05c7da62011-05-05 17:23:56135 result &= params->GetDouble("dpi", &parameters.dpi);
136 result &= params->GetDouble("min_shrink", &parameters.min_shrink);
137 result &= params->GetDouble("max_shrink", &parameters.max_shrink);
[email protected]a65175d2010-08-17 04:00:57138 result &= params->GetBoolean("selection_only", &parameters.selection_only);
[email protected]73852b8f2010-05-14 00:38:12139 return result;
140}
141
[email protected]e8368e92011-08-20 04:05:56142string16 GetSwitchValueString16(const CommandLine& command_line,
143 const char* switchName) {
[email protected]31662202013-03-23 19:10:54144#if defined(OS_WIN)
[email protected]e8368e92011-08-20 04:05:56145 CommandLine::StringType native_switch_val;
146 native_switch_val = command_line.GetSwitchValueNative(switchName);
147 return string16(native_switch_val);
148#elif defined(OS_POSIX)
149 // POSIX Command line string types are different.
150 CommandLine::StringType native_switch_val;
151 native_switch_val = command_line.GetSwitchValueASCII(switchName);
152 // Convert the ASCII string to UTF16 to prepare to pass.
153 return string16(ASCIIToUTF16(native_switch_val));
154#endif
155}
156
[email protected]73852b8f2010-05-14 00:38:12157void CloudPrintDataSenderHelper::CallJavascriptFunction(
158 const std::wstring& function_name) {
[email protected]47e870b2013-02-24 21:14:53159 web_ui_->CallJavascriptFunction(WideToASCII(function_name));
[email protected]73852b8f2010-05-14 00:38:12160}
161
162void CloudPrintDataSenderHelper::CallJavascriptFunction(
163 const std::wstring& function_name, const Value& arg) {
[email protected]47e870b2013-02-24 21:14:53164 web_ui_->CallJavascriptFunction(WideToASCII(function_name), arg);
[email protected]73852b8f2010-05-14 00:38:12165}
166
167void CloudPrintDataSenderHelper::CallJavascriptFunction(
168 const std::wstring& function_name, const Value& arg1, const Value& arg2) {
[email protected]47e870b2013-02-24 21:14:53169 web_ui_->CallJavascriptFunction(WideToASCII(function_name), arg1, arg2);
[email protected]73852b8f2010-05-14 00:38:12170}
171
[email protected]e8368e92011-08-20 04:05:56172void CloudPrintDataSenderHelper::CallJavascriptFunction(
173 const std::wstring& function_name,
174 const Value& arg1,
175 const Value& arg2,
176 const Value& arg3) {
[email protected]46adf7ff2011-12-30 00:53:09177 web_ui_->CallJavascriptFunction(
[email protected]47e870b2013-02-24 21:14:53178 WideToASCII(function_name), arg1, arg2, arg3);
[email protected]e8368e92011-08-20 04:05:56179}
180
[email protected]73852b8f2010-05-14 00:38:12181// Clears out the pointer we're using to communicate. Either routine is
182// potentially expensive enough that stopping whatever is in progress
183// is worth it.
184void CloudPrintDataSender::CancelPrintDataFile() {
[email protected]20305ec2011-01-21 04:55:52185 base::AutoLock lock(lock_);
[email protected]73852b8f2010-05-14 00:38:12186 // We don't own helper, it was passed in to us, so no need to
187 // delete, just let it go.
188 helper_ = NULL;
189}
190
[email protected]38e08982010-10-22 17:28:43191CloudPrintDataSender::CloudPrintDataSender(CloudPrintDataSenderHelper* helper,
[email protected]a984bdf2011-03-15 20:17:16192 const string16& print_job_title,
[email protected]e8368e92011-08-20 04:05:56193 const string16& print_ticket,
[email protected]a9723e12013-03-05 04:02:45194 const std::string& file_type,
195 const base::RefCountedMemory* data)
[email protected]38e08982010-10-22 17:28:43196 : helper_(helper),
[email protected]a984bdf2011-03-15 20:17:16197 print_job_title_(print_job_title),
[email protected]e8368e92011-08-20 04:05:56198 print_ticket_(print_ticket),
[email protected]a9723e12013-03-05 04:02:45199 file_type_(file_type),
200 data_(data) {
[email protected]38e08982010-10-22 17:28:43201}
202
203CloudPrintDataSender::~CloudPrintDataSender() {}
204
[email protected]73852b8f2010-05-14 00:38:12205// We have the data in hand that needs to be pushed into the dialog
206// contents; do so from the IO thread.
207
208// TODO(scottbyer): If the print data ends up being larger than the
209// upload limit (currently 10MB), what we need to do is upload that
210// large data to google docs and set the URL in the printing
211// JavaScript to that location, and make sure it gets deleted when not
212// needed. - 4/1/2010
[email protected]a9723e12013-03-05 04:02:45213void CloudPrintDataSender::SendPrintData() {
[email protected]ba4f1132010-10-09 02:02:35214 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
[email protected]a9723e12013-03-05 04:02:45215 if (!data_ || !data_->size())
216 return;
217
218 std::string base64_data;
219 base::Base64Encode(
220 base::StringPiece(reinterpret_cast<const char*>(data_->front()),
221 data_->size()),
222 &base64_data);
223 data_ = NULL;
224 std::string header("data:");
225 header.append(file_type_);
226 header.append(";base64,");
227 base64_data.insert(0, header);
228
[email protected]20305ec2011-01-21 04:55:52229 base::AutoLock lock(lock_);
[email protected]a9723e12013-03-05 04:02:45230 if (helper_) {
[email protected]9848c7e2010-06-03 16:06:56231 StringValue title(print_job_title_);
[email protected]e8368e92011-08-20 04:05:56232 StringValue ticket(print_ticket_);
233 // TODO(abodenha): Change Javascript call to pass in print ticket
234 // after server side support is added. Add test for it.
[email protected]73852b8f2010-05-14 00:38:12235
236 // Send the print data to the dialog contents. The JavaScript
237 // function is a preliminary API for prototyping purposes and is
238 // subject to change.
239 const_cast<CloudPrintDataSenderHelper*>(helper_)->CallJavascriptFunction(
[email protected]a9723e12013-03-05 04:02:45240 L"printApp._printDataUrl", StringValue(base64_data), title);
[email protected]73852b8f2010-05-14 00:38:12241 }
242}
243
244
[email protected]a9723e12013-03-05 04:02:45245CloudPrintFlowHandler::CloudPrintFlowHandler(const base::RefCountedMemory* data,
[email protected]a984bdf2011-03-15 20:17:16246 const string16& print_job_title,
[email protected]e8368e92011-08-20 04:05:56247 const string16& print_ticket,
[email protected]4cd49022012-01-19 20:37:37248 const std::string& file_type,
249 bool close_after_signin,
250 const base::Closure& callback)
[email protected]c7bf7452011-09-12 21:31:50251 : dialog_delegate_(NULL),
[email protected]a9723e12013-03-05 04:02:45252 data_(data),
[email protected]a984bdf2011-03-15 20:17:16253 print_job_title_(print_job_title),
[email protected]e8368e92011-08-20 04:05:56254 print_ticket_(print_ticket),
[email protected]4cd49022012-01-19 20:37:37255 file_type_(file_type),
256 close_after_signin_(close_after_signin),
257 callback_(callback) {
[email protected]38e08982010-10-22 17:28:43258}
259
260CloudPrintFlowHandler::~CloudPrintFlowHandler() {
261 // This will also cancel any task in flight.
262 CancelAnyRunningTask();
263}
264
265
[email protected]73852b8f2010-05-14 00:38:12266void CloudPrintFlowHandler::SetDialogDelegate(
[email protected]5835871a2012-04-25 21:56:55267 CloudPrintWebDialogDelegate* delegate) {
[email protected]7b748982011-02-14 19:28:23268 // Even if setting a new WebUI, it means any previous task needs
[email protected]a2c92a1c2012-04-03 12:32:14269 // to be canceled, its now invalid.
[email protected]ba4f1132010-10-09 02:02:35270 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
[email protected]73852b8f2010-05-14 00:38:12271 CancelAnyRunningTask();
272 dialog_delegate_ = delegate;
273}
274
275// Cancels any print data sender we have in flight and removes our
276// reference to it, so when the task that is calling it finishes and
[email protected]a2c92a1c2012-04-03 12:32:14277// removes its reference, it goes away.
[email protected]73852b8f2010-05-14 00:38:12278void CloudPrintFlowHandler::CancelAnyRunningTask() {
[email protected]ba4f1132010-10-09 02:02:35279 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
[email protected]73852b8f2010-05-14 00:38:12280 if (print_data_sender_.get()) {
281 print_data_sender_->CancelPrintDataFile();
282 print_data_sender_ = NULL;
283 }
284}
285
[email protected]73852b8f2010-05-14 00:38:12286void CloudPrintFlowHandler::RegisterMessages() {
[email protected]73852b8f2010-05-14 00:38:12287 // TODO(scottbyer) - This is where we will register messages for the
288 // UI JS to use. Needed: Call to update page setup parameters.
[email protected]46adf7ff2011-12-30 00:53:09289 web_ui()->RegisterMessageCallback("ShowDebugger",
[email protected]ba4fc242011-10-04 18:56:56290 base::Bind(&CloudPrintFlowHandler::HandleShowDebugger,
291 base::Unretained(this)));
[email protected]46adf7ff2011-12-30 00:53:09292 web_ui()->RegisterMessageCallback("SendPrintData",
[email protected]ba4fc242011-10-04 18:56:56293 base::Bind(&CloudPrintFlowHandler::HandleSendPrintData,
294 base::Unretained(this)));
[email protected]46adf7ff2011-12-30 00:53:09295 web_ui()->RegisterMessageCallback("SetPageParameters",
[email protected]ba4fc242011-10-04 18:56:56296 base::Bind(&CloudPrintFlowHandler::HandleSetPageParameters,
297 base::Unretained(this)));
[email protected]73852b8f2010-05-14 00:38:12298
[email protected]0eb25c42011-08-11 14:50:14299 // Register for appropriate notifications, and re-direct the URL
300 // to the real server URL, now that we've gotten an HTML dialog
301 // going.
[email protected]c5eed492012-01-04 17:07:50302 NavigationController* controller =
[email protected]01ec4ec2012-01-18 04:13:47303 &web_ui()->GetWebContents()->GetController();
[email protected]10f417c52011-12-28 21:04:23304 NavigationEntry* pending_entry = controller->GetPendingEntry();
[email protected]0eb25c42011-08-11 14:50:14305 if (pending_entry) {
[email protected]46adf7ff2011-12-30 00:53:09306 Profile* profile = Profile::FromWebUI(web_ui());
[email protected]4cd49022012-01-19 20:37:37307 if (close_after_signin_) {
308 pending_entry->SetURL(
309 CloudPrintURL(profile).GetCloudPrintSigninURL());
310 } else {
311 pending_entry->SetURL(
312 CloudPrintURL(profile).GetCloudPrintServiceDialogURL());
313 }
[email protected]73852b8f2010-05-14 00:38:12314 }
[email protected]0eb25c42011-08-11 14:50:14315 registrar_.Add(this, content::NOTIFICATION_LOAD_STOP,
[email protected]c5eed492012-01-04 17:07:50316 content::Source<NavigationController>(controller));
[email protected]57c8cf22013-03-02 16:50:00317 registrar_.Add(this, content::NOTIFICATION_NAV_ENTRY_COMMITTED,
318 content::Source<NavigationController>(controller));
[email protected]73852b8f2010-05-14 00:38:12319}
320
[email protected]6c2381d2011-10-19 02:52:53321void CloudPrintFlowHandler::Observe(
322 int type,
323 const content::NotificationSource& source,
324 const content::NotificationDetails& details) {
[email protected]57c8cf22013-03-02 16:50:00325 switch (type) {
326 case content::NOTIFICATION_NAV_ENTRY_COMMITTED: {
327 NavigationEntry* entry =
328 web_ui()->GetWebContents()->GetController().GetActiveEntry();
329 if (entry)
330 NavigationToURLDidCloseDialog(entry->GetURL());
331 break;
[email protected]20c52d22011-06-20 22:42:42332 }
[email protected]57c8cf22013-03-02 16:50:00333 case content::NOTIFICATION_LOAD_STOP: {
334 // Take the opportunity to set some (minimal) additional
335 // script permissions required for the web UI.
336 GURL url = web_ui()->GetWebContents()->GetURL();
337 GURL dialog_url = CloudPrintURL(
338 Profile::FromWebUI(web_ui())).GetCloudPrintServiceDialogURL();
339 if (url.host() == dialog_url.host() &&
340 url.path() == dialog_url.path() &&
341 url.scheme() == dialog_url.scheme()) {
342 RenderViewHost* rvh = web_ui()->GetWebContents()->GetRenderViewHost();
343 if (rvh) {
[email protected]3184f90b2013-05-01 18:17:53344 WebPreferences webkit_prefs = rvh->GetWebkitPreferences();
[email protected]57c8cf22013-03-02 16:50:00345 webkit_prefs.allow_scripts_to_close_windows = true;
346 rvh->UpdateWebkitPreferences(webkit_prefs);
347 } else {
348 NOTREACHED();
349 }
350 }
[email protected]20c52d22011-06-20 22:42:42351
[email protected]57c8cf22013-03-02 16:50:00352 // Choose one or the other. If you need to debug, bring up the
353 // debugger. You can then use the various chrome.send()
354 // registrations above to kick of the various function calls,
355 // including chrome.send("SendPrintData") in the javaScript
356 // console and watch things happen with:
357 // HandleShowDebugger(NULL);
358 HandleSendPrintData(NULL);
359 break;
360 }
[email protected]73852b8f2010-05-14 00:38:12361 }
362}
363
[email protected]88942a22010-08-19 20:34:43364void CloudPrintFlowHandler::HandleShowDebugger(const ListValue* args) {
[email protected]73852b8f2010-05-14 00:38:12365 ShowDebugger();
366}
367
368void CloudPrintFlowHandler::ShowDebugger() {
[email protected]46adf7ff2011-12-30 00:53:09369 if (web_ui()) {
[email protected]01ec4ec2012-01-18 04:13:47370 RenderViewHost* rvh = web_ui()->GetWebContents()->GetRenderViewHost();
[email protected]73852b8f2010-05-14 00:38:12371 if (rvh)
[email protected]aebdd072011-07-07 12:36:59372 DevToolsWindow::OpenDevToolsWindow(rvh);
[email protected]73852b8f2010-05-14 00:38:12373 }
374}
375
376scoped_refptr<CloudPrintDataSender>
377CloudPrintFlowHandler::CreateCloudPrintDataSender() {
[email protected]46adf7ff2011-12-30 00:53:09378 DCHECK(web_ui());
379 print_data_helper_.reset(new CloudPrintDataSenderHelper(web_ui()));
[email protected]a9723e12013-03-05 04:02:45380 scoped_refptr<CloudPrintDataSender> sender(
381 new CloudPrintDataSender(print_data_helper_.get(), print_job_title_,
382 print_ticket_, file_type_, data_));
383 data_ = NULL;
384 return sender;
[email protected]73852b8f2010-05-14 00:38:12385}
386
[email protected]88942a22010-08-19 20:34:43387void CloudPrintFlowHandler::HandleSendPrintData(const ListValue* args) {
[email protected]ba4f1132010-10-09 02:02:35388 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
[email protected]73852b8f2010-05-14 00:38:12389 // This will cancel any ReadPrintDataFile() or SendPrintDataFile()
390 // requests in flight (this is anticipation of when setting page
391 // setup parameters becomes asynchronous and may be set while some
392 // data is in flight). Then we can clear out the print data.
393 CancelAnyRunningTask();
[email protected]46adf7ff2011-12-30 00:53:09394 if (web_ui()) {
[email protected]73852b8f2010-05-14 00:38:12395 print_data_sender_ = CreateCloudPrintDataSender();
[email protected]3e2dd4fa2011-11-10 06:06:40396 BrowserThread::PostTask(
[email protected]a9723e12013-03-05 04:02:45397 BrowserThread::IO, FROM_HERE,
398 base::Bind(&CloudPrintDataSender::SendPrintData, print_data_sender_));
[email protected]73852b8f2010-05-14 00:38:12399 }
400}
401
[email protected]88942a22010-08-19 20:34:43402void CloudPrintFlowHandler::HandleSetPageParameters(const ListValue* args) {
[email protected]036056a32011-03-03 21:05:01403 std::string json;
[email protected]e675f7b2011-06-22 17:32:12404 bool ret = args->GetString(0, &json);
405 if (!ret || json.empty()) {
[email protected]036056a32011-03-03 21:05:01406 NOTREACHED() << "Empty json string";
[email protected]73852b8f2010-05-14 00:38:12407 return;
[email protected]036056a32011-03-03 21:05:01408 }
[email protected]73852b8f2010-05-14 00:38:12409
410 // These are backstop default values - 72 dpi to match the screen,
411 // 8.5x11 inch paper with margins subtracted (1/4 inch top, left,
412 // right and 0.56 bottom), and the min page shrink and max page
413 // shrink values appear all over the place with no explanation.
414
415 // TODO(scottbyer): Get a Linux/ChromeOS edge for PrintSettings
416 // working so that we can get the default values from there. Fix up
417 // PrintWebViewHelper to do the same.
418 const int kDPI = 72;
419 const int kWidth = static_cast<int>((8.5-0.25-0.25)*kDPI);
420 const int kHeight = static_cast<int>((11-0.25-0.56)*kDPI);
421 const double kMinPageShrink = 1.25;
422 const double kMaxPageShrink = 2.0;
423
[email protected]1375e3ab2011-03-24 17:07:22424 PrintMsg_Print_Params default_settings;
[email protected]10980442011-12-04 22:33:05425 default_settings.content_size = gfx::Size(kWidth, kHeight);
[email protected]732b8132012-01-10 23:17:32426 default_settings.printable_area = gfx::Rect(0, 0, kWidth, kHeight);
[email protected]73852b8f2010-05-14 00:38:12427 default_settings.dpi = kDPI;
428 default_settings.min_shrink = kMinPageShrink;
429 default_settings.max_shrink = kMaxPageShrink;
430 default_settings.desired_dpi = kDPI;
431 default_settings.document_cookie = 0;
432 default_settings.selection_only = false;
[email protected]718af822011-08-12 22:11:33433 default_settings.preview_request_id = 0;
434 default_settings.is_first_request = true;
[email protected]732b8132012-01-10 23:17:32435 default_settings.print_to_pdf = false;
[email protected]73852b8f2010-05-14 00:38:12436
437 if (!GetPageSetupParameters(json, default_settings)) {
438 NOTREACHED();
439 return;
440 }
441
442 // TODO(scottbyer) - Here is where we would kick the originating
443 // renderer thread with these new parameters in order to get it to
[email protected]a984bdf2011-03-15 20:17:16444 // re-generate the PDF data and hand it back to us. window.print() is
[email protected]73852b8f2010-05-14 00:38:12445 // currently synchronous, so there's a lot of work to do to get to
446 // that point.
447}
448
[email protected]ea161da2010-11-02 21:57:35449void CloudPrintFlowHandler::StoreDialogClientSize() const {
[email protected]01ec4ec2012-01-18 04:13:47450 if (web_ui() && web_ui()->GetWebContents() &&
451 web_ui()->GetWebContents()->GetView()) {
452 gfx::Size size = web_ui()->GetWebContents()->GetView()->GetContainerSize();
[email protected]46adf7ff2011-12-30 00:53:09453 Profile* profile = Profile::FromWebUI(web_ui());
[email protected]0eb25c42011-08-11 14:50:14454 profile->GetPrefs()->SetInteger(prefs::kCloudPrintDialogWidth,
455 size.width());
456 profile->GetPrefs()->SetInteger(prefs::kCloudPrintDialogHeight,
457 size.height());
[email protected]ea161da2010-11-02 21:57:35458 }
459}
460
[email protected]a911c4e2012-10-12 00:37:57461bool CloudPrintFlowHandler::NavigationToURLDidCloseDialog(const GURL& url) {
462 if (close_after_signin_) {
463 GURL dialog_url = CloudPrintURL(
464 Profile::FromWebUI(web_ui())).GetCloudPrintServiceURL();
465
466 if (url.host() == dialog_url.host() &&
[email protected]22f450492013-02-20 22:02:51467 StartsWithASCII(url.path(), dialog_url.path(), false) &&
[email protected]a911c4e2012-10-12 00:37:57468 url.scheme() == dialog_url.scheme()) {
469 StoreDialogClientSize();
470 web_ui()->GetWebContents()->GetRenderViewHost()->ClosePage();
471 callback_.Run();
472 return true;
473 }
474 }
475 return false;
476}
477
[email protected]5835871a2012-04-25 21:56:55478CloudPrintWebDialogDelegate::CloudPrintWebDialogDelegate(
[email protected]b5b79d72012-05-24 19:42:28479 content::BrowserContext* browser_context,
480 gfx::NativeWindow modal_parent,
[email protected]a9723e12013-03-05 04:02:45481 const base::RefCountedMemory* data,
[email protected]9848c7e2010-06-03 16:06:56482 const std::string& json_arguments,
[email protected]e39027a2011-01-24 21:41:54483 const string16& print_job_title,
[email protected]e8368e92011-08-20 04:05:56484 const string16& print_ticket,
[email protected]a984bdf2011-03-15 20:17:16485 const std::string& file_type,
[email protected]4cd49022012-01-19 20:37:37486 bool close_after_signin,
487 const base::Closure& callback)
[email protected]a9723e12013-03-05 04:02:45488 : flow_handler_(
489 new CloudPrintFlowHandler(data, print_job_title, print_ticket,
490 file_type, close_after_signin, callback)),
[email protected]b5b79d72012-05-24 19:42:28491 modal_parent_(modal_parent),
[email protected]6ddda232011-04-22 15:41:47492 owns_flow_handler_(true),
[email protected]b5b79d72012-05-24 19:42:28493 keep_alive_when_non_modal_(true) {
494 Init(browser_context, json_arguments);
[email protected]73852b8f2010-05-14 00:38:12495}
496
[email protected]05acb55472011-02-03 00:11:07497// For unit testing.
[email protected]5835871a2012-04-25 21:56:55498CloudPrintWebDialogDelegate::CloudPrintWebDialogDelegate(
[email protected]73852b8f2010-05-14 00:38:12499 CloudPrintFlowHandler* flow_handler,
[email protected]a9723e12013-03-05 04:02:45500 const std::string& json_arguments)
501 : flow_handler_(flow_handler),
[email protected]b5b79d72012-05-24 19:42:28502 modal_parent_(NULL),
503 owns_flow_handler_(true),
504 keep_alive_when_non_modal_(false) {
505 Init(NULL, json_arguments);
[email protected]73852b8f2010-05-14 00:38:12506}
507
[email protected]b5b79d72012-05-24 19:42:28508// Returns the persisted width/height for the print dialog.
509void GetDialogWidthAndHeightFromPrefs(content::BrowserContext* browser_context,
510 int* width,
511 int* height) {
[email protected]b5b79d72012-05-24 19:42:28512 if (!browser_context) {
513 *width = kDefaultWidth;
514 *height = kDefaultHeight;
515 return;
516 }
517
[email protected]c753f142013-02-10 13:14:04518 PrefService* prefs = Profile::FromBrowserContext(browser_context)->GetPrefs();
[email protected]c753f142013-02-10 13:14:04519 *width = prefs->GetInteger(prefs::kCloudPrintDialogWidth);
520 *height = prefs->GetInteger(prefs::kCloudPrintDialogHeight);
[email protected]b5b79d72012-05-24 19:42:28521}
522
523void CloudPrintWebDialogDelegate::Init(content::BrowserContext* browser_context,
[email protected]5835871a2012-04-25 21:56:55524 const std::string& json_arguments) {
[email protected]73852b8f2010-05-14 00:38:12525 // This information is needed to show the dialog HTML content.
[email protected]ba4f1132010-10-09 02:02:35526 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
[email protected]b5b79d72012-05-24 19:42:28527
[email protected]89f550b2011-06-08 18:34:03528 params_.url = GURL(chrome::kChromeUICloudPrintResourcesURL);
[email protected]b5b79d72012-05-24 19:42:28529 GetDialogWidthAndHeightFromPrefs(browser_context,
530 &params_.width,
531 &params_.height);
[email protected]73852b8f2010-05-14 00:38:12532 params_.json_input = json_arguments;
533
534 flow_handler_->SetDialogDelegate(this);
[email protected]e39027a2011-01-24 21:41:54535 // If we're not modal we can show the dialog with no browser.
536 // We need this to keep Chrome alive while our dialog is up.
[email protected]b5b79d72012-05-24 19:42:28537 if (!modal_parent_ && keep_alive_when_non_modal_)
[email protected]313fce12013-01-30 17:09:04538 chrome::StartKeepAlive();
[email protected]73852b8f2010-05-14 00:38:12539}
540
[email protected]5835871a2012-04-25 21:56:55541CloudPrintWebDialogDelegate::~CloudPrintWebDialogDelegate() {
[email protected]73852b8f2010-05-14 00:38:12542 // If the flow_handler_ is about to outlive us because we don't own
[email protected]a2c92a1c2012-04-03 12:32:14543 // it anymore, we need to have it remove its reference to us.
[email protected]ba4f1132010-10-09 02:02:35544 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
[email protected]73852b8f2010-05-14 00:38:12545 flow_handler_->SetDialogDelegate(NULL);
546 if (owns_flow_handler_) {
547 delete flow_handler_;
548 }
549}
550
[email protected]5835871a2012-04-25 21:56:55551ui::ModalType CloudPrintWebDialogDelegate::GetDialogModalType() const {
[email protected]b5b79d72012-05-24 19:42:28552 return modal_parent_ ? ui::MODAL_TYPE_WINDOW : ui::MODAL_TYPE_NONE;
[email protected]73852b8f2010-05-14 00:38:12553}
554
[email protected]5835871a2012-04-25 21:56:55555string16 CloudPrintWebDialogDelegate::GetDialogTitle() const {
[email protected]bdae5c12011-08-05 21:49:06556 return string16();
[email protected]73852b8f2010-05-14 00:38:12557}
558
[email protected]5835871a2012-04-25 21:56:55559GURL CloudPrintWebDialogDelegate::GetDialogContentURL() const {
[email protected]73852b8f2010-05-14 00:38:12560 return params_.url;
561}
562
[email protected]5835871a2012-04-25 21:56:55563void CloudPrintWebDialogDelegate::GetWebUIMessageHandlers(
[email protected]36e12172011-02-08 23:46:02564 std::vector<WebUIMessageHandler*>* handlers) const {
[email protected]73852b8f2010-05-14 00:38:12565 handlers->push_back(flow_handler_);
566 // We don't own flow_handler_ anymore, but it sticks around until at
567 // least right after OnDialogClosed() is called (and this object is
568 // destroyed).
569 owns_flow_handler_ = false;
570}
571
[email protected]5835871a2012-04-25 21:56:55572void CloudPrintWebDialogDelegate::GetDialogSize(gfx::Size* size) const {
[email protected]73852b8f2010-05-14 00:38:12573 size->set_width(params_.width);
574 size->set_height(params_.height);
575}
576
[email protected]5835871a2012-04-25 21:56:55577std::string CloudPrintWebDialogDelegate::GetDialogArgs() const {
[email protected]73852b8f2010-05-14 00:38:12578 return params_.json_input;
579}
580
[email protected]5835871a2012-04-25 21:56:55581void CloudPrintWebDialogDelegate::OnDialogClosed(
[email protected]73852b8f2010-05-14 00:38:12582 const std::string& json_retval) {
[email protected]ea161da2010-11-02 21:57:35583 // Get the final dialog size and store it.
584 flow_handler_->StoreDialogClientSize();
[email protected]6ddda232011-04-22 15:41:47585
[email protected]e39027a2011-01-24 21:41:54586 // If we're modal we can show the dialog with no browser.
587 // End the keep-alive so that Chrome can exit.
[email protected]b5b79d72012-05-24 19:42:28588 if (!modal_parent_ && keep_alive_when_non_modal_)
[email protected]313fce12013-01-30 17:09:04589 chrome::EndKeepAlive();
[email protected]73852b8f2010-05-14 00:38:12590 delete this;
591}
592
[email protected]5835871a2012-04-25 21:56:55593void CloudPrintWebDialogDelegate::OnCloseContents(WebContents* source,
594 bool* out_close_dialog) {
[email protected]18137e02010-05-25 21:10:35595 if (out_close_dialog)
596 *out_close_dialog = true;
597}
598
[email protected]5835871a2012-04-25 21:56:55599bool CloudPrintWebDialogDelegate::ShouldShowDialogTitle() const {
[email protected]ea161da2010-11-02 21:57:35600 return false;
601}
602
[email protected]5835871a2012-04-25 21:56:55603bool CloudPrintWebDialogDelegate::HandleContextMenu(
[email protected]35be7ec2012-02-12 20:42:51604 const content::ContextMenuParams& params) {
[email protected]34478212011-04-19 01:35:46605 return true;
606}
607
[email protected]a911c4e2012-10-12 00:37:57608bool CloudPrintWebDialogDelegate::HandleOpenURLFromTab(
609 content::WebContents* source,
610 const content::OpenURLParams& params,
611 content::WebContents** out_new_contents) {
612 return flow_handler_->NavigationToURLDidCloseDialog(params.url);
613}
614
[email protected]6085c70d2011-03-22 22:51:07615// Called from the UI thread, starts up the dialog.
[email protected]b5b79d72012-05-24 19:42:28616void CreateDialogImpl(content::BrowserContext* browser_context,
617 gfx::NativeWindow modal_parent,
[email protected]a9723e12013-03-05 04:02:45618 const base::RefCountedMemory* data,
[email protected]6085c70d2011-03-22 22:51:07619 const string16& print_job_title,
[email protected]e8368e92011-08-20 04:05:56620 const string16& print_ticket,
[email protected]6085c70d2011-03-22 22:51:07621 const std::string& file_type,
[email protected]4cd49022012-01-19 20:37:37622 bool close_after_signin,
623 const base::Closure& callback) {
[email protected]ba4f1132010-10-09 02:02:35624 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
[email protected]5835871a2012-04-25 21:56:55625 WebDialogDelegate* dialog_delegate =
626 new internal_cloud_print_helpers::CloudPrintWebDialogDelegate(
[email protected]a9723e12013-03-05 04:02:45627 browser_context, modal_parent, data, std::string(), print_job_title,
628 print_ticket, file_type, close_after_signin, callback);
[email protected]88bfd25b2012-06-22 06:28:33629#if defined(OS_WIN)
630 gfx::NativeWindow window =
631#endif
[email protected]87586fb72012-07-02 13:15:02632 chrome::ShowWebDialog(modal_parent,
633 Profile::FromBrowserContext(browser_context),
634 dialog_delegate);
[email protected]88bfd25b2012-06-22 06:28:33635#if defined(OS_WIN)
[email protected]a9723e12013-03-05 04:02:45636 if (window) {
[email protected]88bfd25b2012-06-22 06:28:33637 HWND dialog_handle;
638#if defined(USE_AURA)
639 dialog_handle = window->GetRootWindow()->GetAcceleratedWidget();
640#else
641 dialog_handle = window;
642#endif
643 if (::GetForegroundWindow() != dialog_handle) {
644 ui::ForegroundHelper::SetForeground(dialog_handle);
645 }
646 }
647#endif
[email protected]73852b8f2010-05-14 00:38:12648}
[email protected]6085c70d2011-03-22 22:51:07649
[email protected]b5b79d72012-05-24 19:42:28650void CreateDialogSigninImpl(content::BrowserContext* browser_context,
651 gfx::NativeWindow modal_parent,
652 const base::Closure& callback) {
[email protected]a9723e12013-03-05 04:02:45653 CreateDialogImpl(browser_context, modal_parent, NULL, string16(),
654 string16(), std::string(), true, callback);
[email protected]4cd49022012-01-19 20:37:37655}
656
[email protected]a9723e12013-03-05 04:02:45657void CreateDialogForFileImpl(content::BrowserContext* browser_context,
658 gfx::NativeWindow modal_parent,
659 const base::FilePath& path_to_file,
660 const string16& print_job_title,
661 const string16& print_ticket,
662 const std::string& file_type,
663 bool delete_on_close) {
664 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
665 scoped_refptr<base::RefCountedMemory> data;
666 int64 file_size = 0;
667 if (file_util::GetFileSize(path_to_file, &file_size) && file_size != 0) {
668 std::string file_data;
669 if (file_size < kuint32max) {
670 file_data.reserve(static_cast<unsigned int>(file_size));
671 } else {
672 DLOG(WARNING) << " print data file too large to reserve space";
673 }
674 if (file_util::ReadFileToString(path_to_file, &file_data)) {
675 data = base::RefCountedString::TakeString(&file_data);
676 }
677 }
678 // Proceed even for empty data to simplify testing.
679 BrowserThread::PostTask(
680 BrowserThread::UI, FROM_HERE,
681 base::Bind(&print_dialog_cloud::CreatePrintDialogForBytes,
682 browser_context, modal_parent, data, print_job_title,
683 print_ticket, file_type));
684 if (delete_on_close)
685 file_util::Delete(path_to_file, false);
[email protected]6ddda232011-04-22 15:41:47686}
687
[email protected]6085c70d2011-03-22 22:51:07688} // namespace internal_cloud_print_helpers
689
690namespace print_dialog_cloud {
691
[email protected]681958c2013-02-21 13:48:14692void RegisterUserPrefs(PrefRegistrySyncable* registry) {
693 registry->RegisterIntegerPref(
694 prefs::kCloudPrintDialogWidth,
695 kDefaultWidth,
696 PrefRegistrySyncable::UNSYNCABLE_PREF);
697 registry->RegisterIntegerPref(
698 prefs::kCloudPrintDialogHeight,
699 kDefaultHeight,
700 PrefRegistrySyncable::UNSYNCABLE_PREF);
701}
702
[email protected]6085c70d2011-03-22 22:51:07703// Called on the FILE or UI thread. This is the main entry point into creating
704// the dialog.
705
[email protected]b5b79d72012-05-24 19:42:28706void CreatePrintDialogForFile(content::BrowserContext* browser_context,
707 gfx::NativeWindow modal_parent,
[email protected]650b2d52013-02-10 03:41:45708 const base::FilePath& path_to_file,
[email protected]6085c70d2011-03-22 22:51:07709 const string16& print_job_title,
[email protected]e8368e92011-08-20 04:05:56710 const string16& print_ticket,
[email protected]6085c70d2011-03-22 22:51:07711 const std::string& file_type,
[email protected]d955fc92011-09-19 20:49:03712 bool delete_on_close) {
[email protected]6085c70d2011-03-22 22:51:07713 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE) ||
714 BrowserThread::CurrentlyOn(BrowserThread::UI));
[email protected]6085c70d2011-03-22 22:51:07715 BrowserThread::PostTask(
[email protected]a9723e12013-03-05 04:02:45716 BrowserThread::FILE, FROM_HERE,
717 base::Bind(&internal_cloud_print_helpers::CreateDialogForFileImpl,
[email protected]b5b79d72012-05-24 19:42:28718 browser_context, modal_parent, path_to_file, print_job_title,
719 print_ticket, file_type, delete_on_close));
[email protected]d955fc92011-09-19 20:49:03720}
721
[email protected]b5b79d72012-05-24 19:42:28722void CreateCloudPrintSigninDialog(content::BrowserContext* browser_context,
723 gfx::NativeWindow modal_parent,
724 const base::Closure& callback) {
[email protected]4cd49022012-01-19 20:37:37725 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
726
727 BrowserThread::PostTask(
728 BrowserThread::UI, FROM_HERE,
729 base::Bind(&internal_cloud_print_helpers::CreateDialogSigninImpl,
[email protected]a9723e12013-03-05 04:02:45730 browser_context, modal_parent, callback));
[email protected]4cd49022012-01-19 20:37:37731}
732
[email protected]b5b79d72012-05-24 19:42:28733void CreatePrintDialogForBytes(content::BrowserContext* browser_context,
734 gfx::NativeWindow modal_parent,
[email protected]a9723e12013-03-05 04:02:45735 const base::RefCountedMemory* data,
[email protected]d955fc92011-09-19 20:49:03736 const string16& print_job_title,
737 const string16& print_ticket,
[email protected]b5b79d72012-05-24 19:42:28738 const std::string& file_type) {
[email protected]a9723e12013-03-05 04:02:45739 internal_cloud_print_helpers::CreateDialogImpl(browser_context, modal_parent,
740 data, print_job_title,
741 print_ticket, file_type, false,
742 base::Closure());
[email protected]6085c70d2011-03-22 22:51:07743}
744
[email protected]65c9d89a2011-04-13 21:02:39745bool CreatePrintDialogFromCommandLine(const CommandLine& command_line) {
[email protected]87ab41e72012-01-04 18:45:11746 DCHECK(command_line.HasSwitch(switches::kCloudPrintFile));
[email protected]65c9d89a2011-04-13 21:02:39747 if (!command_line.GetSwitchValuePath(switches::kCloudPrintFile).empty()) {
[email protected]650b2d52013-02-10 03:41:45748 base::FilePath cloud_print_file;
[email protected]65c9d89a2011-04-13 21:02:39749 cloud_print_file =
750 command_line.GetSwitchValuePath(switches::kCloudPrintFile);
751 if (!cloud_print_file.empty()) {
752 string16 print_job_title;
[email protected]e8368e92011-08-20 04:05:56753 string16 print_job_print_ticket;
[email protected]65c9d89a2011-04-13 21:02:39754 if (command_line.HasSwitch(switches::kCloudPrintJobTitle)) {
[email protected]e8368e92011-08-20 04:05:56755 print_job_title =
756 internal_cloud_print_helpers::GetSwitchValueString16(
757 command_line, switches::kCloudPrintJobTitle);
758 }
759 if (command_line.HasSwitch(switches::kCloudPrintPrintTicket)) {
760 print_job_print_ticket =
761 internal_cloud_print_helpers::GetSwitchValueString16(
762 command_line, switches::kCloudPrintPrintTicket);
[email protected]65c9d89a2011-04-13 21:02:39763 }
764 std::string file_type = "application/pdf";
765 if (command_line.HasSwitch(switches::kCloudPrintFileType)) {
766 file_type = command_line.GetSwitchValueASCII(
767 switches::kCloudPrintFileType);
768 }
[email protected]e8368e92011-08-20 04:05:56769
[email protected]d955fc92011-09-19 20:49:03770 bool delete_on_close = CommandLine::ForCurrentProcess()->HasSwitch(
771 switches::kCloudPrintDeleteFile);
772
[email protected]b5b79d72012-05-24 19:42:28773 print_dialog_cloud::CreatePrintDialogForFile(
774 ProfileManager::GetDefaultProfile(),
775 NULL,
776 cloud_print_file,
777 print_job_title,
778 print_job_print_ticket,
779 file_type,
780 delete_on_close);
[email protected]65c9d89a2011-04-13 21:02:39781 return true;
782 }
783 }
784 return false;
785}
786
[email protected]6085c70d2011-03-22 22:51:07787} // end namespace