OLD | NEW |
(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 <SCRIPT> 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 |
OLD | NEW |