summaryrefslogtreecommitdiffstats
path: root/chrome/browser/printing
diff options
context:
space:
mode:
Diffstat (limited to 'chrome/browser/printing')
-rw-r--r--chrome/browser/printing/print_job.cc12
-rw-r--r--chrome/browser/printing/print_job.h9
-rw-r--r--chrome/browser/printing/print_job_unittest.cc117
3 files changed, 136 insertions, 2 deletions
diff --git a/chrome/browser/printing/print_job.cc b/chrome/browser/printing/print_job.cc
index a2d07fd..14a487cd 100644
--- a/chrome/browser/printing/print_job.cc
+++ b/chrome/browser/printing/print_job.cc
@@ -24,6 +24,8 @@ PrintJob::PrintJob(PrintedPagesSource* source)
is_job_pending_(false),
is_print_dialog_box_shown_(false),
is_canceling_(false) {
+ DCHECK(ui_message_loop_);
+ ui_message_loop_->AddDestructionObserver(this);
}
PrintJob::PrintJob()
@@ -34,14 +36,18 @@ PrintJob::PrintJob()
is_job_pending_(false),
is_print_dialog_box_shown_(false),
is_canceling_(false) {
+ DCHECK(ui_message_loop_);
+ ui_message_loop_->AddDestructionObserver(this);
}
PrintJob::~PrintJob() {
+ ui_message_loop_->RemoveDestructionObserver(this);
// The job should be finished (or at least canceled) when it is destroyed.
DCHECK(!is_job_pending_);
DCHECK(!is_print_dialog_box_shown_);
DCHECK(!is_canceling_);
- DCHECK(worker_->message_loop() == NULL);
+ if (worker_.get())
+ DCHECK(worker_->message_loop() == NULL);
DCHECK_EQ(ui_message_loop_, MessageLoop::current());
}
@@ -142,6 +148,10 @@ int PrintJob::cookie() const {
return document_->cookie();
}
+void PrintJob::WillDestroyCurrentMessageLoop() {
+ NOTREACHED();
+}
+
void PrintJob::GetSettings(GetSettingsAskParam ask_user_for_settings,
HWND parent_window) {
DCHECK_EQ(ui_message_loop_, MessageLoop::current());
diff --git a/chrome/browser/printing/print_job.h b/chrome/browser/printing/print_job.h
index e18d52c..4c334b9 100644
--- a/chrome/browser/printing/print_job.h
+++ b/chrome/browser/printing/print_job.h
@@ -5,6 +5,7 @@
#ifndef CHROME_BROWSER_PRINTING_PRINT_JOB_H_
#define CHROME_BROWSER_PRINTING_PRINT_JOB_H_
+#include "base/message_loop.h"
#include "base/ref_counted.h"
#include "chrome/browser/printing/print_job_worker_owner.h"
#include "chrome/common/notification_service.h"
@@ -32,7 +33,8 @@ class PrinterQuery;
// runs in the UI thread.
class PrintJob : public base::RefCountedThreadSafe<PrintJob>,
public NotificationObserver,
- public PrintJobWorkerOwner {
+ public PrintJobWorkerOwner,
+ public MessageLoop::DestructionObserver {
public:
// GetSettings() UI parameter.
enum GetSettingsAskParam {
@@ -48,6 +50,8 @@ class PrintJob : public base::RefCountedThreadSafe<PrintJob>,
PrintJob();
virtual ~PrintJob();
+ // Grabs the ownership of the PrintJobWorker from another job, which is
+ // usually a PrinterQuery.
void Initialize(PrintJobWorkerOwner* job, PrintedPagesSource* source);
// NotificationObserver
@@ -69,6 +73,9 @@ class PrintJob : public base::RefCountedThreadSafe<PrintJob>,
virtual const PrintSettings& settings() const { return settings_; }
virtual int cookie() const;
+ // DestructionObserver
+ virtual void WillDestroyCurrentMessageLoop();
+
// Initializes the printing context. This can be done synchronously or not. It
// is fine to call this function multiple times to reinitialize the settings.
// |parent_window| parameter will be the owner of the print setting dialog
diff --git a/chrome/browser/printing/print_job_unittest.cc b/chrome/browser/printing/print_job_unittest.cc
new file mode 100644
index 0000000..6c65a5d
--- /dev/null
+++ b/chrome/browser/printing/print_job_unittest.cc
@@ -0,0 +1,117 @@
+// Copyright (c) 2008 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "base/message_loop.h"
+#include "chrome/browser/printing/print_job.h"
+#include "chrome/browser/printing/printed_pages_source.h"
+#include "googleurl/src/gurl.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+namespace {
+
+class TestSource : public printing::PrintedPagesSource {
+ public:
+ virtual void RenderOnePrintedPage(printing::PrintedDocument* document,
+ int page_number) {
+ }
+ virtual std::wstring RenderSourceName() {
+ return L"";
+ }
+ virtual GURL RenderSourceUrl() {
+ return GURL();
+ }
+};
+
+class TestPrintJob : public printing::PrintJob {
+ public:
+ TestPrintJob(printing::PrintedPagesSource* source, volatile bool* check)
+ : printing::PrintJob(source), check_(check) {
+ }
+ TestPrintJob(volatile bool* check) : check_(check) {
+ }
+ ~TestPrintJob() {
+ *check_ = true;
+ }
+ private:
+ volatile bool* check_;
+};
+
+class TestPrintNotifObserv : public NotificationObserver {
+ public:
+ // NotificationObserver
+ virtual void Observe(NotificationType type,
+ const NotificationSource& source,
+ const NotificationDetails& details) {
+ ASSERT_EQ(NOTIFY_PRINT_JOB_EVENT, type);
+ printing::JobEventDetails::Type event_type =
+ Details<printing::JobEventDetails>(details)->type();
+ EXPECT_NE(printing::JobEventDetails::NEW_DOC, event_type);
+ EXPECT_NE(printing::JobEventDetails::NEW_PAGE, event_type);
+ EXPECT_NE(printing::JobEventDetails::PAGE_DONE, event_type);
+ EXPECT_NE(printing::JobEventDetails::DOC_DONE, event_type);
+ EXPECT_NE(printing::JobEventDetails::JOB_DONE, event_type);
+ EXPECT_NE(printing::JobEventDetails::ALL_PAGES_REQUESTED, event_type);
+ if (event_type == printing::JobEventDetails::USER_INIT_DONE ||
+ event_type == printing::JobEventDetails::USER_INIT_CANCELED ||
+ event_type == printing::JobEventDetails::DEFAULT_INIT_DONE ||
+ event_type == printing::JobEventDetails::FAILED) {
+ MessageLoop::current()->Quit();
+ return;
+ }
+ }
+};
+
+} // namespace
+
+TEST(PrintJobTest, SimplePrint) {
+ // Test the multithreaded nature of PrintJob to make sure we can use it with
+ // known livetime.
+ TestPrintNotifObserv observ;
+ MessageLoop current;
+ NotificationService::current()->AddObserver(
+ &observ, NOTIFY_ALL,
+ NotificationService::AllSources());
+ TestSource source;
+ volatile bool check = false;
+ scoped_refptr<printing::PrintJob> job(new TestPrintJob(&source, &check));
+ job->GetSettings(printing::PrintJob::DEFAULTS, NULL);
+ EXPECT_EQ(MessageLoop::current(), job->message_loop());
+ current.Run();
+ job->Stop();
+ job = NULL;
+ EXPECT_TRUE(check);
+}
+
+TEST(PrintJobTest, SimplePrintLateInit) {
+ volatile bool check = false;
+ MessageLoop current;
+ scoped_refptr<printing::PrintJob> job(new TestPrintJob(&check));
+ job = NULL;
+ EXPECT_TRUE(check);
+ /* TODO(maruel): Test these.
+ job->Initialize()
+ job->Observe();
+ job->GetSettingsDone();
+ job->DetachWorker();
+ job->message_loop();
+ job->settings();
+ job->cookie();
+ job->GetSettings(printing::DEFAULTS, printing::ASK_USER, NULL);
+ job->StartPrinting();
+ job->Stop();
+ job->Cancel();
+ job->RequestMissingPages();
+ job->FlushJob(timeout_ms);
+ job->DisconnectSource();
+ job->is_job_pending();
+ job->is_print_dialog_box_shown();
+ job->document();
+ // Private
+ job->UpdatePrintedDocument(NULL);
+ scoped_refptr<printing::JobEventDetails> event_details;
+ job->OnNotifyPrintJobEvent(event_details);
+ job->OnDocumentDone();
+ job->ControlledWorkerShutdown();
+ */
+}