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

Side by Side Diff: samples/gpu2d/svgloader.js

Issue 652016: Added the bulk of the algorithm for GPU accelerated 2D vector curve... (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src/o3d/
Patch Set: '' Created 10 years, 10 months 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 | Annotate | Revision Log
« no previous file with comments | « samples/gpu2d/svg_thin_crescent.html ('k') | samples/gpu2d/svgsample.js » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Property Changes:
Added: svn:eol-style
+ LF
OLDNEW
(Empty)
1 /*
2 * Copyright 2010, Google Inc.
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are
7 * met:
8 *
9 * * Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * * Redistributions in binary form must reproduce the above
12 * copyright notice, this list of conditions and the following disclaimer
13 * in the documentation and/or other materials provided with the
14 * distribution.
15 * * Neither the name of Google Inc. nor the names of its
16 * contributors may be used to endorse or promote products derived from
17 * this software without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 */
31
32 /**
33 * @fileoverview Loads SVG files and creates o3djs.gpu2d.Paths from
34 * them, inserting all of the associated geometric nodes under the
35 * passed Transform.
36 * <P>
37 * This file depends on the O3D APIs, o3djs.base, o3djs.io, o3djs.math,
38 * o3djs.gpu2d, and the XML for &lt;SCRIPT&gt; parser available from
39 * http://xmljs.sourceforge.net/ .
40 */
41
42 /**
43 * Constructs a new SVGLoader.
44 * @constructor
45 */
46 function SVGLoader() {
47 }
48
49 /**
50 * Helper function to defer the execution of another function.
51 * @param {function(): void} func function to execute later.
52 * @private
53 */
54 function runLater_(func) {
55 setTimeout(func, 0);
56 }
57
58 /**
59 * Loads an SVG file at the given URL. The file is downloaded in the
60 * background. Graphical objects are allocated in the given Pack, and
61 * created materials are registered under the given DrawList,
62 * typically the zOrderedDrawList from an
63 * o3djs.rendergraph.ViewInfo. The created Shapes are parented under
64 * the given Transform. If the optional completion callback is
65 * specified, it is called once the file has been downloaded and
66 * processed.
67 * @param {string} url URL of the SVG file to load.
68 * @param {boolean} flipY Whether the Y coordinates should be flipped
69 * to better match the 3D coordinate system.
70 * @param {!o3d.Pack} pack Pack to manage created objects.
71 * @param {!o3d.DrawList} drawList DrawList to use for created
72 * materials.
73 * @param {!o3d.Transform} transform Transform under which to place
74 * created Shapes.
75 * @param {function(string, *): void}
76 * opt_completionCallback Optional completion callback which is
77 * called after the file has been processed. The URL of the file
78 * is passed as the first argument. The second argument indicates
79 * whether the file was loaded successfully (true) or not (false).
80 * The third argument is an error detail if the file failed to
81 * load successfully.
82 */
83 SVGLoader.prototype.load = function(url,
84 flipY,
85 pack,
86 drawList,
87 transform,
88 opt_completionCallback) {
89 var that = this;
90 o3djs.io.loadTextFile(url, function(text, exception) {
91 if (exception) {
92 runLater_(function() {
93 if (opt_completionCallback)
94 opt_completionCallback(url,
95 false,
96 e);
97 });
98 } else {
99 runLater_(function() {
100 try {
101 that.parse_(text,
102 flipY,
103 pack,
104 drawList,
105 transform);
106 if (opt_completionCallback)
107 opt_completionCallback(url, true, null);
108 } catch (e) {
109 if (window.console)
110 window.console.log(e);
111 if (opt_completionCallback)
112 opt_completionCallback(url, false, e);
113 }
114 });
115 }
116 });
117 };
118
119 /**
120 * Does the parsing of the SVG file.
121 * @param {string} svgText the text of the SVG file.
122 * @param {boolean} flipY Whether the Y coordinates should be flipped
123 * to better match the 3D coordinate system.
124 * @param {!o3d.Pack} pack Pack to manage created objects.
125 * @param {!o3d.DrawList} drawList DrawList to use for created
126 * materials.
127 * @param {!o3d.Transform} transform Transform under which to place
128 * created Shapes.
129 * @private
130 */
131 SVGLoader.prototype.parse_ = function(svgText,
132 flipY,
133 pack,
134 drawList,
135 transform) {
136 /**
137 * The pack in which to create shapes, materials, etc.
138 * @type {!o3d.Pack}
139 * @private
140 */
141 this.pack_ = pack;
142
143 /**
144 * Stack of matrices. Entering a new graphics context pushes a new
145 * matrix.
146 * @type {!Array.<!o3djs.math.Matrix4>}
147 * @private
148 */
149 this.matrixStack_ = [];
150
151 /**
152 * Stack of transforms. Entering a new graphics context pushes a new
153 * transform.
154 * @type {!Array.<!o3d.Transform>}
155 * @private
156 */
157 this.transformStack_ = [];
158
159 this.matrixStack_.push(o3djs.math.identity(4));
160 this.transformStack_.push(transform);
161 var parser = new SAXDriver();
162 var eventHandler = new SVGSAXHandler_(this,
163 parser,
164 flipY,
165 pack,
166 drawList);
167 parser.setDocumentHandler(eventHandler);
168 parser.parse(svgText);
169 };
170
171 /**
172 * Returns the current transform.
173 * @return {!o3d.Transform}
174 * @private
175 */
176 SVGLoader.prototype.currentTransform_ = function() {
177 var len = this.transformStack_.length;
178 return this.transformStack_[len - 1];
179 };
180
181 /**
182 * Returns the current matrix.
183 * @return {!o3djs.math.Matrix4}
184 * @private
185 */
186 SVGLoader.prototype.currentMatrix_ = function() {
187 var len = this.matrixStack_.length;
188 return this.matrixStack_[len - 1];
189 };
190
191 /**
192 * Sets the current matrix.
193 * @param {!o3djs.math.Matrix4} matrix the new current matrix.
194 * @private
195 */
196 SVGLoader.prototype.setCurrentMatrix_ = function(matrix) {
197 var length = this.matrixStack_.length;
198 this.matrixStack_[length - 1] = matrix;
199 this.transformStack_[length - 1].localMatrix = matrix;
200 };
201
202 /**
203 * Pushes a new transform / matrix pair.
204 * @private
205 */
206 SVGLoader.prototype.pushTransform_ = function() {
207 this.matrixStack_.push(o3djs.math.identity(4));
208 var xform = this.pack_.createObject('o3d.Transform');
209 xform.parent = this.currentTransform_();
210 this.transformStack_.push(xform);
211 };
212
213 /**
214 * Pops a transform / matrix pair.
215 * @private
216 */
217 SVGLoader.prototype.popTransform_ = function() {
218 this.matrixStack_.pop();
219 this.transformStack_.pop();
220 };
221
222 /**
223 * Supports the "matrix" command in the graphics context; not yet
224 * implemented.
225 * @param {number} a matrix element
226 * @param {number} b matrix element
227 * @param {number} c matrix element
228 * @param {number} d matrix element
229 * @param {number} e matrix element
230 * @param {number} f matrix element
231 * @private
232 */
233 SVGLoader.prototype.matrix_ = function(a, b, c, d, e, f) {
234 // TODO(kbr): implement
235 throw 'matrix command not yet implemented';
236 };
237
238 /**
239 * Supports the "translate" command in the graphics context.
240 * @param {number} x x translation
241 * @param {number} y y translation
242 * @private
243 */
244 SVGLoader.prototype.translate_ = function(x, y) {
245 var tmp = o3djs.math.matrix4.translation([x, y, 0]);
246 this.setCurrentMatrix_(
247 o3djs.math.mulMatrixMatrix(tmp, this.currentMatrix_()));
248 };
249
250 /**
251 * Supports the "scale" command in the graphics context; not yet
252 * implemented.
253 * @param {number} sx x scale
254 * @param {number} sy y scale
255 * @private
256 */
257 SVGLoader.prototype.scale_ = function(sx, sy) {
258 // TODO(kbr): implement
259 throw 'scale command not yet implemented';
260 };
261
262 /**
263 * Supports the "rotate" command in the graphics context.
264 * @param {number} angle angle to rotate, in degrees.
265 * @param {number} cx x component of rotation center.
266 * @param {number} cy y component of rotation center.
267 * @private
268 */
269 SVGLoader.prototype.rotate_ = function(angle, cx, cy) {
270 var rot = o3djs.math.matrix4.rotationZ(o3djs.math.degToRad(angle));
271 if (cx || cy) {
272 var xlate1 = o3djs.math.matrix4.translation([cx, cy, 0]);
273 var xlate2 = o3djs.math.matrix4.translation([-cx, -cy, 0]);
274 rot = o3djs.math.mulMatrixMatrix(xlate2, rot);
275 rot = o3djs.math.mulMatrixMatrix(rot, xlate1);
276 }
277 this.setCurrentMatrix_(rot);
278 };
279
280 /**
281 * Supports the "skewX" command in the graphics context; not yet
282 * implemented.
283 * @param {number} angle skew X angle, in degrees.
284 * @private
285 */
286 SVGLoader.prototype.skewX_ = function(angle) {
287 // TODO(kbr): implement
288 throw 'skewX command not yet implemented';
289 };
290
291 /**
292 * Supports the "skewY" command in the graphics context; not yet
293 * implemented.
294 * @param {number} angle skew Y angle, in degrees.
295 * @private
296 */
297 SVGLoader.prototype.skewY_ = function(angle) {
298 // TODO(kbr): implement
299 throw 'skewY command not yet implemented';
300 };
301
302 /**
303 * Parses the data from an SVG path element, constructing an
304 * o3djs.gpu2d.Path from it.
305 * @param {string} pathData the path's data (the "d" attribute).
306 * @param {number} lineNumber the line number of the current parse,
307 * for error reporting.
308 * @param {boolean} flipY Whether the Y coordinates should be flipped
309 * to better match the 3D coordinate system.
310 * @param {!o3d.Pack} pack Pack to manage created objects.
311 * @param {!o3d.DrawList} drawList DrawList to use for created
312 * materials.
313 * @private
314 */
315 SVGLoader.prototype.parsePath_ = function(pathData,
316 lineNumber,
317 flipY,
318 pack,
319 drawList) {
320 var parser = new PathDataParser_(this,
321 lineNumber,
322 flipY,
323 pack,
324 drawList);
325 var path = parser.parse(pathData);
326 if (this.fill_) {
327 path.setFill(this.fill_);
328 }
329 this.currentTransform_().addShape(path.shape);
330 };
331
332 /**
333 * Parses the style from an SVG path element or graphics context,
334 * preparing to set it on the next created o3djs.gpu2d.Path. If it
335 * doesn't know how to handle it, the default color of solid black
336 * will be used.
337 * @param {string} styleData the string containing the "style"
338 * attribute.
339 * @param {number} lineNumber the line number of the current parse,
340 * for error reporting.
341 * @param {!o3d.Pack} pack Pack to manage created objects.
342 * @private
343 */
344 SVGLoader.prototype.parseStyle_ = function(styleData,
345 lineNumber,
346 pack) {
347 this.fill_ = null;
348 var portions = styleData.split(';');
349 for (var i = 0; i < portions.length; i++) {
350 var keyVal = portions[i].split(':');
351 var key = keyVal[0];
352 var val = keyVal[1];
353 if (key == 'stroke') {
354 // TODO(kbr): support strokes
355 } else if (key == 'stroke-width') {
356 // TODO(kbr): support stroke width
357 } else if (key == 'fill') {
358 if (val.charAt(0) == '#') {
359 var r = parseInt(val.substr(1, 2), 16);
360 var g = parseInt(val.substr(3, 2), 16);
361 var b = parseInt(val.substr(5, 2), 16);
362 var fill = o3djs.gpu2d.createColor(pack,
363 r / 255.0,
364 g / 255.0,
365 b / 255.0,
366 1.0);
367 this.fill_ = fill;
368 } else if (val.substr(0, 4) == 'rgb(' &&
369 val.charAt(val.length - 1) == ')') {
370 var rgbStrings = val.substr(4, val.length - 5).split(',');
371 var r = parseInt(rgbStrings[0]);
372 var g = parseInt(rgbStrings[1]);
373 var b = parseInt(rgbStrings[2]);
374 var fill = o3djs.gpu2d.createColor(pack,
375 r / 255.0,
376 g / 255.0,
377 b / 255.0,
378 1.0);
379 this.fill_ = fill;
380 }
381 }
382 }
383 };
384
385 /**
386 * Parses the data from an SVG transform attribute, storing the result
387 * in the current o3d.Transform.
388 * @param {string} data the transform's data.
389 * @param {number} lineNumber the line number of the current parse,
390 * for error reporting.
391 * @private
392 */
393 SVGLoader.prototype.parseTransform_ = function(data,
394 lineNumber) {
395 var parser = new TransformDataParser_(this,
396 lineNumber);
397 parser.parse(data);
398 };
399
400 //----------------------------------------------------------------------
401 // BaseDataParser -- base class for parsers dealing with SVG data
402
403 /**
404 * Base class for parsers dealing with SVG data.
405 * @param {SVGLoader} loader The SVG loader.
406 * @param {number} lineNumber the line number of the current parse,
407 * for error reporting.
408 * @constructor
409 * @private
410 */
411 BaseDataParser_ = function(loader,
412 lineNumber) {
413 this.loader_ = loader;
414 this.lineNumber_ = lineNumber;
415 this.putBackToken_ = null;
416 }
417
418 /**
419 * Types of tokens.
420 * @enum
421 * @private
422 */
423 BaseDataParser_.TokenTypes = {
424 WORD: 1,
425 NUMBER: 2,
426 LPAREN: 3,
427 RPAREN: 4,
428 NONE: 5
429 };
430
431 /**
432 * Parses the given SVG data.
433 * @param {string} data the SVG data.
434 * @private
435 */
436 BaseDataParser_.prototype.parse = function(data) {
437 var parseState = {
438 // First index of current token
439 firstIndex: 0,
440 // Last index of current token (exclusive)
441 lastIndex: 0,
442 // Line number of the path element
443 lineNumber: this.lineNumber_
444 };
445 var done = false;
446 while (!done) {
447 var tok = this.nextToken_(parseState, data);
448 switch (tok.kind) {
449 case BaseDataParser_.TokenTypes.WORD:
450 // Allow the parsing to override the specification of the last
451 // command
452 this.setLastCommand_(tok.val);
453 this.parseWord_(parseState, data, tok.val);
454 break;
455 case BaseDataParser_.TokenTypes.NUMBER:
456 // Assume this is a repeat of the last command
457 this.putBack_(tok);
458 this.parseWord_(parseState, data, this.lastCommand_);
459 break;
460 default:
461 done = true;
462 break;
463 }
464 }
465 };
466
467 /**
468 * Sets the last parsed command.
469 * @param {string} command the last parsed command.
470 * @private
471 */
472 BaseDataParser_.prototype.setLastCommand_ = function(command) {
473 this.lastCommand_ = command;
474 };
475
476 /**
477 * Returns true if the given character is a whitespace or separator.
478 * @param {string} c the character to test.
479 * @private
480 */
481 BaseDataParser_.prototype.isWhitespaceOrSeparator_ = function(c) {
482 return (c == ',' ||
483 c == ' ' ||
484 c == '\t' ||
485 c == '\r' ||
486 c == '\n');
487 };
488
489 /**
490 * Puts back a token to be consumed during the next iteration. There
491 * is only a one-token put back buffer.
492 * @param {!{kind: BaseDataParser_TokenTypes, val: string}} tok The
493 * token to put back.
494 * @private
495 */
496 BaseDataParser_.prototype.putBack_ = function(tok) {
497 this.putBackToken_ = tok;
498 };
499
500 /**
501 * Returns the next token.
502 * @param {!{firstIndex: number, lastIndex: number, lineNumber:
503 * number}} parseState The parse state.
504 * @param {string} data The data being parsed.
505 * @private
506 */
507 BaseDataParser_.prototype.nextToken_ = function(parseState, data) {
508 if (this.putBackToken_) {
509 var tmp = this.putBackToken_;
510 this.putBackToken_ = null;
511 return tmp;
512 }
513 parseState.firstIndex = parseState.lastIndex;
514 if (parseState.firstIndex < data.length) {
515 // Eat whitespace and separators
516 while (true) {
517 var curChar = data.charAt(parseState.firstIndex);
518 if (this.isWhitespaceOrSeparator_(curChar)) {
519 ++parseState.firstIndex;
520 if (parseState.firstIndex >= data.length)
521 break;
522 } else {
523 break;
524 }
525 }
526 }
527 if (parseState.firstIndex >= data.length)
528 return { kind: BaseDataParser_.TokenTypes.NONE, val: null };
529 parseState.lastIndex = parseState.firstIndex;
530 // Surround the next token
531 var curChar = data.charAt(parseState.lastIndex++);
532 if (curChar == '-' ||
533 curChar == '.' ||
534 (curChar >= '0' && curChar <= '9')) {
535 while (true) {
536 var t = data.charAt(parseState.lastIndex);
537 if (t == '.' ||
538 (t >= '0' && t <= '9')) {
539 ++parseState.lastIndex;
540 } else {
541 break;
542 }
543 }
544 // See whether an exponential format follows: i.e. 136e-3
545 if (data.charAt(parseState.lastIndex) == 'e') {
546 ++parseState.lastIndex;
547 if (data.charAt(parseState.lastIndex) == '-') {
548 ++parseState.lastIndex;
549 }
550 while (true) {
551 var t = data.charAt(parseState.lastIndex);
552 if (t >= '0' && t <= '9') {
553 ++parseState.lastIndex;
554 } else {
555 break;
556 }
557 }
558 }
559 return { kind: BaseDataParser_.TokenTypes.NUMBER,
560 val: parseFloat(data.substring(parseState.firstIndex,
561 parseState.lastIndex)) };
562 } else if ((curChar >= 'A' && curChar <= 'Z') ||
563 (curChar >= 'a' && curChar <= 'z')) {
564 // Consume all adjacent letters -- this is satisfactory both for
565 // the grammars of the "transform" and "d" attributes
566 while (true) {
567 var t = data.charAt(parseState.lastIndex);
568 if ((t >= 'A' && t <= 'Z') ||
569 (t >= 'a' && t <= 'z')) {
570 ++parseState.lastIndex;
571 } else {
572 break;
573 }
574 }
575 return { kind: BaseDataParser_.TokenTypes.WORD,
576 val: data.substring(parseState.firstIndex,
577 parseState.lastIndex) };
578 } else if (curChar == '(') {
579 return { kind: BaseDataParser_.TokenTypes.LPAREN,
580 val: data.substring(parseState.firstIndex,
581 parseState.lastIndex) };
582 } else if (curChar == ')') {
583 return { kind: BaseDataParser_.TokenTypes.RPAREN,
584 val: data.substring(parseState.firstIndex,
585 parseState.lastIndex) };
586 }
587 throw 'Expected number or word at line ' + parseState.lineNumber;
588 };
589
590 /**
591 * Verifies that the next token is of the given kind, throwing an
592 * exception if not.
593 * @param {!{firstIndex: number, lastIndex: number, lineNumber:
594 * number}} parseState The parse state.
595 * @param {string} data The data being parsed.
596 * @param {BaseDataParser_TokenTypes} tokenType The expected token
597 * type.
598 */
599 BaseDataParser_.prototype.expect_ = function(parseState, data, tokenType) {
600 var tok = this.nextToken_(parseState, data);
601 if (tok.kind != tokenType) {
602 throw 'At line number ' + parseState.lineNumber +
603 ': expected token type ' + tokenType +
604 ', got ' + tok.kind;
605 }
606 };
607
608 /**
609 * Parses a series of floating-point numbers.
610 * @param {!{firstIndex: number, lastIndex: number, lineNumber:
611 * number}} parseState The parse state.
612 * @param {string} data The data being parsed.
613 * @param {number} numFloats The number of floating-point numbers to
614 * parse.
615 * @return {!Array.<number>}
616 * @private
617 */
618 BaseDataParser_.prototype.parseFloats_ = function(parseState,
619 data,
620 numFloats) {
621 var result = [];
622 for (var i = 0; i < numFloats; ++i) {
623 var tok = this.nextToken_(parseState, data);
624 if (tok.kind != BaseDataParser_.TokenTypes.NUMBER)
625 throw "Expected number at line " +
626 parseState.lineNumber +
627 ", character " +
628 parseState.firstIndex +
629 "; got \"" + tok.val + "\"";
630 result.push(tok.val);
631 }
632 return result;
633 };
634
635 //----------------------------------------------------------------------
636 // PathDataParser
637
638 /**
639 * Parser for the data in an SVG path.
640 * @param {SVGLoader} loader The SVG loader.
641 * @param {number} lineNumber the line number of the current parse,
642 * for error reporting.
643 * @param {boolean} flipY Whether the Y coordinates should be flipped
644 * to better match the 3D coordinate system.
645 * @param {!o3d.Pack} pack Pack to manage created objects.
646 * @param {!o3d.DrawList} drawList DrawList to use for created
647 * materials.
648 * @constructor
649 * @extends {BaseDataParser_}
650 * @private
651 */
652 PathDataParser_ = function(loader,
653 lineNumber,
654 flipY,
655 pack,
656 drawList) {
657 BaseDataParser_.call(this, loader, lineNumber);
658 this.flipY_ = flipY;
659 this.pack_ = pack;
660 this.drawList_ = drawList;
661 this.curX_ = 0;
662 this.curY_ = 0;
663 };
664
665 o3djs.base.inherit(PathDataParser_, BaseDataParser_);
666
667 /**
668 * Parses data from an SVG path.
669 * @param {string} data SVG path data.
670 * @private
671 */
672 PathDataParser_.prototype.parse = function(data) {
673 this.path_ = o3djs.gpu2d.createPath(this.pack_, this.drawList_);
674 BaseDataParser_.prototype.parse.call(this, data);
675 this.path_.update();
676 return this.path_;
677 };
678
679 /**
680 * Parses the data for the given word (command) from the path data.
681 * @param {!{firstIndex: number, lastIndex: number, lineNumber:
682 * number}} parseState The parse state.
683 * @param {string} data The data being parsed.
684 * @param {string} word The character for the current command.
685 * @private
686 */
687 PathDataParser_.prototype.parseWord_ = function(parseState,
688 data,
689 word) {
690 if (word == 'M' || word == 'm') {
691 var absolute = (word == 'M');
692 var coords = this.parseFloats_(parseState, data, 2);
693 this.doYFlip_(coords);
694 if (!absolute) {
695 this.makeAbsolute_(coords);
696 }
697 this.path_.moveTo(coords[0], coords[1]);
698 this.curX_ = coords[0];
699 this.curY_ = coords[1];
700 // This is a horrible portion of the SVG spec
701 if (absolute) {
702 this.setLastCommand_('L');
703 } else {
704 this.setLastCommand_('l');
705 }
706 } else if (word == 'L' || word == 'l') {
707 var absolute = (word == 'L');
708 var coords = this.parseFloats_(parseState, data, 2);
709 this.doYFlip_(coords);
710 if (!absolute) {
711 this.makeAbsolute_(coords);
712 }
713 this.path_.lineTo(coords[0], coords[1]);
714 this.curX_ = coords[0];
715 this.curY_ = coords[1];
716 } else if (word == 'H' || word == 'h') {
717 var absolute = (word == 'H');
718 var coords = this.parseFloats_(parseState, data, 1);
719 if (!absolute) {
720 coords[0] += this.curX_;
721 }
722 this.path_.lineTo(coords[0], this.curY_);
723 this.curX_ = coords[0];
724 } else if (word == 'V' || word == 'v') {
725 var absolute = (word == 'V');
726 var coords = this.parseFloats_(parseState, data, 1);
727 if (this.flipY_) {
728 coords[0] = -coords[0];
729 }
730 if (!absolute) {
731 coords[0] += this.curY_;
732 }
733 this.path_.lineTo(this.curX_, coords[0]);
734 this.curY_ = coords[0];
735 } else if (word == 'Q' || word == 'q') {
736 var absolute = (word == 'Q');
737 var coords = this.parseFloats_(parseState, data, 4);
738 this.doYFlip_(coords);
739 if (!absolute) {
740 this.makeAbsolute_(coords);
741 }
742 this.path_.quadraticTo(coords[0], coords[1], coords[2], coords[3]);
743 this.curX_ = coords[2];
744 this.curY_ = coords[3];
745 // TODO(kbr): support shorthand quadraticTo (T/t)
746 } else if (word == 'C' || word == 'c') {
747 var absolute = (word == 'C');
748 var coords = this.parseFloats_(parseState, data, 6);
749 this.doYFlip_(coords);
750 if (!absolute) {
751 this.makeAbsolute_(coords);
752 }
753 this.path_.cubicTo(coords[0], coords[1],
754 coords[2], coords[3],
755 coords[4], coords[5]);
756 this.curX_ = coords[4];
757 this.curY_ = coords[5];
758 // TODO(kbr): support shorthand cubicTo (S/s)
759 } else if (word == 'Z' || word == 'z') {
760 this.path_.close();
761 } else {
762 throw 'Unknown word ' + word + ' at line ' + parseState.lineNumber;
763 }
764 };
765
766 /**
767 * Negates the Y coordinates of the passed 2D coordinate list if the
768 * flipY flag is set on this parser.
769 * @param {!Array.<number>} coords Array of 2D coordinates.
770 * @private
771 */
772
773 PathDataParser_.prototype.doYFlip_ = function(coords) {
774 if (this.flipY_) {
775 for (var i = 0; i < coords.length; i += 2) {
776 coords[i+1] = -coords[i+1];
777 }
778 }
779 };
780
781 /**
782 * Transforms relative coordinates to absolute coordinates.
783 * @param {!Array.<number>} coords Array of 2D coordinates.
784 * @private
785 */
786 PathDataParser_.prototype.makeAbsolute_ = function(coords) {
787 for (var i = 0; i < coords.length; i += 2) {
788 coords[i] += this.curX_;
789 coords[i+1] += this.curY_;
790 }
791 };
792
793 //----------------------------------------------------------------------
794 // TransformDataParser
795
796 /**
797 * Parser for the data in an SVG transform.
798 * @param {SVGLoader} loader The SVG loader.
799 * @param {number} lineNumber the line number of the current parse,
800 * for error reporting.
801 * @constructor
802 * @extends {BaseDataParser_}
803 * @private
804 */
805 TransformDataParser_ = function(loader,
806 lineNumber) {
807 BaseDataParser_.call(this, loader, lineNumber);
808 };
809
810 o3djs.base.inherit(TransformDataParser_, BaseDataParser_);
811
812 /**
813 * Parses the data for the given word (command) from the path data.
814 * @param {!{firstIndex: number, lastIndex: number, lineNumber:
815 * number}} parseState The parse state.
816 * @param {string} data The data being parsed.
817 * @param {string} word The character for the current command.
818 * @private
819 */
820 TransformDataParser_.prototype.parseWord_ = function(parseState,
821 data,
822 word) {
823 if (!((word == 'matrix') ||
824 (word == 'translate') ||
825 (word == 'scale') ||
826 (word == 'rotate') ||
827 (word == 'skewX') ||
828 (word == 'skewY'))) {
829 throw 'Unknown transform definition ' + word +
830 ' at line ' + parseState.lineNumber;
831 }
832
833 this.expect_(parseState, data, BaseDataParser_.TokenTypes.LPAREN);
834 // Some of the commands take a variable number of arguments
835 var floats;
836 switch (word) {
837 case 'matrix':
838 floats = this.parseFloats_(parseState, data, 6);
839 this.loader_.matrix_(floats[0], floats[1],
840 floats[2], floats[3],
841 floats[4], floats[5]);
842 break;
843 case 'translate':
844 floats = this.parseFloats_(parseState, data, 1);
845 var tok = this.nextToken_(parseState, data);
846 if (tok.kind == BaseDataParser_.TokenTypes.NUMBER) {
847 floats.push(tok.val);
848 } else {
849 this.putBack_(tok);
850 floats.push(0);
851 }
852 this.loader_.translate_(floats[0], floats[1]);
853 break;
854 case 'scale':
855 floats = this.parseFloats_(parseState, data, 1);
856 var tok = this.nextToken_(parseState, data);
857 if (tok.kind == BaseDataParser_.TokenTypes.NUMBER) {
858 floats.push(tok.val);
859 } else {
860 this.putBack_(tok);
861 floats.push(floats[0]);
862 }
863 this.loader_.scale_(floats[0], floats[1]);
864 break;
865 case 'rotate':
866 floats = this.parseFloats_(parseState, data, 1);
867 var tok = this.nextToken_(parseState, data);
868 this.putBack_(tok);
869 if (tok.kind == BaseDataParser_.TokenTypes.NUMBER) {
870 floats = floats.concat(this.parseFloats_(parseState, data, 2));
871 } else {
872 floats.push(0);
873 floats.push(0);
874 }
875 this.loader_.rotate_(floats[0], floats[1], floats[2]);
876 break;
877 case 'skewX':
878 floats = this.parseFloats_(parseState, data, 1);
879 this.loader_.skewX_(floats[0]);
880 break;
881 case 'skewY':
882 floats = this.parseFloats_(parseState, data, 1);
883 this.loader_.skewY_(floats[0]);
884 break;
885 default:
886 throw 'Unknown word ' + word + ' at line ' + parseState.lineNumber;
887 }
888 this.expect_(parseState, data, BaseDataParser_.TokenTypes.RPAREN);
889 };
890
891 //----------------------------------------------------------------------
892 // SVGSaxHandler
893
894 /**
895 * Handler which integrates with the XMLJS SAX parser.
896 * @param {!SVGLoader} loader The SVG loader.
897 * @param {Object} parser the XMLJS SAXDriver.
898 * @param {boolean} flipY Whether the Y coordinates should be flipped
899 * to better match the 3D coordinate system.
900 * @param {!o3d.Pack} pack Pack to manage created objects.
901 * @param {!o3d.DrawList} drawList DrawList to use for created
902 * materials.
903 * @constructor
904 * @private
905 */
906 SVGSAXHandler_ = function(loader, parser, flipY, pack, drawList) {
907 this.loader_ = loader;
908 this.parser_ = parser;
909 this.flipY_ = flipY;
910 this.pack_ = pack;
911 this.drawList_ = drawList;
912 };
913
914 /**
915 * Handler for the start of an element.
916 * @param {string} name the name of the element.
917 * @param {Object} attributes the XMLJS attributes object.
918 * @private
919 */
920 SVGSAXHandler_.prototype.startElement = function(name, attributes) {
921 switch (name) {
922 case 'path':
923 var pushed = false;
924 var transformData = attributes.getValueByName('transform');
925 if (transformData) {
926 pushed = true;
927 this.loader_.pushTransform_();
928 this.loader_.parseTransform_(transformData,
929 this.parser_.getLineNumber());
930 }
931 if (attributes.getValueByName('style')) {
932 this.loader_.parseStyle_(attributes.getValueByName('style'),
933 this.parser_.getLineNumber(),
934 this.pack_);
935 }
936 this.loader_.parsePath_(attributes.getValueByName('d'),
937 this.parser_.getLineNumber(),
938 this.flipY_,
939 this.pack_,
940 this.drawList_);
941 if (pushed) {
942 this.loader_.popTransform_();
943 }
944 break;
945 case 'g':
946 this.loader_.pushTransform_();
947 if (attributes.getValueByName('style')) {
948 this.loader_.parseStyle_(attributes.getValueByName('style'),
949 this.parser_.getLineNumber(),
950 this.pack_);
951 }
952 var data = attributes.getValueByName('transform');
953 if (data) {
954 this.loader_.parseTransform_(data,
955 this.parser_.getLineNumber());
956 }
957 break;
958 default:
959 // No other commands supported yet
960 break;
961 }
962 };
963
964 /**
965 * Handler for the end of an element.
966 * @param {string} name the name of the element.
967 * @private
968 */
969 SVGSAXHandler_.prototype.endElement = function(name) {
970 if (name == 'g') {
971 this.loader_.popTransform_();
972 }
973 };
974
OLDNEW
« no previous file with comments | « samples/gpu2d/svg_thin_crescent.html ('k') | samples/gpu2d/svgsample.js » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698