| Index: win8/metro_driver/print_document_source.h | 
| diff --git a/win8/metro_driver/print_document_source.h b/win8/metro_driver/print_document_source.h | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..fed333e3706727e17dd5a51e5b85c24879fcf3be | 
| --- /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_ | 
|  |