diff options
Diffstat (limited to 'win8/metro_driver/print_document_source.h')
-rw-r--r-- | win8/metro_driver/print_document_source.h | 164 |
1 files changed, 164 insertions, 0 deletions
diff --git a/win8/metro_driver/print_document_source.h b/win8/metro_driver/print_document_source.h new file mode 100644 index 0000000..fed333e --- /dev/null +++ b/win8/metro_driver/print_document_source.h @@ -0,0 +1,164 @@ +// 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_UI_METRO_DRIVER_PRINT_DOCUMENT_SOURCE_H_ +#define CHROME_BROWSER_UI_METRO_DRIVER_PRINT_DOCUMENT_SOURCE_H_ + +#include <documentsource.h> +#include <printpreview.h> +#include <windows.graphics.printing.h> + +#include <vector> + +#include "base/basictypes.h" +#include "base/memory/ref_counted.h" +#include "base/memory/scoped_ptr.h" +#include "base/synchronization/condition_variable.h" +#include "base/synchronization/waitable_event.h" +#include "win8/metro_driver/winrt_utils.h" + +// Hack to be removed once we don't need to build with an SDK earlier than +// 8441 where the name of the interface has been changed. +// TODO(mad): remove once we don't run mixed SDK/OS anymore. +#ifndef __IPrintPreviewDxgiPackageTarget_FWD_DEFINED__ +typedef IPrintPreviewDXGIPackageTarget IPrintPreviewDxgiPackageTarget; +#endif + + +namespace base { +class Lock; +}; // namespace base + +namespace metro_driver { + +// This class is given to Metro as a source for print documents. +// The methodless IPrintDocumentSource interface is used to identify it as such. +// Then, the other interfaces are used to generate preview and print documents. +// It also exposes a few methods for the print handler to control the document. +class PrintDocumentSource + : public mswr::RuntimeClass< + mswr::RuntimeClassFlags<mswr::WinRtClassicComMix>, + wingfx::Printing::IPrintDocumentSource, + IPrintDocumentPageSource, + IPrintPreviewPageCollection> { + public: + // A set of interfaces for the DirectX context that our parent owns + // and that don't need to change from document to document. + struct DirectXContext { + DirectXContext() {} + DirectXContext(ID3D11Device1* device_3d, + ID2D1Factory1* factory_2d, + ID2D1Device* device_2d, + ID2D1DeviceContext* context_2d, + IWICImagingFactory2* factory_wic) + : d3d_device(device_3d), + d2d_factory(factory_2d), + d2d_device(device_2d), + d2d_context(context_2d), + wic_factory(factory_wic) { + } + DirectXContext(const DirectXContext& other) + : d3d_device(other.d3d_device), + d2d_factory(other.d2d_factory), + d2d_device(other.d2d_device), + d2d_context(other.d2d_context), + wic_factory(other.wic_factory) { + } + mswr::ComPtr<ID3D11Device1> d3d_device; + mswr::ComPtr<ID2D1Factory1> d2d_factory; + mswr::ComPtr<ID2D1Device> d2d_device; + mswr::ComPtr<ID2D1DeviceContext> d2d_context; + mswr::ComPtr<IWICImagingFactory2> wic_factory; + }; + + // Construction / Initialization. + explicit PrintDocumentSource(); + HRESULT RuntimeClassInitialize(const DirectXContext& directx_context, + base::Lock* parent_lock); + // Aborts any pending asynchronous operation. + void Abort(); + + // classic COM interface IPrintDocumentPageSource methods + STDMETHOD(GetPreviewPageCollection) ( + IPrintDocumentPackageTarget* package_target, + IPrintPreviewPageCollection** page_collection); + STDMETHOD(MakeDocument)(IInspectable* options, + IPrintDocumentPackageTarget* package_target); + + // classic COM interface IPrintPreviewPageCollection methods + STDMETHOD(Paginate)(uint32 page, IInspectable* options); + STDMETHOD(MakePage)(uint32 desired_page, float width, float height); + + // If the screen DPI changes, we must be warned here. + void ResetDpi(float dpi); + + // When the page count is known, this is called and we can setup our data. + void SetPageCount(size_t page_count); + + // Every time a page is ready, this is called and we can read the data if + // we were waiting for it, or store it for later use. + void AddPage(size_t page_number, IStream* metafile_stream); + + private: + // Print the page given in the metafile stream to the given print control. + HRESULT PrintPage(ID2D1PrintControl* print_control, + ID2D1GdiMetafile* metafile_stream, + D2D1_SIZE_F pageSize); + + // Helper function to wait for the page count to be ready. + // Returns 0 when aborted. + size_t WaitAndGetPageCount(); + + // Helper function to wait for a given page to be ready. + // Returns S_FALSE when aborted. + HRESULT WaitAndGetPage(size_t page_number, + ID2D1GdiMetafile** metafile_stream); + + DirectXContext directx_context_; + + // Once page data is available, it's added to this vector. + std::vector<mswr::ComPtr<IStream>> pages_; + + // When page count is set, the size of this vector is set to that number. + // Then, every time page data is added to pages_, the associated condition + // variable in this vector is signaled. This is only filled when we receive + // the page count, so we must wait on page_count_ready_ before accessing + // the content of this vector. + std::vector<scoped_ptr<base::ConditionVariable> > pages_ready_state_; + + // This event is signaled when we receive a page count from Chrome. We should + // not receive any page data before the count, so we can check this event + // while waiting for pages too, in case we ask for page data before we got + // the count, so before we properly initialized pages_ready_state_. + base::WaitableEvent page_count_ready_; + + // The preview target interface set from within GetPreviewPageCollection + // but used from within MakePage. + mswr::ComPtr<IPrintPreviewDxgiPackageTarget> dxgi_preview_target_; + + // Our parent's lock (to make sure it is initialized and destroyed early + // enough), which we use to protect access to our data members. + base::Lock* parent_lock_; + + // The width/height requested in Paginate and used in MakePage. + // TODO(mad): Height is currently not used, and width is only used for + // scaling. We need to add a way to specify width and height when we request + // pages from Chrome. + float height_; + float width_; + + // The DPI is set by Windows and we need to give it to DirectX. + float dpi_; + + // A flag identiying that we have been aborted. Needed to properly handle + // asynchronous callbacks. + bool aborted_; + + // TODO(mad): remove once we don't run mixed SDK/OS anymore. + bool using_old_preview_interface_; +}; + +} // namespace metro_driver + +#endif // CHROME_BROWSER_UI_METRO_DRIVER_PRINT_DOCUMENT_SOURCE_H_ |