OLD | NEW |
---|---|
1 /* | 1 /* |
2 * Copyright 2012 Google Inc. | 2 * Copyright 2012 Google Inc. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license that can be | 4 * Use of this source code is governed by a BSD-style license that can be |
5 * found in the LICENSE file. | 5 * found in the LICENSE file. |
6 */ | 6 */ |
7 | 7 |
8 #include "SkCanvas.h" | 8 #include "SkCanvas.h" |
9 #include "SkDevice.h" | 9 #include "SkDevice.h" |
10 #include "SkForceLinking.h" | 10 #include "SkForceLinking.h" |
11 #include "SkGraphics.h" | 11 #include "SkGraphics.h" |
12 #include "SkImageEncoder.h" | 12 #include "SkImageEncoder.h" |
13 #include "SkOSFile.h" | 13 #include "SkOSFile.h" |
14 #include "SkCommandLineFlags.h" | |
mtklein
2014/08/06 18:48:36
questionable alphabetism?
hal.canary
2014/08/06 20:11:23
Done.
| |
14 #include "SkPicture.h" | 15 #include "SkPicture.h" |
15 #include "SkPixelRef.h" | 16 #include "SkPixelRef.h" |
16 #include "SkStream.h" | 17 #include "SkStream.h" |
17 #include "SkTArray.h" | 18 #include "SkTArray.h" |
18 #include "PdfRenderer.h" | 19 #include "PdfRenderer.h" |
19 #include "picture_utils.h" | 20 #include "picture_utils.h" |
20 | 21 |
21 __SK_FORCE_IMAGE_DECODER_LINKING; | 22 __SK_FORCE_IMAGE_DECODER_LINKING; |
22 | 23 |
23 #ifdef SK_USE_CDB | 24 #ifdef SK_USE_CDB |
24 #include "win_dbghelp.h" | 25 #include "win_dbghelp.h" |
25 #endif | 26 #endif |
26 | 27 |
27 /** | 28 /** |
28 * render_pdfs | 29 * render_pdfs |
29 * | 30 * |
30 * Given list of directories and files to use as input, expects to find .skp | 31 * Given list of directories and files to use as input, expects to find .skp |
31 * files and it will convert them to .pdf files writing them in the output | 32 * files and it will convert them to .pdf files writing them in the output |
32 * directory. | 33 * directory. |
33 * | 34 * |
34 * Returns zero exit code if all .skp files were converted successfully, | 35 * Returns zero exit code if all .skp files were converted successfully, |
35 * otherwise returns error code 1. | 36 * otherwise returns error code 1. |
36 */ | 37 */ |
37 | 38 |
38 static const char PDF_FILE_EXTENSION[] = "pdf"; | 39 static const char PDF_FILE_EXTENSION[] = "pdf"; |
39 static const char SKP_FILE_EXTENSION[] = "skp"; | 40 static const char SKP_FILE_EXTENSION[] = "skp"; |
40 | 41 |
41 static void usage(const char* argv0) { | 42 |
42 SkDebugf("SKP to PDF rendering tool\n"); | 43 DEFINE_string2(inputPaths, r, NULL, |
mtklein
2014/08/06 18:48:36
Usually we default strings to "".
hal.canary
2014/08/06 20:11:24
Done.
| |
43 SkDebugf("\n" | 44 "A list of directories and files to use as input. " |
44 "Usage: \n" | 45 "Files are expected to have the .skp extension."); |
45 " %s <input>... [-w <outputDir>] [--jpegQuality N] \n" | 46 |
46 , argv0); | 47 DEFINE_string2(outputDir, w, NULL, |
47 SkDebugf("\n\n"); | 48 "Directory to write the rendered pdfs."); |
48 SkDebugf( | 49 |
49 " input: A list of directories and files to use as input. Files are\n" | 50 DEFINE_string2(match, m, NULL, |
50 " expected to have the .skp extension.\n\n"); | 51 "[~][^]substring[$] [...] of filenames to run.\n" |
51 SkDebugf( | 52 "Multiple matches may be separated by spaces.\n" |
52 " outputDir: directory to write the rendered pdfs.\n\n"); | 53 "~ causes a matching file to always be skipped\n" |
53 SkDebugf("\n"); | 54 "^ requires the start of the file to match\n" |
54 SkDebugf( | 55 "$ requires the end of the file to match\n" |
55 " jpegQuality N: encodes images in JPEG at quality level N, which can\n" | 56 "^ and $ requires an exact match\n" |
56 " be in range 0-100).\n" | 57 "If a file does not match any list entry,\n" |
57 " N = -1 will disable JPEG compression.\n" | 58 "it is skipped unless some list entry starts with ~"); |
58 " Default is N = 100, maximum quality.\n\n"); | 59 |
59 SkDebugf("\n"); | 60 DEFINE_int32(jpegQuality, 100, |
60 } | 61 "Encodes images in JPEG at quality level N, which can be in " |
62 "range 0-100). N = -1 will disable JPEG compression. " | |
63 "Default is N = 100, maximum quality."); | |
61 | 64 |
62 /** Replaces the extension of a file. | 65 /** Replaces the extension of a file. |
63 * @param path File name whose extension will be changed. | 66 * @param path File name whose extension will be changed. |
64 * @param old_extension The old extension. | 67 * @param old_extension The old extension. |
65 * @param new_extension The new extension. | 68 * @param new_extension The new extension. |
66 * @returns false if the file did not has the expected extension. | 69 * @returns false if the file did not has the expected extension. |
67 * if false is returned, contents of path are undefined. | 70 * if false is returned, contents of path are undefined. |
68 */ | 71 */ |
69 static bool replace_filename_extension(SkString* path, | 72 static bool replace_filename_extension(SkString* path, |
70 const char old_extension[], | 73 const char old_extension[], |
71 const char new_extension[]) { | 74 const char new_extension[]) { |
72 if (path->endsWith(old_extension)) { | 75 if (path->endsWith(old_extension)) { |
73 path->remove(path->size() - strlen(old_extension), | 76 path->remove(path->size() - strlen(old_extension), |
74 strlen(old_extension)); | 77 strlen(old_extension)); |
75 if (!path->endsWith(".")) { | 78 if (!path->endsWith(".")) { |
76 return false; | 79 return false; |
77 } | 80 } |
78 path->append(new_extension); | 81 path->append(new_extension); |
79 return true; | 82 return true; |
80 } | 83 } |
81 return false; | 84 return false; |
82 } | 85 } |
83 | 86 |
84 int gJpegQuality = 100; | |
85 // the size_t* parameter is deprecated, so we ignore it | 87 // the size_t* parameter is deprecated, so we ignore it |
86 static SkData* encode_to_dct_data(size_t*, const SkBitmap& bitmap) { | 88 static SkData* encode_to_dct_data(size_t*, const SkBitmap& bitmap) { |
87 if (gJpegQuality == -1) { | 89 if (FLAGS_jpegQuality == -1) { |
88 return NULL; | 90 return NULL; |
89 } | 91 } |
90 | 92 |
91 SkBitmap bm = bitmap; | 93 SkBitmap bm = bitmap; |
92 #if defined(SK_BUILD_FOR_MAC) | 94 #if defined(SK_BUILD_FOR_MAC) |
93 // Workaround bug #1043 where bitmaps with referenced pixels cause | 95 // Workaround bug #1043 where bitmaps with referenced pixels cause |
94 // CGImageDestinationFinalize to crash | 96 // CGImageDestinationFinalize to crash |
95 SkBitmap copy; | 97 SkBitmap copy; |
96 bitmap.deepCopyTo(©); | 98 bitmap.deepCopyTo(©); |
97 bm = copy; | 99 bm = copy; |
98 #endif | 100 #endif |
99 | 101 |
100 return SkImageEncoder::EncodeData(bm, | 102 return SkImageEncoder::EncodeData( |
101 SkImageEncoder::kJPEG_Type, | 103 bm, SkImageEncoder::kJPEG_Type, FLAGS_jpegQuality); |
102 gJpegQuality); | |
103 } | 104 } |
104 | 105 |
105 /** Builds the output filename. path = dir/name, and it replaces expected | 106 /** Builds the output filename. path = dir/name, and it replaces expected |
106 * .skp extension with .pdf extention. | 107 * .skp extension with .pdf extention. |
107 * @param path Output filename. | 108 * @param path Output filename. |
108 * @param name The name of the file. | 109 * @param name The name of the file. |
109 * @returns false if the file did not has the expected extension. | 110 * @returns false if the file did not has the expected extension. |
110 * if false is returned, contents of path are undefined. | 111 * if false is returned, contents of path are undefined. |
111 */ | 112 */ |
112 static bool make_output_filepath(SkString* path, const SkString& dir, | 113 static bool make_output_filepath(SkString* path, const SkString& dir, |
(...skipping 27 matching lines...) Expand all Loading... | |
140 } | 141 } |
141 | 142 |
142 return stream; | 143 return stream; |
143 } | 144 } |
144 | 145 |
145 /** Reads an skp file, renders it to pdf and writes the output to a pdf file | 146 /** Reads an skp file, renders it to pdf and writes the output to a pdf file |
146 * @param inputPath The skp file to be read. | 147 * @param inputPath The skp file to be read. |
147 * @param outputDir Output dir. | 148 * @param outputDir Output dir. |
148 * @param renderer The object responsible to render the skp object into pdf. | 149 * @param renderer The object responsible to render the skp object into pdf. |
149 */ | 150 */ |
150 static bool render_pdf(const SkString& inputPath, const SkString& outputDir, | 151 static bool render_pdf(const char* inputPath, const SkString& outputDir, |
151 sk_tools::PdfRenderer& renderer) { | 152 sk_tools::PdfRenderer& renderer) { |
152 SkString inputFilename = SkOSPath::Basename(inputPath.c_str()); | 153 SkString inputFilename = SkOSPath::Basename(inputPath); |
153 | 154 |
154 SkFILEStream inputStream; | 155 SkFILEStream inputStream; |
155 inputStream.setPath(inputPath.c_str()); | 156 inputStream.setPath(inputPath); |
156 if (!inputStream.isValid()) { | 157 if (!inputStream.isValid()) { |
157 SkDebugf("Could not open file %s\n", inputPath.c_str()); | 158 SkDebugf("Could not open file %s\n", inputPath); |
158 return false; | 159 return false; |
159 } | 160 } |
160 | 161 |
161 SkAutoTUnref<SkPicture> picture(SkPicture::CreateFromStream(&inputStream)); | 162 SkAutoTUnref<SkPicture> picture(SkPicture::CreateFromStream(&inputStream)); |
162 | 163 |
163 if (NULL == picture.get()) { | 164 if (NULL == picture.get()) { |
164 SkDebugf("Could not read an SkPicture from %s\n", inputPath.c_str()); | 165 SkDebugf("Could not read an SkPicture from %s\n", inputPath); |
165 return false; | 166 return false; |
166 } | 167 } |
167 | 168 |
168 SkDebugf("exporting... [%i %i] %s\n", picture->width(), picture->height(), | 169 SkDebugf("exporting... [%-4i %6i] %s\n", |
169 inputPath.c_str()); | 170 picture->width(), picture->height(), inputPath); |
170 | 171 |
171 SkWStream* stream(open_stream(outputDir, inputFilename)); | 172 SkWStream* stream(open_stream(outputDir, inputFilename)); |
172 | 173 |
173 if (!stream) { | 174 if (!stream) { |
174 return false; | 175 return false; |
175 } | 176 } |
176 | 177 |
177 renderer.init(picture, stream); | 178 renderer.init(picture, stream); |
178 | 179 |
179 bool success = renderer.render(); | 180 bool success = renderer.render(); |
180 SkDELETE(stream); | 181 SkDELETE(stream); |
181 | 182 |
182 renderer.end(); | 183 renderer.end(); |
183 | 184 |
184 return success; | 185 return success; |
185 } | 186 } |
186 | 187 |
188 #include <stdlib.h> | |
189 static int compare_strings(const SkString& a, const SkString& b) { | |
190 size_t len = (a.size() < b.size()) ? a.size() : b.size(); | |
191 int cmp = memcmp(a.c_str(), b.c_str(), len); | |
192 if (0 == cmp) { | |
193 return b.size() - a.size(); | |
194 } else { | |
195 return cmp; | |
196 } | |
197 } | |
198 static int compare_void_strings(const void* a,const void* b) { | |
199 return compare_strings(*(const SkString*) a, | |
200 *(const SkString*) b); | |
201 } | |
202 | |
203 static void sort_string_array(SkTArray<SkString>* array) { | |
204 qsort(array->begin(), array->count(), sizeof(SkString), | |
mtklein
2014/08/06 18:48:36
Using qsort with non-POD data seems error prone.
hal.canary
2014/08/06 20:11:23
Done.
| |
205 compare_void_strings); | |
206 } | |
207 | |
187 /** For each file in the directory or for the file passed in input, call | 208 /** For each file in the directory or for the file passed in input, call |
188 * render_pdf. | 209 * render_pdf. |
189 * @param input A directory or an skp file. | 210 * @param input A directory or an skp file. |
190 * @param outputDir Output dir. | 211 * @param outputDir Output dir. |
191 * @param renderer The object responsible to render the skp object into pdf. | 212 * @param renderer The object responsible to render the skp object into pdf. |
192 */ | 213 */ |
193 static int process_input(const SkString& input, const SkString& outputDir, | 214 static int process_input( |
194 sk_tools::PdfRenderer& renderer) { | 215 const SkCommandLineFlags::StringArray& inputs, |
195 int failures = 0; | 216 const SkString& outputDir, |
196 if (sk_isdir(input.c_str())) { | 217 sk_tools::PdfRenderer& renderer) { |
197 SkOSFile::Iter iter(input.c_str(), SKP_FILE_EXTENSION); | 218 SkTArray<SkString> files; |
198 SkString inputFilename; | 219 for (int i = 0; i < inputs.count(); i ++) { |
199 while (iter.next(&inputFilename)) { | 220 const char* input = inputs[i]; |
200 SkString inputPath = SkOSPath::Join(input.c_str(), inputFilename.c_s tr()); | 221 if (sk_isdir(input)) { |
201 if (!render_pdf(inputPath, outputDir, renderer)) { | 222 SkOSFile::Iter iter(input, SKP_FILE_EXTENSION); |
202 ++failures; | 223 SkString inputFilename; |
224 while (iter.next(&inputFilename)) { | |
225 if (!SkCommandLineFlags::ShouldSkip( | |
226 FLAGS_match, inputFilename.c_str())) { | |
227 files.push_back( | |
228 SkOSPath::Join(input, inputFilename.c_str())); | |
229 } | |
230 } | |
231 } else { | |
232 if (!SkCommandLineFlags::ShouldSkip(FLAGS_match, input)) { | |
233 files.push_back(SkString(input)); | |
203 } | 234 } |
204 } | 235 } |
205 } else { | 236 } |
206 SkString inputPath(input); | 237 sort_string_array(&files); |
mtklein
2014/08/06 18:48:36
Or maybe just don't sort?
hal.canary
2014/08/06 20:11:24
I am sorting so that I can easily verify that the
| |
207 if (!render_pdf(inputPath, outputDir, renderer)) { | 238 int failures = 0; |
239 for (int i = 0; i < files.count(); i ++) { | |
240 if (!render_pdf(files[i].c_str(), outputDir, renderer)) { | |
208 ++failures; | 241 ++failures; |
209 } | 242 } |
210 } | 243 } |
211 return failures; | 244 return failures; |
212 } | 245 } |
213 | 246 |
214 static void parse_commandline(int argc, char* const argv[], | |
215 SkTArray<SkString>* inputs, | |
216 SkString* outputDir) { | |
217 const char* argv0 = argv[0]; | |
218 char* const* stop = argv + argc; | |
219 | |
220 for (++argv; argv < stop; ++argv) { | |
221 if ((0 == strcmp(*argv, "-h")) || (0 == strcmp(*argv, "--help"))) { | |
222 usage(argv0); | |
223 exit(-1); | |
224 } else if (0 == strcmp(*argv, "-w")) { | |
225 ++argv; | |
226 if (argv >= stop) { | |
227 SkDebugf("Missing outputDir for -w\n"); | |
228 usage(argv0); | |
229 exit(-1); | |
230 } | |
231 *outputDir = SkString(*argv); | |
232 } else if (0 == strcmp(*argv, "--jpegQuality")) { | |
233 ++argv; | |
234 if (argv >= stop) { | |
235 SkDebugf("Missing argument for --jpegQuality\n"); | |
236 usage(argv0); | |
237 exit(-1); | |
238 } | |
239 gJpegQuality = atoi(*argv); | |
240 if (gJpegQuality < -1 || gJpegQuality > 100) { | |
241 SkDebugf("Invalid argument for --jpegQuality\n"); | |
242 usage(argv0); | |
243 exit(-1); | |
244 } | |
245 } else { | |
246 inputs->push_back(SkString(*argv)); | |
247 } | |
248 } | |
249 | |
250 if (inputs->count() < 1) { | |
251 usage(argv0); | |
252 exit(-1); | |
253 } | |
254 } | |
255 | |
256 int tool_main_core(int argc, char** argv); | 247 int tool_main_core(int argc, char** argv); |
257 int tool_main_core(int argc, char** argv) { | 248 int tool_main_core(int argc, char** argv) { |
249 SkCommandLineFlags::Parse(argc, argv); | |
250 | |
258 SkAutoGraphics ag; | 251 SkAutoGraphics ag; |
259 SkTArray<SkString> inputs; | |
260 | 252 |
261 SkAutoTUnref<sk_tools::PdfRenderer> | 253 SkAutoTUnref<sk_tools::PdfRenderer> |
262 renderer(SkNEW_ARGS(sk_tools::SimplePdfRenderer, (encode_to_dct_data))); | 254 renderer(SkNEW_ARGS(sk_tools::SimplePdfRenderer, (encode_to_dct_data))); |
263 SkASSERT(renderer.get()); | 255 SkASSERT(renderer.get()); |
264 | 256 |
265 SkString outputDir; | 257 SkString outputDir; |
266 parse_commandline(argc, argv, &inputs, &outputDir); | 258 if (FLAGS_outputDir.count() > 0) { |
259 outputDir = FLAGS_outputDir[0]; | |
260 } | |
267 | 261 |
268 int failures = 0; | 262 int failures = process_input(FLAGS_inputPaths, outputDir, *renderer); |
269 for (int i = 0; i < inputs.count(); i ++) { | |
270 failures += process_input(inputs[i], outputDir, *renderer); | |
271 } | |
272 | 263 |
273 if (failures != 0) { | 264 if (failures != 0) { |
274 SkDebugf("Failed to render %i PDFs.\n", failures); | 265 SkDebugf("Failed to render %i PDFs.\n", failures); |
275 return 1; | 266 return 1; |
276 } | 267 } |
277 | 268 |
278 return 0; | 269 return 0; |
279 } | 270 } |
280 | 271 |
281 int tool_main(int argc, char** argv); | 272 int tool_main(int argc, char** argv); |
(...skipping 11 matching lines...) Expand all Loading... | |
293 } | 284 } |
294 #endif | 285 #endif |
295 return 0; | 286 return 0; |
296 } | 287 } |
297 | 288 |
298 #if !defined SK_BUILD_FOR_IOS | 289 #if !defined SK_BUILD_FOR_IOS |
299 int main(int argc, char * const argv[]) { | 290 int main(int argc, char * const argv[]) { |
300 return tool_main(argc, (char**) argv); | 291 return tool_main(argc, (char**) argv); |
301 } | 292 } |
302 #endif | 293 #endif |
OLD | NEW |