OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "printing/pdf_metafile_skia.h" | 5 #include "printing/pdf_metafile_skia.h" |
6 | 6 |
7 #include "base/containers/hash_tables.h" | 7 #include "base/containers/hash_tables.h" |
8 #include "base/files/file_util.h" | 8 #include "base/files/file_util.h" |
9 #include "base/metrics/histogram.h" | 9 #include "base/metrics/histogram.h" |
10 #include "base/numerics/safe_conversions.h" | 10 #include "base/numerics/safe_conversions.h" |
11 #include "base/posix/eintr_wrapper.h" | 11 #include "base/posix/eintr_wrapper.h" |
12 #include "skia/ext/refptr.h" | 12 #include "skia/ext/refptr.h" |
13 #include "skia/ext/vector_canvas.h" | 13 #include "skia/ext/vector_canvas.h" |
14 #include "third_party/skia/include/core/SkData.h" | 14 #include "third_party/skia/include/core/SkData.h" |
| 15 #include "third_party/skia/include/core/SkDocument.h" |
| 16 #include "third_party/skia/include/core/SkPictureRecorder.h" |
| 17 #include "third_party/skia/include/core/SkRect.h" |
15 #include "third_party/skia/include/core/SkRefCnt.h" | 18 #include "third_party/skia/include/core/SkRefCnt.h" |
16 #include "third_party/skia/include/core/SkScalar.h" | 19 #include "third_party/skia/include/core/SkScalar.h" |
| 20 #include "third_party/skia/include/core/SkSize.h" |
17 #include "third_party/skia/include/core/SkStream.h" | 21 #include "third_party/skia/include/core/SkStream.h" |
18 #include "third_party/skia/include/core/SkTypeface.h" | |
19 #include "third_party/skia/include/pdf/SkPDFDevice.h" | |
20 #include "third_party/skia/include/pdf/SkPDFDocument.h" | |
21 #include "ui/gfx/point.h" | 22 #include "ui/gfx/point.h" |
22 #include "ui/gfx/rect.h" | 23 #include "ui/gfx/rect.h" |
23 #include "ui/gfx/size.h" | 24 #include "ui/gfx/size.h" |
| 25 #include "ui/gfx/skia_util.h" |
24 | 26 |
25 #if defined(OS_MACOSX) | 27 #if defined(OS_MACOSX) |
26 #include "printing/pdf_metafile_cg_mac.h" | 28 #include "printing/pdf_metafile_cg_mac.h" |
27 #endif | 29 #endif |
28 | 30 |
29 #if defined(OS_POSIX) | 31 #if defined(OS_POSIX) |
30 #include "base/file_descriptor_posix.h" | 32 #include "base/file_descriptor_posix.h" |
31 #endif | 33 #endif |
32 | 34 |
| 35 namespace { |
| 36 // This struct represents all the data we need to draw and redraw this |
| 37 // page into a SkDocument. |
| 38 struct Page { |
| 39 Page(const SkSize& page_size, const SkRect& content_area) |
| 40 : page_size_(page_size), |
| 41 content_area_(content_area), |
| 42 content_(/*NULL*/) {} |
| 43 SkSize page_size_; |
| 44 SkRect content_area_; |
| 45 skia::RefPtr<SkPicture> content_; |
| 46 }; |
| 47 } // namespace |
| 48 |
| 49 static bool WriteAssetToBuffer(const SkStreamAsset* asset, |
| 50 void* buffer, |
| 51 size_t size) { |
| 52 skia::RefPtr<SkStreamAsset> assetCopy = skia::AdoptRef(asset->duplicate()); |
| 53 // keeps original asset state unchanged. |
| 54 size_t length = assetCopy->getLength(); |
| 55 if (length > size) |
| 56 return false; |
| 57 return (length == assetCopy->read(buffer, length)); |
| 58 } |
| 59 |
33 namespace printing { | 60 namespace printing { |
34 | 61 |
35 struct PdfMetafileSkiaData { | 62 struct PdfMetafileSkiaData { |
36 skia::RefPtr<SkPDFDevice> current_page_; | 63 SkPictureRecorder recorder_; // Current recording |
37 skia::RefPtr<SkCanvas> current_page_canvas_; | 64 |
38 SkPDFDocument pdf_doc_; | 65 std::vector<Page> pages_; |
39 SkDynamicMemoryWStream pdf_stream_; | 66 skia::RefPtr<SkStreamAsset> pdf_data_; |
| 67 |
40 #if defined(OS_MACOSX) | 68 #if defined(OS_MACOSX) |
41 PdfMetafileCg pdf_cg_; | 69 PdfMetafileCg pdf_cg_; |
42 #endif | 70 #endif |
43 }; | 71 }; |
44 | 72 |
45 PdfMetafileSkia::~PdfMetafileSkia() {} | 73 PdfMetafileSkia::~PdfMetafileSkia() {} |
46 | 74 |
47 bool PdfMetafileSkia::Init() { | 75 bool PdfMetafileSkia::Init() { |
48 return true; | 76 return true; |
49 } | 77 } |
| 78 |
| 79 // TODO(halcanary): Create a Metafile class that only stores data. |
| 80 // Metafile::InitFromData is orthogonal to what the rest of |
| 81 // PdfMetafileSkia does. |
50 bool PdfMetafileSkia::InitFromData(const void* src_buffer, | 82 bool PdfMetafileSkia::InitFromData(const void* src_buffer, |
51 uint32 src_buffer_size) { | 83 uint32 src_buffer_size) { |
52 return data_->pdf_stream_.write(src_buffer, src_buffer_size); | 84 data_->pdf_data_ = |
| 85 skia::AdoptRef(new SkMemoryStream(src_buffer, src_buffer_size, true)); |
| 86 return true; |
53 } | 87 } |
54 | 88 |
55 bool PdfMetafileSkia::StartPage(const gfx::Size& page_size, | 89 bool PdfMetafileSkia::StartPage(const gfx::Size& page_size, |
56 const gfx::Rect& content_area, | 90 const gfx::Rect& content_area, |
57 const float& scale_factor) { | 91 const float& scale_factor) { |
58 DCHECK(!data_->current_page_canvas_); | 92 if (data_->recorder_.getRecordingCanvas()) |
| 93 this->FinishPage(); |
| 94 DCHECK(!data_->recorder_.getRecordingCanvas()); |
| 95 SkSize sk_page_size = gfx::SizeToSkSize(page_size); |
| 96 data_->pages_.push_back(Page(sk_page_size, gfx::RectToSkRect(content_area))); |
59 | 97 |
60 // Adjust for the margins and apply the scale factor. | 98 SkCanvas* recordingCanvas = data_->recorder_.beginRecording( |
61 SkMatrix transform; | 99 sk_page_size.width(), sk_page_size.height(), NULL, 0); |
62 transform.setTranslate(SkIntToScalar(content_area.x()), | 100 // recordingCanvas is owned by the data_->recorder_. No ref() necessary. |
63 SkIntToScalar(content_area.y())); | 101 if (!recordingCanvas) |
64 transform.preScale(SkFloatToScalar(scale_factor), | 102 return false; |
65 SkFloatToScalar(scale_factor)); | 103 recordingCanvas->scale(scale_factor, scale_factor); |
66 | |
67 SkISize pdf_page_size = SkISize::Make(page_size.width(), page_size.height()); | |
68 SkISize pdf_content_size = | |
69 SkISize::Make(content_area.width(), content_area.height()); | |
70 | |
71 data_->current_page_ = skia::AdoptRef( | |
72 new SkPDFDevice(pdf_page_size, pdf_content_size, transform)); | |
73 data_->current_page_canvas_ = | |
74 skia::AdoptRef(new SkCanvas(data_->current_page_.get())); | |
75 return true; | 104 return true; |
76 } | 105 } |
77 | 106 |
78 skia::VectorCanvas* PdfMetafileSkia::GetVectorCanvasForNewPage( | 107 skia::VectorCanvas* PdfMetafileSkia::GetVectorCanvasForNewPage( |
79 const gfx::Size& page_size, | 108 const gfx::Size& page_size, |
80 const gfx::Rect& content_area, | 109 const gfx::Rect& content_area, |
81 const float& scale_factor) { | 110 const float& scale_factor) { |
82 if (!StartPage(page_size, content_area, scale_factor)) | 111 if (!StartPage(page_size, content_area, scale_factor)) |
83 return nullptr; | 112 return nullptr; |
84 return data_->current_page_canvas_.get(); | 113 return data_->recorder_.getRecordingCanvas(); |
85 } | 114 } |
86 | 115 |
87 bool PdfMetafileSkia::FinishPage() { | 116 bool PdfMetafileSkia::FinishPage() { |
88 DCHECK(data_->current_page_canvas_); | 117 if (!data_->recorder_.getRecordingCanvas()) |
89 DCHECK(data_->current_page_); | 118 return false; |
90 | 119 DCHECK(!(data_->pages_.back().content_)); |
91 data_->current_page_canvas_.clear(); // Unref SkCanvas. | 120 data_->pages_.back().content_ = |
92 data_->pdf_doc_.appendPage(data_->current_page_.get()); | 121 skia::AdoptRef(data_->recorder_.endRecording()); |
93 return true; | 122 return true; |
94 } | 123 } |
95 | 124 |
96 bool PdfMetafileSkia::FinishDocument() { | 125 bool PdfMetafileSkia::FinishDocument() { |
97 // Don't do anything if we've already set the data in InitFromData. | 126 // If we've already set the data in InitFromData, overwrite it. |
98 if (data_->pdf_stream_.getOffset()) | 127 if (data_->pdf_data_) |
99 return true; | 128 data_->pdf_data_.clear(); // Free up RAM first. |
100 | 129 |
101 if (data_->current_page_canvas_) | 130 if (data_->recorder_.getRecordingCanvas()) |
102 FinishPage(); | 131 this->FinishPage(); |
103 | 132 |
104 data_->current_page_.clear(); | 133 SkDynamicMemoryWStream pdf_stream; |
| 134 skia::RefPtr<SkDocument> pdf_doc = |
| 135 skia::AdoptRef(SkDocument::CreatePDF(&pdf_stream)); |
| 136 for (const auto& page : data_->pages_) { |
| 137 SkCanvas* canvas = pdf_doc->beginPage( |
| 138 page.page_size_.width(), page.page_size_.height(), &page.content_area_); |
| 139 canvas->drawPicture(page.content_.get()); |
| 140 pdf_doc->endPage(); |
| 141 } |
| 142 if (!pdf_doc->close()) |
| 143 return false; |
105 | 144 |
106 int font_counts[SkAdvancedTypefaceMetrics::kOther_Font + 2]; | 145 data_->pdf_data_ = skia::AdoptRef(pdf_stream.detachAsStream()); |
107 data_->pdf_doc_.getCountOfFontTypes(font_counts); | 146 return true; |
108 for (int type = 0; | |
109 type <= SkAdvancedTypefaceMetrics::kOther_Font + 1; | |
110 type++) { | |
111 for (int count = 0; count < font_counts[type]; count++) { | |
112 UMA_HISTOGRAM_ENUMERATION( | |
113 "PrintPreview.FontType", type, | |
114 SkAdvancedTypefaceMetrics::kOther_Font + 2); | |
115 } | |
116 } | |
117 | |
118 return data_->pdf_doc_.emitPDF(&data_->pdf_stream_); | |
119 } | 147 } |
120 | 148 |
121 uint32 PdfMetafileSkia::GetDataSize() const { | 149 uint32 PdfMetafileSkia::GetDataSize() const { |
122 return base::checked_cast<uint32>(data_->pdf_stream_.getOffset()); | 150 if (!data_->pdf_data_) |
| 151 return 0; |
| 152 return base::checked_cast<uint32>(data_->pdf_data_->getLength()); |
123 } | 153 } |
124 | 154 |
125 bool PdfMetafileSkia::GetData(void* dst_buffer, | 155 bool PdfMetafileSkia::GetData(void* dst_buffer, |
126 uint32 dst_buffer_size) const { | 156 uint32 dst_buffer_size) const { |
127 if (dst_buffer_size < GetDataSize()) | 157 if (!data_->pdf_data_) |
128 return false; | 158 return false; |
129 | 159 return WriteAssetToBuffer(data_->pdf_data_.get(), dst_buffer, |
130 SkAutoDataUnref data(data_->pdf_stream_.copyToData()); | 160 base::checked_cast<size_t>(dst_buffer_size)); |
131 memcpy(dst_buffer, data->bytes(), dst_buffer_size); | |
132 return true; | |
133 } | 161 } |
134 | 162 |
135 gfx::Rect PdfMetafileSkia::GetPageBounds(unsigned int page_number) const { | 163 gfx::Rect PdfMetafileSkia::GetPageBounds(unsigned int page_number) const { |
136 // TODO(vandebo) add a method to get the page size for a given page to | 164 if (page_number < data_->pages_.size()) |
137 // SkPDFDocument. | 165 return gfx::Rect(gfx::SkSizeToSize(data_->pages_[page_number].page_size_)); |
138 NOTIMPLEMENTED(); | |
139 return gfx::Rect(); | 166 return gfx::Rect(); |
140 } | 167 } |
141 | 168 |
142 unsigned int PdfMetafileSkia::GetPageCount() const { | 169 unsigned int PdfMetafileSkia::GetPageCount() const { |
143 // TODO(vandebo) add a method to get the number of pages to SkPDFDocument. | 170 return base::checked_cast<unsigned int>(data_->pages_.size()); |
144 NOTIMPLEMENTED(); | |
145 return 0; | |
146 } | 171 } |
147 | 172 |
148 gfx::NativeDrawingContext PdfMetafileSkia::context() const { | 173 gfx::NativeDrawingContext PdfMetafileSkia::context() const { |
149 NOTREACHED(); | 174 NOTREACHED(); |
150 return NULL; | 175 return NULL; |
151 } | 176 } |
152 | 177 |
153 #if defined(OS_WIN) | 178 #if defined(OS_WIN) |
154 bool PdfMetafileSkia::Playback(gfx::NativeDrawingContext hdc, | 179 bool PdfMetafileSkia::Playback(gfx::NativeDrawingContext hdc, |
155 const RECT* rect) const { | 180 const RECT* rect) const { |
(...skipping 11 matching lines...) Expand all Loading... |
167 rasterized output. Even if that flow uses PdfMetafileCg::RenderPage, | 192 rasterized output. Even if that flow uses PdfMetafileCg::RenderPage, |
168 the drawing of the PDF into the canvas may result in a rasterized output. | 193 the drawing of the PDF into the canvas may result in a rasterized output. |
169 PDFMetafileSkia::RenderPage should be not implemented as shown and instead | 194 PDFMetafileSkia::RenderPage should be not implemented as shown and instead |
170 should do something like the following CL in PluginInstance::PrintPDFOutput: | 195 should do something like the following CL in PluginInstance::PrintPDFOutput: |
171 http://codereview.chromium.org/7200040/diff/1/webkit/plugins/ppapi/ppapi_plugin_
instance.cc | 196 http://codereview.chromium.org/7200040/diff/1/webkit/plugins/ppapi/ppapi_plugin_
instance.cc |
172 */ | 197 */ |
173 bool PdfMetafileSkia::RenderPage(unsigned int page_number, | 198 bool PdfMetafileSkia::RenderPage(unsigned int page_number, |
174 CGContextRef context, | 199 CGContextRef context, |
175 const CGRect rect, | 200 const CGRect rect, |
176 const MacRenderPageParams& params) const { | 201 const MacRenderPageParams& params) const { |
177 DCHECK_GT(data_->pdf_stream_.getOffset(), 0U); | 202 DCHECK_GT(GetDataSize(), 0U); |
178 if (data_->pdf_cg_.GetDataSize() == 0) { | 203 if (data_->pdf_cg_.GetDataSize() == 0) { |
179 SkAutoDataUnref data(data_->pdf_stream_.copyToData()); | 204 if (GetDataSize() == 0) |
180 data_->pdf_cg_.InitFromData(data->bytes(), data->size()); | 205 return false; |
| 206 size_t length = data_->pdf_data_->getLength(); |
| 207 std::vector<uint8_t> buffer(length); |
| 208 (void)WriteAssetToBuffer(data_->pdf_data_.get(), &buffer[0], length); |
| 209 data_->pdf_cg_.InitFromData(&buffer[0], base::checked_cast<uint32>(length)); |
181 } | 210 } |
182 return data_->pdf_cg_.RenderPage(page_number, context, rect, params); | 211 return data_->pdf_cg_.RenderPage(page_number, context, rect, params); |
183 } | 212 } |
184 #endif | 213 #endif |
185 | 214 |
186 bool PdfMetafileSkia::SaveTo(base::File* file) const { | 215 bool PdfMetafileSkia::SaveTo(base::File* file) const { |
187 if (GetDataSize() == 0U) | 216 if (GetDataSize() == 0U) |
188 return false; | 217 return false; |
189 SkAutoDataUnref data(data_->pdf_stream_.copyToData()); | 218 DCHECK(data_->pdf_data_.get()); |
190 // TODO(halcanary): rewrite this function without extra data copy | 219 |
191 // using SkStreamAsset. | 220 skia::RefPtr<SkStreamAsset> asset = |
192 const char* ptr = reinterpret_cast<const char*>(data->data()); | 221 skia::AdoptRef(data_->pdf_data_->duplicate()); |
193 int size = base::checked_cast<int>(data->size()); | 222 // keeps original asset state unchanged. |
194 return file->WriteAtCurrentPos(ptr, size) == size; | 223 |
| 224 const size_t maximum_buffer_size = 1024 * 1024; |
| 225 std::vector<char> buffer(std::min(maximum_buffer_size, asset->getLength())); |
| 226 do { |
| 227 size_t read_size = asset->read(&buffer[0], buffer.size()); |
| 228 if (read_size == 0) |
| 229 break; |
| 230 DCHECK_GE(buffer.size(), read_size); |
| 231 if (!file->WriteAtCurrentPos(&buffer[0], |
| 232 base::checked_cast<int>(read_size))) { |
| 233 return false; |
| 234 } |
| 235 } while (!asset->isAtEnd()); |
| 236 |
| 237 return true; |
195 } | 238 } |
196 | 239 |
197 #if defined(OS_CHROMEOS) || defined(OS_ANDROID) | 240 #if defined(OS_CHROMEOS) || defined(OS_ANDROID) |
198 bool PdfMetafileSkia::SaveToFD(const base::FileDescriptor& fd) const { | 241 bool PdfMetafileSkia::SaveToFD(const base::FileDescriptor& fd) const { |
199 DCHECK_GT(data_->pdf_stream_.getOffset(), 0U); | 242 DCHECK_GT(GetDataSize(), 0U); |
200 | 243 |
201 if (fd.fd < 0) { | 244 if (fd.fd < 0) { |
202 DLOG(ERROR) << "Invalid file descriptor!"; | 245 DLOG(ERROR) << "Invalid file descriptor!"; |
203 return false; | 246 return false; |
204 } | 247 } |
205 base::File file(fd.fd); | 248 base::File file(fd.fd); |
206 bool result = SaveTo(&file); | 249 bool result = SaveTo(&file); |
207 DLOG_IF(ERROR, !result) << "Failed to save file with fd " << fd.fd; | 250 DLOG_IF(ERROR, !result) << "Failed to save file with fd " << fd.fd; |
208 | 251 |
209 if (!fd.auto_close) | 252 if (!fd.auto_close) |
210 file.TakePlatformFile(); | 253 file.TakePlatformFile(); |
211 return result; | 254 return result; |
212 } | 255 } |
213 #endif | 256 #endif |
214 | 257 |
215 PdfMetafileSkia::PdfMetafileSkia() : data_(new PdfMetafileSkiaData) { | 258 PdfMetafileSkia::PdfMetafileSkia() : data_(new PdfMetafileSkiaData) { |
216 } | 259 } |
217 | 260 |
218 scoped_ptr<PdfMetafileSkia> PdfMetafileSkia::GetMetafileForCurrentPage() { | 261 scoped_ptr<PdfMetafileSkia> PdfMetafileSkia::GetMetafileForCurrentPage() { |
219 scoped_ptr<PdfMetafileSkia> metafile; | 262 // If we only ever need the metafile for the last page, should we |
220 SkPDFDocument pdf_doc(SkPDFDocument::kDraftMode_Flags); | 263 // only keep a handle on one SkPicture? |
221 if (!pdf_doc.appendPage(data_->current_page_.get())) | 264 scoped_ptr<PdfMetafileSkia> metafile(new PdfMetafileSkia); |
| 265 |
| 266 if (data_->pages_.size() == 0) |
222 return metafile.Pass(); | 267 return metafile.Pass(); |
223 | 268 |
224 SkDynamicMemoryWStream pdf_stream; | 269 if (data_->recorder_.getRecordingCanvas()) // page outstanding |
225 if (!pdf_doc.emitPDF(&pdf_stream)) | |
226 return metafile.Pass(); | 270 return metafile.Pass(); |
227 | 271 |
228 SkAutoDataUnref data_copy(pdf_stream.copyToData()); | 272 const Page& page = data_->pages_.back(); |
229 if (data_copy->size() == 0) | |
230 return scoped_ptr<PdfMetafileSkia>(); | |
231 | 273 |
232 metafile.reset(new PdfMetafileSkia); | 274 metafile->data_->pages_.push_back(page); // Copy page data; |
233 if (!metafile->InitFromData(data_copy->bytes(), | 275 // Should increment refcnt on page->content_. |
234 base::checked_cast<uint32>(data_copy->size()))) { | 276 |
| 277 if (!metafile->FinishDocument()) // Generate PDF. |
235 metafile.reset(); | 278 metafile.reset(); |
236 } | 279 |
237 return metafile.Pass(); | 280 return metafile.Pass(); |
238 } | 281 } |
239 | 282 |
240 } // namespace printing | 283 } // namespace printing |
OLD | NEW |