1 /** 2 * @fileOverview 3 * primitive.gl.enchant.js 4 * @version 0.3.5 5 * @require gl.enchant.js v0.3.5+ 6 * @author UEI Corporation 7 * 8 * @description 9 * Primitive objects for gl.enchant.js 10 */ 11 12 13 if (enchant.gl !== undefined) { 14 (function() { 15 enchant.gl.primitive = {}; 16 enchant.gl.primitive.Plane = enchant.Class.create(enchant.gl.Sprite3D, { 17 initialize: function(scale) { 18 enchant.gl.Sprite3D.call(this); 19 this.mesh = enchant.gl.Mesh.createPlane(scale); 20 } 21 }); 22 enchant.gl.primitive.PlaneXY = enchant.Class.create(enchant.gl.Sprite3D, { 23 initialize: function(scale) { 24 enchant.gl.Sprite3D.call(this); 25 this.mesh = enchant.gl.Mesh.createPlaneXY(scale); 26 } 27 }); 28 enchant.gl.primitive.PlaneYZ = enchant.Class.create(enchant.gl.Sprite3D, { 29 initialize: function(scale) { 30 enchant.gl.Sprite3D.call(this); 31 this.mesh = enchant.gl.Mesh.createPlaneYZ(scale); 32 } 33 }); 34 enchant.gl.primitive.PlaneXZ = enchant.Class.create(enchant.gl.Sprite3D, { 35 initialize: function(scale) { 36 enchant.gl.Sprite3D.call(this); 37 this.mesh = enchant.gl.Mesh.createPlaneXZ(scale); 38 } 39 }); 40 enchant.gl.primitive.Billboard = enchant.Class.create(enchant.gl.primitive.Plane, { 41 initialize: function(scale) { 42 var core = enchant.Core.instance; 43 enchant.gl.primitive.Plane.call(this, scale); 44 this.addEventListener('enterframe', function() { 45 if (core.currentScene3D._camera) { 46 this.rotation = core.currentScene3D._camera.invMat; 47 } 48 }); 49 } 50 }); 51 enchant.gl.primitive.BillboardAnimation = enchant.Class.create(enchant.gl.primitive.Billboard, { 52 initialize: function(divide, scale) { 53 enchant.gl.primitive.Billboard.call(this, scale); 54 if (typeof divide !== 'undefined') { 55 this.divide = divide; 56 } else { 57 this.divide = 4; 58 } 59 this.frame = 0; 60 }, 61 frame: { 62 get: function() { 63 return this._frame; 64 }, 65 set: function(frame) { 66 this._frame = frame; 67 var left = (frame % this.divide) / this.divide; 68 var top = 1 - ((frame / this.divide) | 0) / this.divide; 69 var right = left + (1 / this.divide); 70 var bottom = top - (1 / this.divide); 71 this.mesh.texCoords = [ 72 right, top, 73 left, top, 74 left, bottom, 75 right, bottom, 76 right, top, 77 left, top, 78 left, bottom, 79 right, bottom 80 ]; 81 } 82 } 83 }); 84 enchant.gl.primitive.BillboardY = enchant.Class.create(enchant.gl.primitive.Plane, { 85 initialize: function(scale) { 86 var core = enchant.Core.instance; 87 enchant.gl.primitive.Plane.call(this, scale); 88 this.addEventListener('render', function() { 89 if (core.currentScene3D._camera) { 90 this.rotation = core.currentScene3D._camera.invMatY; 91 } 92 }); 93 } 94 }); 95 enchant.gl.primitive.Box = enchant.Class.create(enchant.gl.Sprite3D, { 96 initialize: function(sx, sy, sz) { 97 enchant.gl.Sprite3D.call(this); 98 this.mesh = enchant.gl.Mesh.createBox(sx, sy, sz); 99 } 100 }); 101 enchant.gl.primitive.Cube = enchant.Class.create(enchant.gl.Sprite3D, { 102 initialize: function(scale) { 103 enchant.gl.Sprite3D.call(this); 104 this.mesh = enchant.gl.Mesh.createCube(scale); 105 } 106 }); 107 enchant.gl.primitive.Sphere = enchant.Class.create(enchant.gl.Sprite3D, { 108 initialize: function(r, v, h) { 109 enchant.gl.Sprite3D.call(this); 110 this.mesh = enchant.gl.Mesh.createSphere(r, v, h); 111 } 112 }); 113 enchant.gl.primitive.Cylinder = enchant.Class.create(enchant.gl.Sprite3D, { 114 initialize: function(r, h, v) { 115 enchant.gl.Sprite3D.call(this); 116 this.mesh = enchant.gl.Mesh.createCylinder(r, h, v); 117 } 118 }); 119 enchant.gl.primitive.Torus = enchant.Class.create(enchant.gl.Sprite3D, { 120 initialize: function(r, r2, v, v2) { 121 enchant.gl.Sprite3D.call(this); 122 this.mesh = enchant.gl.Mesh.createTorus(r, r2, v, v2); 123 } 124 }); 125 126 var proto = Object.getPrototypeOf(enchant.gl.Mesh); 127 proto._createPlane = function(type, scale) { 128 if (typeof scale === 'undefined') { 129 scale = 0.5; 130 } 131 var mesh = new enchant.gl.Mesh(); 132 var vertices; 133 if (type === 'yz') { 134 vertices = [ 135 0.0, 1.0, 1.0, 136 0.0, -1.0, 1.0, 137 0.0, -1.0, -1.0, 138 0.0, 1.0, -1.0, 139 140 0.0, 1.0, 1.0, 141 0.0, -1.0, 1.0, 142 0.0, -1.0, -1.0, 143 0.0, 1.0, -1.0 144 ]; 145 mesh.normals = [ 146 1.0, 0.0, 0.0, 147 1.0, 0.0, 0.0, 148 1.0, 0.0, 0.0, 149 1.0, 0.0, 0.0, 150 151 -1.0, 0.0, 0.0, 152 -1.0, 0.0, 0.0, 153 -1.0, 0.0, 0.0, 154 -1.0, 0.0, 0.0 155 ]; 156 } else if (type === 'xz') { 157 vertices = [ 158 1.0, 0.0, 1.0, 159 -1.0, 0.0, 1.0, 160 -1.0, 0.0, -1.0, 161 1.0, 0.0, -1.0, 162 163 1.0, 0.0, 1.0, 164 -1.0, 0.0, 1.0, 165 -1.0, 0.0, -1.0, 166 1.0, 0.0, -1.0 167 ]; 168 mesh.normals = [ 169 0.0, -1.0, 0.0, 170 0.0, -1.0, 0.0, 171 0.0, -1.0, 0.0, 172 0.0, -1.0, 0.0, 173 174 0.0, 1.0, 0.0, 175 0.0, 1.0, 0.0, 176 0.0, 1.0, 0.0, 177 0.0, 1.0, 0.0 178 ]; 179 } else { 180 vertices = [ 181 1.0, 1.0, 0.0, 182 -1.0, 1.0, 0.0, 183 -1.0, -1.0, 0.0, 184 1.0, -1.0, 0.0, 185 186 1.0, 1.0, 0.0, 187 -1.0, 1.0, 0.0, 188 -1.0, -1.0, 0.0, 189 1.0, -1.0, 0.0 190 ]; 191 mesh.normals = [ 192 0.0, 0.0, 1.0, 193 0.0, 0.0, 1.0, 194 0.0, 0.0, 1.0, 195 0.0, 0.0, 1.0, 196 197 0.0, 0.0, -1.0, 198 0.0, 0.0, -1.0, 199 0.0, 0.0, -1.0, 200 0.0, 0.0, -1.0 201 ]; 202 } 203 for (var i = 0, l = vertices.length; i < l; i++) { 204 vertices[i] *= scale; 205 } 206 mesh.vertices = vertices; 207 mesh.colors = [ 208 1.0, 1.0, 1.0, 1.0, 209 1.0, 1.0, 1.0, 1.0, 210 1.0, 1.0, 1.0, 1.0, 211 1.0, 1.0, 1.0, 1.0, 212 213 1.0, 1.0, 1.0, 1.0, 214 1.0, 1.0, 1.0, 1.0, 215 1.0, 1.0, 1.0, 1.0, 216 1.0, 1.0, 1.0, 1.0 217 ]; 218 mesh.texCoords = [ 219 1.0, 1.0, 220 0.0, 1.0, 221 0.0, 0.0, 222 1.0, 0.0, 223 224 1.0, 1.0, 225 0.0, 1.0, 226 0.0, 0.0, 227 1.0, 0.0 228 ]; 229 mesh.indices = [ 230 0, 1, 2, 231 2, 3, 0, 232 6, 5, 4, 233 4, 7, 6 234 ]; 235 return mesh; 236 }; 237 proto.createPlane = function(scale) { 238 return this._createPlane('xy', scale); 239 }; 240 proto.createPlaneXY = function(scale) { 241 return this._createPlane('xy', scale); 242 }; 243 proto.createPlaneYZ = function(scale) { 244 return this._createPlane('yz', scale); 245 }; 246 proto.createPlaneXZ = function(scale) { 247 return this._createPlane('xz', scale); 248 }; 249 proto.createBox = function(sx, sy, sz) { 250 if (typeof sx === 'undefined') { 251 sx = 0.5; 252 } 253 if (typeof sy === 'undefined') { 254 sy = 0.5; 255 } 256 if (typeof sz === 'undefined') { 257 sz = 0.5; 258 } 259 var mesh = new enchant.gl.Mesh(); 260 var vertices = [ 261 1.0, 1.0, 1.0, 262 -1.0, 1.0, 1.0, 263 -1.0, -1.0, 1.0, 264 1.0, -1.0, 1.0, 265 266 1.0, 1.0, -1.0, 267 -1.0, 1.0, -1.0, 268 -1.0, -1.0, -1.0, 269 1.0, -1.0, -1.0, 270 271 1.0, 1.0, 1.0, 272 -1.0, 1.0, 1.0, 273 -1.0, 1.0, -1.0, 274 1.0, 1.0, -1.0, 275 276 1.0, -1.0, 1.0, 277 -1.0, -1.0, 1.0, 278 -1.0, -1.0, -1.0, 279 1.0, -1.0, -1.0, 280 281 1.0, 1.0, 1.0, 282 1.0, -1.0, 1.0, 283 1.0, -1.0, -1.0, 284 1.0, 1.0, -1.0, 285 286 -1.0, 1.0, 1.0, 287 -1.0, -1.0, 1.0, 288 -1.0, -1.0, -1.0, 289 -1.0, 1.0, -1.0 290 ]; 291 for (var i = 0, l = vertices.length; i < l; i += 3) { 292 vertices[i] *= sx; 293 vertices[i + 1] *= sy; 294 vertices[i + 2] *= sz; 295 } 296 mesh.vertices = vertices; 297 298 mesh.colors = [ 299 1.0, 1.0, 1.0, 1.0, 300 1.0, 1.0, 1.0, 1.0, 301 1.0, 1.0, 1.0, 1.0, 302 1.0, 1.0, 1.0, 1.0, 303 304 1.0, 1.0, 1.0, 1.0, 305 1.0, 1.0, 1.0, 1.0, 306 1.0, 1.0, 1.0, 1.0, 307 1.0, 1.0, 1.0, 1.0, 308 309 1.0, 1.0, 1.0, 1.0, 310 1.0, 1.0, 1.0, 1.0, 311 1.0, 1.0, 1.0, 1.0, 312 1.0, 1.0, 1.0, 1.0, 313 314 1.0, 1.0, 1.0, 1.0, 315 1.0, 1.0, 1.0, 1.0, 316 1.0, 1.0, 1.0, 1.0, 317 1.0, 1.0, 1.0, 1.0, 318 319 1.0, 1.0, 1.0, 1.0, 320 1.0, 1.0, 1.0, 1.0, 321 1.0, 1.0, 1.0, 1.0, 322 1.0, 1.0, 1.0, 1.0, 323 324 1.0, 1.0, 1.0, 1.0, 325 1.0, 1.0, 1.0, 1.0, 326 1.0, 1.0, 1.0, 1.0, 327 1.0, 1.0, 1.0, 1.0 328 ]; 329 mesh.normals = [ 330 0.0, 0.0, 1.0, 331 0.0, 0.0, 1.0, 332 0.0, 0.0, 1.0, 333 0.0, 0.0, 1.0, 334 0.0, 0.0, -1.0, 335 0.0, 0.0, -1.0, 336 0.0, 0.0, -1.0, 337 0.0, 0.0, -1.0, 338 0.0, 1.0, 0.0, 339 0.0, 1.0, 0.0, 340 0.0, 1.0, 0.0, 341 0.0, 1.0, 0.0, 342 0.0, -1.0, 0.0, 343 0.0, -1.0, 0.0, 344 0.0, -1.0, 0.0, 345 0.0, -1.0, 0.0, 346 1.0, 0.0, 0.0, 347 1.0, 0.0, 0.0, 348 1.0, 0.0, 0.0, 349 1.0, 0.0, 0.0, 350 -1.0, 0.0, 0.0, 351 -1.0, 0.0, 0.0, 352 -1.0, 0.0, 0.0, 353 -1.0, 0.0, 0.0 354 ]; 355 mesh.texCoords = [ 356 1.0, 0.0, 357 0.0, 0.0, 358 0.0, 1.0, 359 1.0, 1.0, 360 361 1.0, 0.0, 362 0.0, 0.0, 363 0.0, 1.0, 364 1.0, 1.0, 365 366 1.0, 0.0, 367 0.0, 0.0, 368 0.0, 1.0, 369 1.0, 1.0, 370 371 1.0, 0.0, 372 0.0, 0.0, 373 0.0, 1.0, 374 1.0, 1.0, 375 376 1.0, 0.0, 377 0.0, 0.0, 378 0.0, 1.0, 379 1.0, 1.0, 380 381 1.0, 0.0, 382 0.0, 0.0, 383 0.0, 1.0, 384 1.0, 1.0 385 ]; 386 var a = [ 387 0, 1, 2, 388 2, 3, 0, 389 390 2, 1, 0, 391 0, 3, 2, 392 393 2, 1, 0, 394 0, 3, 2, 395 396 0, 1, 2, 397 2, 3, 0, 398 399 0, 1, 2, 400 2, 3, 0, 401 402 2, 1, 0, 403 0, 3, 2 404 ]; 405 for (i = 0; i < 6 * 6; i++) { 406 a[i] += Math.floor(i / 6) * 4; 407 } 408 mesh.indices = a; 409 return mesh; 410 }; 411 proto.createCube = function(s) { 412 return this.createBox(s, s, s); 413 }; 414 proto.createSphere = function(r, h, v) { 415 var i, j, l; 416 if (typeof r === 'undefined') { 417 r = 1; 418 } 419 if (typeof h === 'undefined') { 420 h = 20; 421 } 422 if (typeof v === 'undefined') { 423 v = 20; 424 } 425 var mesh = new enchant.gl.Mesh(); 426 var vertices = []; 427 var texCoords = []; 428 for (i = 0; i < v; i++) { 429 for (j = 0; j < h; j++) { 430 vertices[vertices.length] = Math.sin(Math.PI * i / (v - 1)) * Math.cos(Math.PI * 2 * j / (h - 1)) * r; 431 vertices[vertices.length] = Math.cos(Math.PI * i / (v - 1)) * r; 432 vertices[vertices.length] = Math.sin(Math.PI * i / (v - 1)) * Math.sin(Math.PI * 2 * j / (h - 1)) * r; 433 texCoords[texCoords.length] = 1.0 - j / (h - 1); 434 texCoords[texCoords.length] = 1.0 - i / (v - 1); 435 } 436 } 437 mesh.vertices = vertices; 438 mesh.texCoords = texCoords; 439 var colors = []; 440 for (i = 0, l = mesh.vertices.length / 3 * 4; i < l; i++) { 441 colors[colors.length] = 1.0; 442 } 443 mesh.colors = colors; 444 mesh.normals = vertices.slice(0); 445 var indices = []; 446 447 for (i = 0; i < v - 1; i++) { 448 for (j = 0; j < h; j++) { 449 indices[indices.length] = h * (i + 1) + j; 450 indices[indices.length] = h * i + j; 451 indices[indices.length] = h * (i + 1) + (1 + j) % h; 452 indices[indices.length] = h * i + (1 + j) % h; 453 indices[indices.length] = h * (i + 1) + (1 + j) % h; 454 indices[indices.length] = h * i + j; 455 } 456 } 457 mesh.indices = indices; 458 return mesh; 459 }; 460 proto.createCylinder = function(r, h, v) { 461 if (typeof r === 'undefined') { 462 r = 0.5; 463 } 464 if (typeof h === 'undefined') { 465 h = 1; 466 } 467 if (typeof v === 'undefined') { 468 v = 20; 469 } 470 var vertices = []; 471 var indices = []; 472 var texCoords = []; 473 var normals = []; 474 vertices[vertices.length] = 0; 475 vertices[vertices.length] = h; 476 vertices[vertices.length] = 0; 477 478 normals[normals.length] = 0; 479 normals[normals.length] = 1; 480 normals[normals.length] = 0; 481 482 texCoords[texCoords.length] = 0; 483 texCoords[texCoords.length] = 1; 484 485 vertices[vertices.length] = 0; 486 vertices[vertices.length] = -h; 487 vertices[vertices.length] = 0; 488 489 normals[normals.length] = 0; 490 normals[normals.length] = -1; 491 normals[normals.length] = 0; 492 493 texCoords[texCoords.length] = 0; 494 texCoords[texCoords.length] = 0; 495 496 var cos = 0; 497 var sin = 0; 498 for (var i = 0; i < v; i++) { 499 cos = Math.cos(Math.PI * 2 * i / (v - 1)); 500 sin = Math.sin(Math.PI * 2 * i / (v - 1)); 501 502 vertices[vertices.length] = cos * r; 503 vertices[vertices.length] = h; 504 vertices[vertices.length] = sin * r; 505 506 normals[normals.length] = 0; 507 normals[normals.length] = 1; 508 normals[normals.length] = 0; 509 510 texCoords[texCoords.length] = i / (v - 1); 511 texCoords[texCoords.length] = 1; 512 513 vertices[vertices.length] = cos * r; 514 vertices[vertices.length] = -h; 515 vertices[vertices.length] = sin * r; 516 517 normals[normals.length] = 0; 518 normals[normals.length] = -1; 519 normals[normals.length] = 0; 520 521 texCoords[texCoords.length] = i / (v - 1); 522 texCoords[texCoords.length] = 0; 523 524 vertices[vertices.length] = cos * r; 525 vertices[vertices.length] = h; 526 vertices[vertices.length] = sin * r; 527 528 normals[normals.length] = cos; 529 normals[normals.length] = 0; 530 normals[normals.length] = sin; 531 532 texCoords[texCoords.length] = i / (v - 1); 533 texCoords[texCoords.length] = 1; 534 535 vertices[vertices.length] = cos * r; 536 vertices[vertices.length] = -h; 537 vertices[vertices.length] = sin * r; 538 539 normals[normals.length] = cos; 540 normals[normals.length] = 0; 541 normals[normals.length] = sin; 542 543 texCoords[texCoords.length] = i / (v - 1); 544 texCoords[texCoords.length] = 0; 545 } 546 for (i = 0; i < v - 1; i++) { 547 indices[indices.length] = 0; 548 indices[indices.length] = 2 + i * 4 + 4; 549 indices[indices.length] = 2 + i * 4 + 0; 550 indices[indices.length] = 1; 551 indices[indices.length] = 2 + i * 4 + 1; 552 indices[indices.length] = 2 + i * 4 + 5; 553 554 indices[indices.length] = 2 + i * 4 + 2; 555 indices[indices.length] = 2 + i * 4 + 6; 556 indices[indices.length] = 2 + i * 4 + 3; 557 indices[indices.length] = 2 + i * 4 + 6; 558 indices[indices.length] = 2 + i * 4 + 7; 559 indices[indices.length] = 2 + i * 4 + 3; 560 } 561 562 var mesh = new enchant.gl.Mesh(); 563 mesh.vertices = vertices; 564 mesh.indices = indices; 565 mesh.texCoords = texCoords; 566 mesh.normals = normals; 567 mesh.setBaseColor('#ffffff'); 568 return mesh; 569 }; 570 proto.createTorus = function(r, r2, v, v2) { 571 var i, j; 572 if (typeof r === 'undefined') { 573 r = 1.0; 574 } 575 if (typeof r2 === 'undefined') { 576 r2 = 0.3; 577 } 578 if (typeof v === 'undefined') { 579 v = 20; 580 } 581 if (typeof v2 === 'undefined') { 582 v2 = 20; 583 } 584 var ring = []; 585 var norm = []; 586 var rad; 587 var cos; 588 var sin; 589 for (i = 0; i < v2; i++) { 590 rad = Math.PI * 2 * i / (v2 - 1); 591 cos = Math.cos(rad); 592 sin = Math.sin(rad); 593 ring[ring.length] = 0; 594 ring[ring.length] = sin * r2; 595 ring[ring.length] = cos * r2 + r - r2; 596 norm[norm.length] = 0; 597 norm[norm.length] = sin; 598 norm[norm.length] = cos; 599 } 600 var vertices = []; 601 var normals = []; 602 var texCoords = []; 603 for (i = 0; i < v; i++) { 604 rad = Math.PI * 2 * i / (v - 1); 605 cos = Math.cos(rad); 606 sin = Math.sin(rad); 607 for (j = 0; j < ring.length; j += 3) { 608 vertices[vertices.length] = ring[j] * cos - ring[j + 2] * sin; 609 vertices[vertices.length] = ring[j + 1]; 610 vertices[vertices.length] = ring[j + 2] * cos + ring[j] * sin; 611 normals[normals.length] = norm[j] * cos - ring[j + 2] * sin; 612 normals[normals.length] = norm[j + 1]; 613 normals[normals.length] = norm[j + 2] * cos + ring[j] + sin; 614 texCoords[texCoords.length] = 1.0 - i / (v - 1); 615 texCoords[texCoords.length] = j / (ring.length - 3); 616 } 617 } 618 var indices = []; 619 var c, c2; 620 for (i = 0; i < v - 1; i++) { 621 for (j = 0; j < v2 - 1; j++) { 622 c = v2 * i + j; 623 c2 = v2 * (i + 1) + j; 624 indices[indices.length] = c; 625 indices[indices.length] = c + 1; 626 indices[indices.length] = c2; 627 indices[indices.length] = c2 + 1; 628 indices[indices.length] = c2; 629 indices[indices.length] = c + 1; 630 } 631 } 632 var mesh = new enchant.gl.Mesh(); 633 mesh.vertices = vertices; 634 mesh.normals = normals; 635 mesh.texCoords = texCoords; 636 mesh.indices = indices; 637 mesh.setBaseColor('#ffffff'); 638 639 return mesh; 640 }; 641 }()); 642 } 643