Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(401)

Side by Side Diff: printing/pdf_metafile_skia.cc

Issue 731143004: Revert of Remove calls to deprecated SkPDFDevice and SkPDFDocuemnt. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Created 6 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « no previous file | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
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"
18 #include "third_party/skia/include/core/SkRefCnt.h" 15 #include "third_party/skia/include/core/SkRefCnt.h"
19 #include "third_party/skia/include/core/SkScalar.h" 16 #include "third_party/skia/include/core/SkScalar.h"
20 #include "third_party/skia/include/core/SkSize.h"
21 #include "third_party/skia/include/core/SkStream.h" 17 #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"
22 #include "ui/gfx/point.h" 21 #include "ui/gfx/point.h"
23 #include "ui/gfx/rect.h" 22 #include "ui/gfx/rect.h"
24 #include "ui/gfx/size.h" 23 #include "ui/gfx/size.h"
25 24
26 #if defined(OS_MACOSX) 25 #if defined(OS_MACOSX)
27 #include "printing/pdf_metafile_cg_mac.h" 26 #include "printing/pdf_metafile_cg_mac.h"
28 #endif 27 #endif
29 28
30 #if defined(OS_POSIX) 29 #if defined(OS_POSIX)
31 #include "base/file_descriptor_posix.h" 30 #include "base/file_descriptor_posix.h"
32 #endif 31 #endif
33 32
34 namespace {
35 // This struct represents all the data we need to draw and redraw this
36 // page into a SkDocument.
37 struct Page {
38 Page(const SkSize& page_size, const SkRect& content_area)
39 : page_size_(page_size),
40 content_area_(content_area),
41 content_(/*NULL*/) {}
42 SkSize page_size_;
43 SkRect content_area_;
44 skia::RefPtr<SkPicture> content_;
45 };
46 } // namespace
47
48 static SkSize ToSkSize(const gfx::Size& size) {
49 return SkSize::Make(SkIntToScalar(size.width()),
50 SkIntToScalar(size.height()));
51 }
52
53 static SkRect ToSkRect(const gfx::Rect& rect) {
54 return SkRect::MakeLTRB(SkIntToScalar(rect.x()), SkIntToScalar(rect.y()),
55 SkIntToScalar(rect.right()),
56 SkIntToScalar(rect.bottom()));
57 }
58
59 static gfx::Size ToGfxSize(const SkSize& size) {
60 return gfx::Size(SkScalarTruncToInt(size.width()),
61 SkScalarTruncToInt(size.height()));
62 }
63
64 static bool WriteAssetToBuffer(SkStreamAsset* asset,
65 void* buffer,
66 size_t size) {
67 DCHECK(asset->getPosition() == 0); // Be kind: please rewind.
68 size_t length = asset->getLength();
69 if (length > size)
70 return false;
71 bool success = (length == asset->read(buffer, length));
72 (void)asset->rewind();
73 return success;
74 }
75
76 namespace printing { 33 namespace printing {
77 34
78 struct PdfMetafileSkiaData { 35 struct PdfMetafileSkiaData {
79 scoped_ptr<SkPictureRecorder> recorder_; // Current recording 36 skia::RefPtr<SkPDFDevice> current_page_;
80 37 skia::RefPtr<SkCanvas> current_page_canvas_;
81 std::vector<Page> pages_; 38 SkPDFDocument pdf_doc_;
82 skia::RefPtr<SkStreamAsset> pdf_data_; 39 SkDynamicMemoryWStream pdf_stream_;
83
84 #if defined(OS_MACOSX) 40 #if defined(OS_MACOSX)
85 PdfMetafileCg pdf_cg_; 41 PdfMetafileCg pdf_cg_;
86 #endif 42 #endif
87 }; 43 };
88 44
89 PdfMetafileSkia::~PdfMetafileSkia() {} 45 PdfMetafileSkia::~PdfMetafileSkia() {}
90 46
91 bool PdfMetafileSkia::Init() { 47 bool PdfMetafileSkia::Init() {
92 return true; 48 return true;
93 } 49 }
94
95 // TODO(halcanary): Create a Metafile class that only stores data.
96 // Metafile::InitFromData is orthogonal to what the rest of
97 // PdfMetafileSkia does.
98 bool PdfMetafileSkia::InitFromData(const void* src_buffer, 50 bool PdfMetafileSkia::InitFromData(const void* src_buffer,
99 uint32 src_buffer_size) { 51 uint32 src_buffer_size) {
100 if (data_->pdf_data_) 52 return data_->pdf_stream_.write(src_buffer, src_buffer_size);
101 data_->pdf_data_.clear(); // free up RAM first.
102 SkDynamicMemoryWStream dynamic_memory;
103 if (!dynamic_memory.write(src_buffer, src_buffer_size))
104 return false;
105 data_->pdf_data_ = skia::AdoptRef(dynamic_memory.detachAsStream());
106 return true;
107 } 53 }
108 54
109 bool PdfMetafileSkia::StartPage(const gfx::Size& page_size, 55 bool PdfMetafileSkia::StartPage(const gfx::Size& page_size,
110 const gfx::Rect& content_area, 56 const gfx::Rect& content_area,
111 const float& scale_factor) { 57 const float& scale_factor) {
112 if (data_->recorder_) 58 DCHECK(!data_->current_page_canvas_);
113 this->FinishPage();
114 DCHECK(!data_->recorder_);
115 data_->recorder_.reset(SkNEW(SkPictureRecorder));
116 SkSize sk_page_size = ToSkSize(page_size);
117 data_->pages_.push_back(Page(sk_page_size, ToSkRect(content_area)));
118 59
119 SkCanvas* recordingCanvas = data_->recorder_->beginRecording( 60 // Adjust for the margins and apply the scale factor.
120 sk_page_size.width(), sk_page_size.height(), NULL, 0); 61 SkMatrix transform;
121 // recordingCanvas is owned by the data_->recorder_. No ref() necessary. 62 transform.setTranslate(SkIntToScalar(content_area.x()),
122 if (!recordingCanvas) 63 SkIntToScalar(content_area.y()));
123 return false; 64 transform.preScale(SkFloatToScalar(scale_factor),
124 recordingCanvas->scale(scale_factor, scale_factor); 65 SkFloatToScalar(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()));
125 return true; 75 return true;
126 } 76 }
127 77
128 skia::VectorCanvas* PdfMetafileSkia::GetVectorCanvasForNewPage( 78 skia::VectorCanvas* PdfMetafileSkia::GetVectorCanvasForNewPage(
129 const gfx::Size& page_size, 79 const gfx::Size& page_size,
130 const gfx::Rect& content_area, 80 const gfx::Rect& content_area,
131 const float& scale_factor) { 81 const float& scale_factor) {
132 if (!StartPage(page_size, content_area, scale_factor)) 82 if (!StartPage(page_size, content_area, scale_factor))
133 return nullptr; 83 return nullptr;
134 return data_->recorder_->getRecordingCanvas(); 84 return data_->current_page_canvas_.get();
135 } 85 }
136 86
137 bool PdfMetafileSkia::FinishPage() { 87 bool PdfMetafileSkia::FinishPage() {
138 if (!data_->recorder_) 88 DCHECK(data_->current_page_canvas_);
139 return false; 89 DCHECK(data_->current_page_);
140 DCHECK(!(data_->pages_.back().content_)); 90
141 data_->pages_.back().content_ = 91 data_->current_page_canvas_.clear(); // Unref SkCanvas.
142 skia::AdoptRef(data_->recorder_->endRecording()); 92 data_->pdf_doc_.appendPage(data_->current_page_.get());
143 data_->recorder_.reset();
144 return true; 93 return true;
145 } 94 }
146 95
147 bool PdfMetafileSkia::FinishDocument() { 96 bool PdfMetafileSkia::FinishDocument() {
148 // If we've already set the data in InitFromData, overwrite it. 97 // Don't do anything if we've already set the data in InitFromData.
149 if (data_->pdf_data_) 98 if (data_->pdf_stream_.getOffset())
150 data_->pdf_data_.clear(); // Free up RAM first. 99 return true;
151 100
152 if (data_->recorder_) 101 if (data_->current_page_canvas_)
153 this->FinishPage(); 102 FinishPage();
154 103
155 SkDynamicMemoryWStream pdf_stream; 104 data_->current_page_.clear();
156 skia::RefPtr<SkDocument> pdf_doc = 105
157 skia::AdoptRef(SkDocument::CreatePDF(&pdf_stream)); 106 int font_counts[SkAdvancedTypefaceMetrics::kOther_Font + 2];
158 for (const auto& page : data_->pages_) { 107 data_->pdf_doc_.getCountOfFontTypes(font_counts);
159 SkCanvas* canvas = pdf_doc->beginPage( 108 for (int type = 0;
160 page.page_size_.width(), page.page_size_.height(), &page.content_area_); 109 type <= SkAdvancedTypefaceMetrics::kOther_Font + 1;
161 canvas->drawPicture(page.content_.get()); 110 type++) {
162 pdf_doc->endPage(); 111 for (int count = 0; count < font_counts[type]; count++) {
112 UMA_HISTOGRAM_ENUMERATION(
113 "PrintPreview.FontType", type,
114 SkAdvancedTypefaceMetrics::kOther_Font + 2);
115 }
163 } 116 }
164 if (!pdf_doc->close())
165 return false;
166 117
167 data_->pdf_data_ = skia::AdoptRef(pdf_stream.detachAsStream()); 118 return data_->pdf_doc_.emitPDF(&data_->pdf_stream_);
168 return true;
169 } 119 }
170 120
171 uint32 PdfMetafileSkia::GetDataSize() const { 121 uint32 PdfMetafileSkia::GetDataSize() const {
172 if (!data_->pdf_data_) 122 return base::checked_cast<uint32>(data_->pdf_stream_.getOffset());
173 return 0;
174 return base::checked_cast<uint32>(data_->pdf_data_->getLength());
175 } 123 }
176 124
177 bool PdfMetafileSkia::GetData(void* dst_buffer, 125 bool PdfMetafileSkia::GetData(void* dst_buffer,
178 uint32 dst_buffer_size) const { 126 uint32 dst_buffer_size) const {
179 if (!data_->pdf_data_) 127 if (dst_buffer_size < GetDataSize())
180 return false; 128 return false;
181 return WriteAssetToBuffer(data_->pdf_data_.get(), dst_buffer, 129
182 base::checked_cast<size_t>(dst_buffer_size)); 130 SkAutoDataUnref data(data_->pdf_stream_.copyToData());
131 memcpy(dst_buffer, data->bytes(), dst_buffer_size);
132 return true;
183 } 133 }
184 134
185 gfx::Rect PdfMetafileSkia::GetPageBounds(unsigned int page_number) const { 135 gfx::Rect PdfMetafileSkia::GetPageBounds(unsigned int page_number) const {
186 if (page_number < data_->pages_.size()) 136 // TODO(vandebo) add a method to get the page size for a given page to
187 return gfx::Rect(ToGfxSize(data_->pages_[page_number].page_size_)); 137 // SkPDFDocument.
138 NOTIMPLEMENTED();
188 return gfx::Rect(); 139 return gfx::Rect();
189 } 140 }
190 141
191 unsigned int PdfMetafileSkia::GetPageCount() const { 142 unsigned int PdfMetafileSkia::GetPageCount() const {
192 return base::checked_cast<unsigned int>(data_->pages_.size()); 143 // TODO(vandebo) add a method to get the number of pages to SkPDFDocument.
144 NOTIMPLEMENTED();
145 return 0;
193 } 146 }
194 147
195 gfx::NativeDrawingContext PdfMetafileSkia::context() const { 148 gfx::NativeDrawingContext PdfMetafileSkia::context() const {
196 NOTREACHED(); 149 NOTREACHED();
197 return NULL; 150 return NULL;
198 } 151 }
199 152
200 #if defined(OS_WIN) 153 #if defined(OS_WIN)
201 bool PdfMetafileSkia::Playback(gfx::NativeDrawingContext hdc, 154 bool PdfMetafileSkia::Playback(gfx::NativeDrawingContext hdc,
202 const RECT* rect) const { 155 const RECT* rect) const {
(...skipping 11 matching lines...) Expand all
214 rasterized output. Even if that flow uses PdfMetafileCg::RenderPage, 167 rasterized output. Even if that flow uses PdfMetafileCg::RenderPage,
215 the drawing of the PDF into the canvas may result in a rasterized output. 168 the drawing of the PDF into the canvas may result in a rasterized output.
216 PDFMetafileSkia::RenderPage should be not implemented as shown and instead 169 PDFMetafileSkia::RenderPage should be not implemented as shown and instead
217 should do something like the following CL in PluginInstance::PrintPDFOutput: 170 should do something like the following CL in PluginInstance::PrintPDFOutput:
218 http://codereview.chromium.org/7200040/diff/1/webkit/plugins/ppapi/ppapi_plugin_ instance.cc 171 http://codereview.chromium.org/7200040/diff/1/webkit/plugins/ppapi/ppapi_plugin_ instance.cc
219 */ 172 */
220 bool PdfMetafileSkia::RenderPage(unsigned int page_number, 173 bool PdfMetafileSkia::RenderPage(unsigned int page_number,
221 CGContextRef context, 174 CGContextRef context,
222 const CGRect rect, 175 const CGRect rect,
223 const MacRenderPageParams& params) const { 176 const MacRenderPageParams& params) const {
224 DCHECK_GT(GetDataSize(), 0U); 177 DCHECK_GT(data_->pdf_stream_.getOffset(), 0U);
225 if (data_->pdf_cg_.GetDataSize() == 0) { 178 if (data_->pdf_cg_.GetDataSize() == 0) {
226 if (GetDataSize() == 0) 179 SkAutoDataUnref data(data_->pdf_stream_.copyToData());
227 return false; 180 data_->pdf_cg_.InitFromData(data->bytes(), data->size());
228 size_t length = data_->pdf_data_->getLength();
229 std::vector<uint8_t> buffer(length);
230 (void)WriteAssetToBuffer(data_->pdf_data_.get(), &buffer[0], length);
231 data_->pdf_cg_.InitFromData(&buffer[0], base::checked_cast<uint32>(length));
232 } 181 }
233 return data_->pdf_cg_.RenderPage(page_number, context, rect, params); 182 return data_->pdf_cg_.RenderPage(page_number, context, rect, params);
234 } 183 }
235 #endif 184 #endif
236 185
237 bool PdfMetafileSkia::SaveTo(base::File* file) const { 186 bool PdfMetafileSkia::SaveTo(base::File* file) const {
238 if (GetDataSize() == 0U) 187 if (GetDataSize() == 0U)
239 return false; 188 return false;
240 DCHECK(data_->pdf_data_.get()); 189 SkAutoDataUnref data(data_->pdf_stream_.copyToData());
241 SkStreamAsset* asset = data_->pdf_data_.get(); 190 // TODO(halcanary): rewrite this function without extra data copy
242 DCHECK_EQ(asset->getPosition(), 0U); 191 // using SkStreamAsset.
243 192 const char* ptr = reinterpret_cast<const char*>(data->data());
244 const size_t maximum_buffer_size = 1024 * 1024; 193 int size = base::checked_cast<int>(data->size());
245 std::vector<char> buffer(std::min(maximum_buffer_size, asset->getLength())); 194 return file->WriteAtCurrentPos(ptr, size) == size;
246 do {
247 size_t read_size = asset->read(&buffer[0], buffer.size());
248 if (read_size == 0)
249 break;
250 DCHECK_GE(buffer.size(), read_size);
251 if (!file->WriteAtCurrentPos(&buffer[0],
252 base::checked_cast<int>(read_size))) {
253 (void)asset->rewind();
254 return false;
255 }
256 } while (!asset->isAtEnd());
257
258 (void)asset->rewind();
259 return true;
260 } 195 }
261 196
262 #if defined(OS_CHROMEOS) || defined(OS_ANDROID) 197 #if defined(OS_CHROMEOS) || defined(OS_ANDROID)
263 bool PdfMetafileSkia::SaveToFD(const base::FileDescriptor& fd) const { 198 bool PdfMetafileSkia::SaveToFD(const base::FileDescriptor& fd) const {
264 DCHECK_GT(GetDataSize(), 0U); 199 DCHECK_GT(data_->pdf_stream_.getOffset(), 0U);
265 200
266 if (fd.fd < 0) { 201 if (fd.fd < 0) {
267 DLOG(ERROR) << "Invalid file descriptor!"; 202 DLOG(ERROR) << "Invalid file descriptor!";
268 return false; 203 return false;
269 } 204 }
270 base::File file(fd.fd); 205 base::File file(fd.fd);
271 bool result = SaveTo(&file); 206 bool result = SaveTo(&file);
272 DLOG_IF(ERROR, !result) << "Failed to save file with fd " << fd.fd; 207 DLOG_IF(ERROR, !result) << "Failed to save file with fd " << fd.fd;
273 208
274 if (!fd.auto_close) 209 if (!fd.auto_close)
275 file.TakePlatformFile(); 210 file.TakePlatformFile();
276 return result; 211 return result;
277 } 212 }
278 #endif 213 #endif
279 214
280 PdfMetafileSkia::PdfMetafileSkia() : data_(new PdfMetafileSkiaData) { 215 PdfMetafileSkia::PdfMetafileSkia() : data_(new PdfMetafileSkiaData) {
281 } 216 }
282 217
283 scoped_ptr<PdfMetafileSkia> PdfMetafileSkia::GetMetafileForCurrentPage() { 218 scoped_ptr<PdfMetafileSkia> PdfMetafileSkia::GetMetafileForCurrentPage() {
284 // If we only ever need the metafile for the last page, should we 219 scoped_ptr<PdfMetafileSkia> metafile;
285 // only keep a handle on one SkPicture? 220 SkPDFDocument pdf_doc(SkPDFDocument::kDraftMode_Flags);
286 scoped_ptr<PdfMetafileSkia> metafile(new PdfMetafileSkia); 221 if (!pdf_doc.appendPage(data_->current_page_.get()))
287
288 if (data_->pages_.size() == 0)
289 return metafile.Pass(); 222 return metafile.Pass();
290 223
291 if (data_->recorder_) // page outstanding 224 SkDynamicMemoryWStream pdf_stream;
225 if (!pdf_doc.emitPDF(&pdf_stream))
292 return metafile.Pass(); 226 return metafile.Pass();
293 227
294 const Page& page = data_->pages_.back(); 228 SkAutoDataUnref data_copy(pdf_stream.copyToData());
229 if (data_copy->size() == 0)
230 return scoped_ptr<PdfMetafileSkia>();
295 231
296 metafile->data_->pages_.push_back(page); // Copy page data; 232 metafile.reset(new PdfMetafileSkia);
297 // Should increment refcnt on page->content_. 233 if (!metafile->InitFromData(data_copy->bytes(),
298 234 base::checked_cast<uint32>(data_copy->size()))) {
299 if (!metafile->FinishDocument()) // Generate PDF.
300 metafile.reset(); 235 metafile.reset();
301 236 }
302 return metafile.Pass(); 237 return metafile.Pass();
303 } 238 }
304 239
305 } // namespace printing 240 } // namespace printing
OLDNEW
« no previous file with comments | « no previous file | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698