// Copyright (c) 2012 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. #ifndef CHROME_BROWSER_PRINTING_PRINT_JOB_H_ #define CHROME_BROWSER_PRINTING_PRINT_JOB_H_ #include <memory> #include <vector> #include "base/macros.h" #include "base/memory/weak_ptr.h" #include "build/build_config.h" #include "chrome/browser/printing/print_job_worker_owner.h" #include "content/public/browser/notification_observer.h" #include "content/public/browser/notification_registrar.h" namespace base { class RefCountedMemory; } namespace printing { class JobEventDetails; class MetafilePlayer; class PdfToEmfConverter; class PrintJobWorker; class PrintedDocument; class PrintedPage; class PrintedPagesSource; class PrinterQuery; // Manages the print work for a specific document. Talks to the printer through // PrintingContext through PrintJobWorker. Hides access to PrintingContext in a // worker thread so the caller never blocks. PrintJob will send notifications on // any state change. While printing, the PrintJobManager instance keeps a // reference to the job to be sure it is kept alive. All the code in this class // runs in the UI thread. class PrintJob : public PrintJobWorkerOwner, public content::NotificationObserver { public: // Create a empty PrintJob. When initializing with this constructor, // post-constructor initialization must be done with Initialize(). PrintJob(); // Grabs the ownership of the PrintJobWorker from another job, which is // usually a PrinterQuery. Set the expected page count of the print job. void Initialize(PrintJobWorkerOwner* job, PrintedPagesSource* source, int page_count); // content::NotificationObserver implementation. void Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) override; // PrintJobWorkerOwner implementation. void GetSettingsDone(const PrintSettings& new_settings, PrintingContext::Result result) override; std::unique_ptr<PrintJobWorker> DetachWorker( PrintJobWorkerOwner* new_owner) override; const PrintSettings& settings() const override; int cookie() const override; // Starts the actual printing. Signals the worker that it should begin to // spool as soon as data is available. void StartPrinting(); // Asks for the worker thread to finish its queued tasks and disconnects the // delegate object. The PrintJobManager will remove its reference. This may // have the side-effect of destroying the object if the caller doesn't have a // handle to the object. Use PrintJob::is_stopped() to check whether the // worker thread has actually stopped. void Stop(); // Cancels printing job and stops the worker thread. Takes effect immediately. void Cancel(); // Synchronously wait for the job to finish. It is mainly useful when the // process is about to be shut down and we're waiting for the spooler to eat // our data. bool FlushJob(base::TimeDelta timeout); // Disconnects the PrintedPage source (PrintedPagesSource). It is done when // the source is being destroyed. void DisconnectSource(); // Returns true if the print job is pending, i.e. between a StartPrinting() // and the end of the spooling. bool is_job_pending() const; // Access the current printed document. Warning: may be NULL. PrintedDocument* document() const; #if defined(OS_WIN) // Let the PrintJob know the 0-based |page_number| of a given printed page. void AppendPrintedPage(int page_number); void StartPdfToEmfConversion( const scoped_refptr<base::RefCountedMemory>& bytes, const gfx::Size& page_size, const gfx::Rect& content_area, bool print_text_with_gdi); void StartPdfToPostScriptConversion( const scoped_refptr<base::RefCountedMemory>& bytes, const gfx::Rect& content_area, const gfx::Point& physical_offset, bool ps_level2); #endif // defined(OS_WIN) protected: ~PrintJob() override; private: // Updates |document_| to a new instance. void UpdatePrintedDocument(PrintedDocument* new_document); // Processes a NOTIFY_PRINT_JOB_EVENT notification. void OnNotifyPrintJobEvent(const JobEventDetails& event_details); // Releases the worker thread by calling Stop(), then broadcasts a JOB_DONE // notification. void OnDocumentDone(); // Terminates the worker thread in a very controlled way, to work around any // eventual deadlock. void ControlledWorkerShutdown(); // Called at shutdown when running a nested message loop. void Quit(); void HoldUntilStopIsCalled(); #if defined(OS_WIN) void OnPdfConversionStarted(int page_count); void OnPdfPageConverted(int page_number, float scale_factor, std::unique_ptr<MetafilePlayer> emf); #endif // defined(OS_WIN) content::NotificationRegistrar registrar_; // Source that generates the PrintedPage's (i.e. a WebContents). It will be // set back to NULL if the source is deleted before this object. PrintedPagesSource* source_; // All the UI is done in a worker thread because many Win32 print functions // are blocking and enters a message loop without your consent. There is one // worker thread per print job. std::unique_ptr<PrintJobWorker> worker_; // Cache of the print context settings for access in the UI thread. PrintSettings settings_; // The printed document. scoped_refptr<PrintedDocument> document_; // Is the worker thread printing. bool is_job_pending_; // Is Canceling? If so, try to not cause recursion if on FAILED notification, // the notified calls Cancel() again. bool is_canceling_; #if defined(OS_WIN) class PdfConversionState; std::unique_ptr<PdfConversionState> pdf_conversion_state_; std::vector<int> pdf_page_mapping_; #endif // defined(OS_WIN) // Used at shutdown so that we can quit a nested message loop. base::WeakPtrFactory<PrintJob> quit_factory_; DISALLOW_COPY_AND_ASSIGN(PrintJob); }; // Details for a NOTIFY_PRINT_JOB_EVENT notification. The members may be NULL. class JobEventDetails : public base::RefCountedThreadSafe<JobEventDetails> { public: // Event type. enum Type { // Print... dialog box has been closed with OK button. USER_INIT_DONE, // Print... dialog box has been closed with CANCEL button. USER_INIT_CANCELED, // An automated initialization has been done, e.g. Init(false, NULL). DEFAULT_INIT_DONE, // A new document started printing. NEW_DOC, // A new page started printing. NEW_PAGE, // A page is done printing. PAGE_DONE, // A document is done printing. The worker thread is still alive. Warning: // not a good moment to release the handle to PrintJob. DOC_DONE, // The worker thread is finished. A good moment to release the handle to // PrintJob. JOB_DONE, // All missing pages have been requested. ALL_PAGES_REQUESTED, // An error occured. Printing is canceled. FAILED, }; JobEventDetails(Type type, PrintedDocument* document, PrintedPage* page); // Getters. PrintedDocument* document() const; PrintedPage* page() const; Type type() const { return type_; } private: friend class base::RefCountedThreadSafe<JobEventDetails>; ~JobEventDetails(); scoped_refptr<PrintedDocument> document_; scoped_refptr<PrintedPage> page_; const Type type_; DISALLOW_COPY_AND_ASSIGN(JobEventDetails); }; } // namespace printing #endif // CHROME_BROWSER_PRINTING_PRINT_JOB_H_