1 /**
  2  * @fileOverview
  3  * collada.gl.enchant.js
  4  * @version v0.4.1
  5  * @require enchant.js v0.6.3+
  6  * @require gl.enchant.js v0.3.5+
  7  * @author UEI Corporation
  8  * @description
  9  * Plugin to load collada format (.dae) files on gl.enchant.js
 10  *
 11  * @detail
 12  * Uses gl-matrix.js in vectors and matrix operations.
 13  * gl-matrix.js:
 14  * https://github.com/toji/gl-matrix/
 15  */
 16 if (enchant.gl !== undefined) {
 17     enchant.Core._loadFuncs['dae'] = function(src, ext, callback, onerror) {
 18         var s = enchant.gl.Sprite3D.loadCollada(src, callback, onerror);
 19         return s;
 20     };
 21     (function() {
 22         /**
 23         * Create Sprite3D from Collada data.
 24         * At present, data that has joint and animation is not supported.
 25         * In addition, vertex attributes should be triangles.
 26         * @example
 27         *   var scene = new Scene3D();
 28         *   Sprite3D.loadCollada('hoge.dae', function(model){
 29         *       scene.addChild(model);
 30         *   });
 31         * @param {String} url Collada model URL,
 32         * @param {Function} onload Callback when loading is complete.
 33         * @param {Function} onload Callback when loading is fail.
 34         * @static
 35         */
 36         enchant.gl.Sprite3D.loadCollada = function(url, onload, onerror) {
 37             if (typeof onload !== 'function') {
 38                 return;
 39             }
 40 
 41             var rootSprite = new enchant.gl.collada.RootColladaSprite3D();
 42             rootSprite.addEventListener('load', onload);
 43             rootSprite.addEventListener('error', onerror);
 44             var e = new enchant.Event(enchant.Event.ERROR);
 45 
 46             var req = new XMLHttpRequest();
 47             req.open('GET', url, true);
 48             req.onerror = function() {
 49                 e.message = 'Cannot load an asset: ' + url;
 50                 enchant.Core.instance.dispatchEvent(e);
 51                 rootSprite.dispatchEvent(e);
 52             };
 53             req.onload = function() {
 54                 try {
 55                     var maxbonenum = 6;
 56                     var lib = {};
 57                     var collada = req.responseXML.getElementsByTagName('COLLADA')[0];
 58                     for (var i = 0, l = availableLibraryFeatures.length; i < l; i++) {
 59                         lib[availableLibraryFeatures[i].libraryName] = availableLibraryFeatures[i].loadLibraryFromXML(collada, url);
 60                     }
 61                     var scene = new Scene(collada.getElementsByTagName('scene')[0]);
 62                     var rootColladaSprite3D = new enchant.gl.collada.ColladaSprite3D(lib);
 63                     var rootColladaSkeletonSprite3D = new enchant.gl.collada.ColladaSkeletonSprite3D(lib);
 64                     if (scene.visualSceneUrl) {
 65                         var visualScene = lib['visual_scenes'][scene.visualSceneUrl];
 66                         for (var nk in visualScene.nodes) {
 67                             visualScene.nodes[nk].resolveChildNodes(lib);
 68                         }
 69                         for (var k in visualScene.nodes) {
 70                             if (visualScene.nodes[k].controllerUrl) {
 71                                 var skeletonContainer = new Node(visualScene.nodes[k].xml);
 72                                 skeletonContainer.nodes = [];
 73                                 var skin = lib['controllers'][visualScene.nodes[k].controllerUrl].skin.getProcessedSkinData();
 74                                 if (visualScene.nodes[k].skeletons.length === 0) {
 75                                     for (var sk in skin.ids) {
 76                                         visualScene.nodes[k].skeletons.push(lib['nodes'][sk]);
 77                                         break;
 78                                     }
 79                                 }
 80                                 for (var key in visualScene.nodes[k].skeletons) {
 81                                     skeletonContainer.nodes[visualScene.nodes[k].skeletons[key].id] = (visualScene.nodes[k].skeletons[key]);
 82                                 }
 83                                 var bone = new enchant.gl.collada.ColladaBone(skeletonContainer, [0, 0, 0]);
 84                                 var skeleton = new enchant.gl.collada.ColladaSkeleton();
 85                                 skeleton.addChild(bone);
 86                                 skeleton.solveFKs();
 87                                 rootColladaSkeletonSprite3D.skeleton = skeleton;
 88                                 skeleton.calculateTableForIds(skin.ids);
 89                                 rootColladaSkeletonSprite3D.addColladaSkeletonSprite3DFromNode(skeletonContainer, skin, skeleton, maxbonenum);
 90                             } else {
 91                                 rootColladaSprite3D.addColladaSprite3DFromNode(visualScene.nodes[k]);
 92                             }
 93                         }
 94                     }
 95                     rootSprite.addChild(rootColladaSprite3D);
 96                     rootSprite.addChild(rootColladaSkeletonSprite3D);
 97                     rootSprite.dispatchEvent(new enchant.Event(enchant.Event.LOAD));
 98                 } catch (err) {
 99                     e.message = err.message;
100                     rootSprite.dispatchEvent(e);
101                 }
102             };
103             req.send(null);
104             return rootSprite;
105         };
106         var Unit = enchant.Class.create({
107             initialize: function(xml) {
108                 this.xml = xml;
109                 this._datas = {};
110                 this.loadParams();
111             },
112             loadParams: function() {
113                 for (var i = 0, l = this._childable.length; i < l; i++) {
114                     var nodes = [];
115                     var param = this._childable[i];
116                     if (this.xml !== undefined) {
117                         for (var j = 0, k = this.xml.childNodes.length; j < k; j++) {
118                             if (this.xml.childNodes[j].nodeName === param) {
119                                 nodes.push(this.xml.childNodes[j]);
120                             }
121                         }
122                     }
123                     this._datas[param] = nodes;
124                 }
125             },
126             parseFloatArray: function(element) {
127                 var array = [];
128                 var floatStrings = (element.textContent).split(/\s+/);
129                 for (var k = 0; k < floatStrings.length; k++) {
130                     var value = parseFloat(floatStrings[k]);
131                     if (!isNaN(value)) {
132                         array.push(value);
133                     }
134                 }
135                 return array;
136             },
137             getParentDirectory: function(path) {
138                 return path.substring(0, path.lastIndexOf('/') + 1);
139             },
140             getReferenceAttribute: function(element, attribute) {
141                 return element.getAttribute(attribute).replace('#', '');
142             },
143             getReferenceTextContent: function(element) {
144                 return element.textContent.replace('#', '');
145             }
146         });
147         Unit.prototype._childable = [];
148         var Scene = enchant.Class.create(Unit, {
149             initialize: function(xml) {
150                 Unit.call(this, xml);
151                 if (this._datas['instance_visual_scene']) {
152                     this.visualSceneUrl = this.getReferenceAttribute(this._datas['instance_visual_scene'][0], 'url');
153                 }
154             }
155         });
156         Scene.prototype._childable = ['instance_visual_scene'];
157         var Image = enchant.Class.create(Unit, {
158             initialize: function(xml, url) {
159                 Unit.call(this, xml);
160                 this.initFrom = '';
161                 if (this._datas['init_from']) {
162                     this.initFrom = this._datas['init_from'][0].textContent;
163                     if (this.initFrom.substr(0, 4) ==='file') {
164                         var spl=this.initFrom.split('/');
165                         this.initFrom=spl[spl.length-1];
166                     }
167                     if (this.initFrom.substr(0, 4) !== 'http') {
168                         if (this.initFrom.substr(0, 2) === './') {
169                             this.initFrom = this.initFrom.substr(2, this.initFrom.length - 2);
170                         }
171                         this.initFrom = this.getParentDirectory(url) + this.initFrom;
172                     }
173                 }
174             }
175         });
176         Image.prototype._childable = ['renderable', 'init_from', 'create_2d', 'create_3d', 'create_map'];
177         var Geometry = enchant.Class.create(Unit, {
178             initialize: function(xml) {
179                 Unit.call(this, xml);
180                 if (this._datas['mesh']) {
181                     this.Mesh = new GeometryMesh(this._datas['mesh'][0]);
182                 }
183             }
184         });
185         Geometry.prototype._childable = ['asset', 'convex_mesh', 'mesh', 'spline', 'extra'];
186         var GeometryMesh = enchant.Class.create(Unit, {
187             initialize: function(xml) {
188                 Unit.call(this, xml);
189                 this.srcs = [];
190                 this.srcs.offset = null;
191                 this.vertices = [];
192                 this.triangles = [];
193                 if (this._datas['source']) {
194                     for (var i = 0, l = this._datas['source'].length; i < l; i++) {
195                         this.srcs[this._datas['source'][i].getAttribute('id')] = this.parseFloatArray(this._datas['source'][i].getElementsByTagName('float_array')[0]);
196                     }
197                 }
198                 if (this._datas['vertices']) {
199                     this.vertices = new Vertices(this._datas['vertices'][0]);
200                 }
201                 if (this._datas['triangles'].length > 0) {
202                     this.triangles = [];
203                     for (var ti = 0; ti < this._datas['triangles'].length; ti++) {
204                         this.triangles[ti] = new Triangle(this._datas['triangles'][ti], this.srcs, this.vertices);
205                     }
206                 }
207                 if (this._datas['polylist'].length > 0) {
208                     this.triangles = [];
209                     for (var pi = 0; pi < this._datas['polylist'].length; pi++) {
210                         this.triangles[pi] = new Polylist(this._datas['polylist'][pi], this.srcs, this.vertices);
211                     }
212                 }
213             }
214         });
215         GeometryMesh.prototype._childable = ['source', 'vertices', 'lines', 'linestripes', 'polygons', 'polylist', 'triangles', 'trifans', 'tristrips', 'extra'];
216         var Vertices = enchant.Class.create(Unit, {
217             initialize: function(xml) {
218                 Unit.call(this, xml);
219                 this.source = {};
220                 for (var i = 0; i < this._datas['input'].length; i++) {
221                     var input = this._datas['input'][i];
222                     this.source[input.getAttribute('semantic')] = this.getReferenceAttribute(input, 'source');
223                 }
224                 this.id = xml.getAttribute('id');
225                 this.position = this.source['POSITION'];
226             }
227         });
228         Vertices.prototype._childable = ['input', 'extras'];
229         var AbstractGeometryMeshTriangleData = enchant.Class.create(Unit, {
230             initialize: function(xml, srcs, vertices) {
231                 Unit.call(this, xml);
232                 this.inputs = [];
233                 this.primitives = [];
234                 this.stride = 0;
235                 this.material = this.xml.getAttribute('material');
236                 if (this._datas['input']) {
237                     for (var i = 0, l = this._datas['input'].length; i < l; i++) {
238                         var sourceId = this.getReferenceAttribute(this._datas['input'][i], 'source');
239                         var offset = parseInt(this._datas['input'][i].getAttribute('offset'), 10);
240                         var set = parseInt(this._datas['input'][i].getAttribute('set'), 10);
241                         if (srcs[sourceId]) {
242                             this._addSource(srcs, sourceId, this._datas['input'][i].getAttribute('semantic'));
243                             this.inputs[this._datas['input'][i].getAttribute('semantic') + 'offset'] = offset;
244                             if (!isNaN(set)) {
245                                 this.inputs[this._datas['input'][i].getAttribute('semantic') + 'set'] = set;
246                             } else {
247                                 this.inputs[this._datas['input'][i].getAttribute('semantic') + 'set'] =0;
248                             }
249                         } else if (vertices.id === sourceId) {
250                             for (var key in vertices.source) {
251                                 if (srcs[vertices.source[key]]) {
252                                     this._addSource(srcs, vertices.source[key], key);
253                                     this.inputs[key + 'offset'] = offset;
254                                 }
255                             }
256                         }
257                         this.stride = Math.max(this.stride, (offset + 1));
258                     }
259                 }
260                 if (this._datas['p']) {
261                     if (this._datas['p'][0]) {
262                         this._calculatePrimitives(this.parseFloatArray(this._datas['p'][0]));
263                     }
264                 }
265             },
266             _calculatePrimitives: function(pointArray) {},
267             _addSource: function(sources, sourceId, inputId) {
268                 var source = sources[sourceId];
269                 this.inputs[inputId] = source;
270                 if (inputId === 'TEXCOORD') {
271                     for (var sj = 1; sj < source.length; sj += 2) {
272                         this.inputs[inputId][sj] = source[sj];
273                     }
274                 }
275             }
276         });
277         var Triangle = enchant.Class.create(AbstractGeometryMeshTriangleData, {
278             _calculatePrimitives: function(pointArray) {
279                 this.primitives = pointArray;
280             }
281         });
282         Triangle.prototype._childable = ['input', 'p'];
283         var Polylist = enchant.Class.create(AbstractGeometryMeshTriangleData, {
284             _calculatePrimitives: function(primitives) {
285                 var vcount = this.parseFloatArray(this._datas['vcount'][0]);
286                 var num = 0;
287                 var last = 0;
288                 var triangles = [];
289                 var first = true;
290                 for (var i = 0, l = primitives.length / this.stride; i < l; i++) {
291                     if (first) {
292                         for (var j = 0; j < this.stride; j++) {
293                             triangles.push(primitives[i * this.stride + j]);
294                         }
295                     } else {
296                         for (var lj = 0; lj < this.stride; lj++) {
297                             triangles.push(primitives[last * this.stride + lj]);
298                         }
299                         for (var prej = 0; prej < this.stride; prej++) {
300                             triangles.push(primitives[(i - 1) * this.stride + prej]);
301                         }
302                         for (var cj = 0; cj < this.stride; cj++) {
303                             triangles.push(primitives[i * this.stride + cj]);
304                         }
305                     }
306                     if (i - last === 2) {
307                         first = false;
308                     }
309                     if (vcount[num] - 1 === i - last) {
310                         this.primitives = this.primitives.concat(triangles);
311                         last += vcount[num];
312                         num += 1;
313                         triangles = [];
314                         first = true;
315                     }
316                 }
317             }
318         });
319         Polylist.prototype._childable = ['input', 'p', 'vcount'];
320         var VisualScene = enchant.Class.create(Unit, {
321             initialize: function(xml) {
322                 Unit.call(this, xml);
323                 this.nodes = [];
324                 for (var i = 0, l = this._datas['node'].length; i < l; i++) {
325                     this.nodes[this._datas['node'][i].getAttribute('id')] = new Node(this._datas['node'][i]);
326                 }
327             }
328         });
329         VisualScene.prototype._childable = ['asset', 'node', 'evaluate_scene', 'extra'];
330         var Node = enchant.Class.create(Unit, {
331             initialize: function(xml) {
332                 this.nodes = [];
333                 this.childNodeIds = [];
334                 this.skeletons = [];
335                 this.poses = [];
336                 this.skeletonChildNodeIds = [];
337                 this.translate = [0, 0, 0];
338                 this.rotate = [];
339                 this.nMatrix = [];
340                 Unit.call(this, xml);
341                 if (xml) {
342                     if (xml.getAttribute('sid')) {
343                         this.sid = xml.getAttribute('sid');
344                     } else {
345                         this.sid = xml.getAttribute('id');
346                     }
347                     this.id = xml.getAttribute('id');
348                     this.type = xml.getAttribute('type');
349                     for (var i = 0, l = this._datas['node'].length; i < l; i++) {
350                         this.nodes[this._datas['node'][i].getAttribute('id')] = new Node(this._datas['node'][i]);
351                     }
352                     if (this._datas['translate'].length > 0) {
353                         this.translate = this.parseFloatArray(this._datas['translate'][0]);
354                     }
355                     for (var ri = 0, rl = this._datas['rotate'].length; ri < rl; ri++) {
356                         this.rotate[this._datas['rotate'][ri].getAttribute('sid')] = this.parseFloatArray(this._datas['rotate'][ri]);
357                     }
358                     for (var mi = 0, ml = this._datas['matrix'].length; mi < ml; mi++) {
359                         this.nMatrix[this._datas['matrix'][mi].getAttribute('sid')] = mat4.transpose(this.parseFloatArray(this._datas['matrix'][0]));
360                     }
361                     var materialNode = null;
362                     if (this._datas['instance_geometry'].length > 0) {
363                         materialNode = this._datas['instance_geometry'][0];
364                         this.geometryUrl = this.getReferenceAttribute(materialNode, 'url');
365                     } else if (this._datas['instance_controller'].length > 0) {
366                         materialNode = this._datas['instance_controller'][0];
367                         this.controllerUrl = this.getReferenceAttribute(materialNode, 'url');
368                         var skels = this._datas['instance_controller'][0].getElementsByTagName('skeleton');
369                         for (i = 0, l = skels.length; i < l; i++) {
370                             this.skeletonChildNodeIds[i] = this.getReferenceTextContent(skels[i]);
371                         }
372                     }
373                     if (this._datas['instance_node']) {
374                         for (i = 0, l = this._datas['instance_node'].length; i < l; i++) {
375                             this.childNodeIds[i] = this.getReferenceAttribute(this._datas['instance_node'][i], 'url');
376                         }
377                     }
378                     if (materialNode != null) {
379                         var material = materialNode.getElementsByTagName('instance_material');
380                         if (material) {
381                             this.materialTarget = {};
382                             for (i = 0; i < material.length; i++) {
383                                 this.materialTarget[material[i].getAttribute('symbol')] = this.getReferenceAttribute(material[i], 'target');
384                             }
385                         }
386                     }
387                 }
388             },
389             resolveChildNodes: function(lib) {
390                 this.__resolveChildNodes(lib, this.childNodeIds, this.nodes);
391                 var libNodes = lib['nodes'];
392                 var libVisualScene = lib['visual_scenes'];
393                 for (var key in this.skeletonChildNodeIds) {
394                     var element = null;
395                     for (var nodeKey in libNodes) {
396                         if (libNodes[nodeKey]._getNodeInHirachy(this.skeletonChildNodeIds[key])) {
397                             element = libNodes[nodeKey];
398                         }
399                         if (element != null) {break;}
400                     }
401                     for (nodeKey in libVisualScene) {
402                         if (element != null) {break;}
403                         if (libVisualScene[nodeKey]._getNodeInHirachy(this.skeletonChildNodeIds[key])) {
404                             element = libNodes[nodeKey];
405                         }
406                     }
407                     if (element != null) {
408                         element.resolveChildNodes(lib);
409                         this.skeletons.push(element);
410                     }
411                 }
412                 for (var i = 0; i < this.skeletons.length; i++) {
413                     for (var j = i + 1; j < this.skeletons.length; j++) {
414                         if (this.skeletons[i]._getNodeInHirachy(this.skeletons[j].id)) {
415                             this.skeletons.splice(j, 1);
416                             j--;
417                         } else if (this.skeletons[j]._getNodeInHirachy(this.skeletons[i].id)) {
418                             this.skeletons.splice(i, 1);
419                             i--;
420                             break;
421                         }
422                     }
423                 }
424             },
425             __resolveChildNodes: function(lib, childNodeIds, childArray) {
426                 for (var key in childArray) {
427                     childArray[key].resolveChildNodes(lib);
428                 }
429                 var libNodes = lib['nodes'];
430                 var libVisualScene = lib['visual_scenes'];
431                 for (var i = 0; i < childNodeIds.length;) {
432                     var element = null;
433                     for (key in libNodes) {
434                         element = libNodes[key]._getNodeInHirachy(childNodeIds[i]);
435                         if (element != null) {break;}
436                     }
437                     for (key in libVisualScene) {
438                         if (element != null) {break;}
439                         var nodes = libVisualScene[key].nodes;
440                         for (var nodeKey in nodes) {
441                             element = nodes[nodeKey]._getNodeInHirachy(childNodeIds[i]);
442                             if (element != null) {break;}
443                         }
444                     }
445                     if (element != null) {
446                         childArray[element.id] = new Node(element.xml);
447                         childArray[element.id].resolveChildNodes(lib);
448                         childNodeIds.splice(i, 1);
449                     } else {
450                         i++;
451                     }
452                 }
453             },
454             _getNodeInHirachy: function(id) {
455                 if (this.id === id) {return this;}
456                 var child = null;
457                 for (var k in this.nodes) {
458                     child = this.nodes[k]._getNodeInHirachy(id);
459                     if (child != null) {break;}
460                 }
461                 return child;
462             },
463             getRotationMatrix: function() {
464                 var rotation = mat4.create();
465                 mat4.identity(rotation);
466                 for (var rotationSid in this.rotate) {
467                     var rotationVec = this.rotate[rotationSid];
468                     var mat = new enchant.gl.Quat(rotationVec[0], rotationVec[1], rotationVec[2], rotationVec[3] * Math.PI / 180);
469                     mat4.multiply(rotation, mat.toMat4(mat4.create()));
470                 }
471                 return rotation;
472             },
473             getTranslationMatrix: function() {
474                 var translation = mat4.create();
475                 mat4.identity(translation);
476                 var position = this.translate;
477                 mat4.translate(translation, [position[0], position[1], position[2]]);
478                 return translation;
479             },
480             getnMatrix: function() {
481                 var matrix = mat4.create();
482                 mat4.identity(matrix);
483                 for (var matrixSid in this.nMatrix) {
484                     mat4.multiply(matrix, this.nMatrix[matrixSid]);
485                 }
486                 return matrix;
487             },
488             getNode: function() {
489                 var table = this.nodes;
490                 var child = [];
491                 for (var key in table){
492                     child[key] = [];
493                     child[key] = table[key].getNode();
494                     for(var ckey in child[key]){
495                         table[ckey] = child[key][ckey];
496                     }
497                 }
498                 return table;
499             },
500             getAnimationMatrixFromOneAnimationNode: function(Animation, libAnimationClips, flag) {
501                 var core = enchant.Core.instance;
502                 var rotation = this.getRotationMatrix();
503                 var translation = this.getTranslationMatrix();
504                 var matrix = this.getnMatrix();
505                 var animationMatrixes = [];
506                 var sid = this.sid;
507                 animationMatrixes[sid] = [];
508                 animationMatrixes[sid][0] = mat4.multiply(translation, rotation, mat4.create());
509                 mat4.multiply(animationMatrixes[sid][0], matrix);
510                 var output = [];
511                 var input = [];
512                 var length = 0;
513                 for (var ci = 0, cl = Animation.channels.length; ci < cl; ci++) {
514                     if (this.id === Animation.channels[ci].target.split('/')[0]) {
515                         var currentLength = Animation.samplers[Animation.channels[ci].samplerId].lerpedinput.length;
516                         length = Math.max(currentLength, length);
517                         if (Animation.samplers[Animation.channels[ci].samplerId].lerpedinput.length === length) {
518                             input = Animation.samplers[Animation.channels[ci].samplerId].lerpedinput;
519                         }
520                         output[Animation.channels[ci].target.split('/')[1].split('.')[0]] = Animation.samplers[Animation.channels[ci].samplerId].lerpedoutput;
521                     }
522                 }
523                 for (var i = 0, l = length; i < l; i++) {
524                     var rot = mat4.create();
525                     var trans = mat4.create();
526                     var nMat = mat4.create();
527                     mat4.identity(rot);
528                     mat4.identity(trans);
529                     mat4.identity(nMat);
530                     mat4.translate(trans, this.translate);
531                     for (var rkey in this.rotate) {
532                         var tmpf = false;
533                         var mat;
534                         for (var okey in output) {
535                             if (rkey === okey) {
536                                 mat = new enchant.gl.Quat(this.rotate[rkey][0], this.rotate[rkey][1], this.rotate[rkey][2], output[okey][i] * Math.PI / 180);
537                                 mat4.multiply(rot, mat.toMat4(mat4.create()));
538                                 tmpf = true;
539                             }
540                         }
541                         if (!tmpf) {
542                             mat = new enchant.gl.Quat(this.rotate[rkey][0], this.rotate[rkey][1], this.rotate[rkey][2], this.rotate[rkey][3] * Math.PI / 180);
543                             mat4.multiply(rot, mat.toMat4(mat4.create()));
544                         }
545                     }
546                     for (var okey2 in output) {
547                         if (okey2 === 'translation') {
548                             mat4.identity(trans);
549                             mat4.translate(trans, [output[okey2][i * 3], output[okey2][i * 3 + 1], output[okey2][i * 3 + 2]]);
550                         } else if (okey2 === 'scale') {
551                             //TODO !
552                         } else if (okey2 === 'matrix') {
553                             var tmpMat = [];
554                             for (var j = 0; j < 16; j++) {
555                                 tmpMat.push(output[okey2][i*16+j]);
556                             }
557                             mat4.transpose(tmpMat);
558                             mat4.multiply(nMat,tmpMat);
559                         } else {
560                             for (var mkey in this.nMatrix) {
561                                 if (okey2.indexOf('(')!==-1) {
562                                     if (mkey === okey2.split('(')[0]) {
563                                         if (!isNaN(output[okey2][i])) {
564                                             nMat[parseInt(okey2.split('(')[1].split(')')[0], 10) * 4 + parseInt(okey2.split(')(')[1].split(')')[0], 10)] = output[okey2][i];
565                                         } else {
566                                             nMat[parseInt(okey2.split('(')[1].split(')')[0], 10) * 4 + parseInt(okey2.split(')(')[1].split(')')[0], 10)] = output[okey2][0];
567                                         }
568                                     }
569                                 } else {
570                                     var tmpMatrix = [];
571                                     for (var oj = 0; oj < 16; oj++) {
572                                         tmpMatrix.push(output[okey2][i * 16 + oj]);
573                                     }
574                                     mat4.transpose(tmpMatrix);
575                                     mat4.multiply(nMat, tmpMatrix);
576                                 }
577                             }
578                         }
579                     }
580                     animationMatrixes[sid][Math.round(core.fps * input[i])] = mat4.multiply(trans, rot, mat4.create());
581                     mat4.multiply(animationMatrixes[sid][Math.round(core.fps * input[i])],nMat);
582                 }
583                 if (Animation.animations.length > 0) {
584                     var child = this.getAnimationMatrixesLocal(Animation.animations, libAnimationClips, true);
585                     for (var ccl in child) {
586                         animationMatrixes[ccl] = child[ccl];
587                     }
588                 }
589                 return animationMatrixes;
590             },
591             getAnimationMatrixesLocal: function(libAnimations, libAnimationClips, flag) {
592                 var core = enchant.Core.instance;
593                 var rotation = this.getRotationMatrix();
594                 var translation = this.getTranslationMatrix();
595                 var matrix = this.getnMatrix();
596                 var animationMatrixes = [];
597                 animationMatrixes[this.sid] = [];
598                 animationMatrixes[this.sid][0] = mat4.multiply(translation, rotation, mat4.create());
599                 mat4.multiply(animationMatrixes[this.sid][0], matrix);
600                 var output = [];
601                 var input = [];
602                 var length = 0;
603                 for (var key in libAnimations) {
604                     for (var ci = 0, cl = libAnimations[key].channels.length; ci < cl; ci++) {
605                         if (this.id === libAnimations[key].channels[ci].target.split('/')[0]) {
606                             var currentLength = libAnimations[key].samplers[libAnimations[key].channels[ci].samplerId].lerpedinput.length;
607                             length = Math.max(currentLength, length);
608                             if (libAnimations[key].samplers[libAnimations[key].channels[ci].samplerId].lerpedinput.length === length) {
609                                 input = libAnimations[key].samplers[libAnimations[key].channels[ci].samplerId].lerpedinput;
610                             }
611                             output[libAnimations[key].channels[ci].target.split('/')[1].split('.')[0]] = libAnimations[key].samplers[libAnimations[key].channels[ci].samplerId].lerpedoutput;
612                         }
613                     }
614                     for (var ackey in libAnimationClips) {
615                         if (libAnimationClips[ackey].urls.indexOf(key) > -1 && flag) {
616                             length = 0;
617                         } 
618                     }
619                     if (libAnimations[key].animations.length > 0 && length > 0) {
620                         var child = this.getAnimationMatrixesLocal(libAnimations[key].animations, libAnimationClips, true);
621                         for (var ckey in child) {
622                             animationMatrixes[ckey] = child[ckey];
623                         }
624                     }
625                 }
626                 for (var i = 0, l = length; i < l; i++) {
627                     var rot = mat4.create();
628                     var trans = mat4.create();
629                     var nMat = mat4.create();
630                     mat4.identity(rot);
631                     mat4.identity(trans);
632                     mat4.identity(nMat);
633                     mat4.translate(trans, this.translate);
634                     for (var rkey in this.rotate) {
635                         var tmpf = false;
636                         var mat;
637                         for (var okey in output) {
638                             if (rkey === okey) {
639                                 mat = new enchant.gl.Quat(this.rotate[rkey][0], this.rotate[rkey][1], this.rotate[rkey][2], output[okey][i] * Math.PI / 180);
640                                 mat4.multiply(rot, mat.toMat4(mat4.create()));
641                                 tmpf = true;
642                             }
643                         }
644                         if (!tmpf) {
645                             mat = new enchant.gl.Quat(this.rotate[rkey][0], this.rotate[rkey][1], this.rotate[rkey][2], this.rotate[rkey][3] * Math.PI / 180);
646                             mat4.multiply(rot, mat.toMat4(mat4.create()));
647                         }
648                     }
649                     for (var okey2 in output) {
650                         if (okey2 === 'translation') {
651                             mat4.identity(trans);
652                             mat4.translate(trans, [output[okey2][i * 3], output[okey2][i * 3 + 1], output[okey2][i * 3 + 2]]);
653                         } else if (okey2 === 'scale') {
654                             //TODO !
655                         } else if (okey2 === 'matrix') {
656                             var tmpMat = [];
657                             for (var j = 0; j < 16; j++) {
658                                 tmpMat.push(output[okey2][i*16+j]);
659                             }
660                             mat4.transpose(tmpMat);
661                             mat4.multiply(nMat,tmpMat);
662                         } else {
663                             for (var mkey in this.nMatrix) {
664                                 if (okey2.indexOf('(')!==-1) {
665                                     if (mkey === okey2.split('(')[0]) {
666                                         if (!isNaN(output[okey2][i])) {
667                                             nMat[parseInt(okey2.split('(')[1].split(')')[0], 10) * 4 + parseInt(okey2.split(')(')[1].split(')')[0], 10)] = output[okey2][i];
668                                         } else {
669                                             nMat[parseInt(okey2.split('(')[1].split(')')[0], 10) * 4 + parseInt(okey2.split(')(')[1].split(')')[0], 10)] = output[okey2][0];
670                                         }
671                                     }
672                                 } else {
673                                     var tmpMatrix = [];
674                                     for (var oj = 0; oj < 16; oj++) {
675                                         tmpMatrix.push(output[okey2][i * 16 + oj]);
676                                     }
677                                     mat4.transpose(tmpMatrix);
678                                     mat4.multiply(nMat, tmpMatrix); 
679                                 }
680                             }
681                         }
682                     }
683                     animationMatrixes[this.sid][Math.round(core.fps * input[i])] = mat4.multiply(trans, rot, mat4.create());
684                     mat4.multiply(animationMatrixes[this.sid][Math.round(core.fps * input[i])],nMat);
685                 }
686                 for (var k in this.nodes) {
687                     var childmat = this.nodes[k].getAnimationMatrixesLocal(libAnimations, libAnimationClips, true);
688                     for (l in childmat) {
689                         animationMatrixes[l] = childmat[l];
690                     }
691                 }
692                 return animationMatrixes;
693             },
694             getAnimationMatrixesLocalFromAnimationClips: function(libAnimations, libAnimationClips) {
695                 var animationMatrixClips = [];
696                 for (var ackey in libAnimationClips) {
697                     var urls = libAnimationClips[ackey].urls;
698                     animationMatrixClips[ackey] = [];
699                     for (var ui = 0, ul = urls.length; ui < ul; ui++) {
700                         var child = [];
701                         if (libAnimations[urls[ui]].channels[0]) {
702                             child = this.getNode()[libAnimations[urls[ui]].channels[0].target.split('/')[0]].getAnimationMatrixFromOneAnimationNode(libAnimations[urls[ui]], libAnimationClips, true);
703                         }
704                         var child2 = [];
705                         if (libAnimations[urls[ui]].animations[0]) {
706                             child2 = this.getAnimationMatrixesLocal(libAnimations[urls[ui]].animations, libAnimationClips, true);
707                         }
708                         for (var l in child) {
709                             animationMatrixClips[ackey][l] = child[l];
710                         }
711                         for (l in child2) {
712                             animationMatrixClips[ackey][l] = child2[l];
713                         }
714                     }
715                 }
716                 return animationMatrixClips;
717             }
718         });
719         Node.prototype._childable = ['node', 'Lookat', 'matrix', 'rotate', 'scale', 'skew', 'translate', 'instance_camera', 'instance_controller', 'instance_geometry', 'instance_light', 'instance_node', 'extra'];
720         var Material = enchant.Class.create(Unit, {
721             initialize: function(xml) {
722                 Unit.call(this, xml);
723                 this.effectUrl = this.getReferenceAttribute(this._datas['instance_effect'][0], 'url');
724             }
725         });
726         Material.prototype._childable = ['asset', 'instance_effect', 'extra'];
727         var Animation = enchant.Class.create(Unit, {
728             initialize: function(xml) {
729                 Unit.call(this, xml);
730                 this.srcs = [];
731                 this.channels = [];
732                 this.samplers = [];
733                 this.animations = [];
734                 for (var ai = 0, al = this._datas['animation'].length; ai < al; ai++) {
735                     this.animations[ai] = new Animation(this._datas['animation'][ai]);
736                 }
737                 for (var ci = 0, cl = this._datas['channel'].length; ci < cl; ci++) {
738                     this.channels[ci]={};
739                     this.channels[ci].target = this.getReferenceAttribute(this._datas['channel'][ci], 'target');
740                     this.channels[ci].samplerId = this.getReferenceAttribute(this._datas['channel'][ci], 'source');
741                 }
742                 for (var i = 0, l = this._datas['source'].length; i < l; i++) {
743                     if (this._datas['source'][i].getElementsByTagName('float_array')[0]) {
744                         this.srcs[this._datas['source'][i].getAttribute('id')] = this.parseFloatArray(this._datas['source'][i].getElementsByTagName('float_array')[0]);
745                     }
746                 }
747                 for (var si = 0, sl = this._datas['sampler'].length; si < sl; si++) {
748                     this.samplers[this._datas['sampler'][si].getAttribute('id')] = new Sampler(this._datas['sampler'][si], this.srcs);
749                 }
750                 if (this._datas['sampler'].length > 0) {
751                     this.sampler = new Sampler(this._datas['sampler'][0], this.srcs);
752                 }
753             }
754         });
755         Animation.prototype._childable = ['asset', 'animation', 'source', 'sampler', 'channel', 'extra'];
756         var AnimationClip = enchant.Class.create(Unit, {
757            initialize: function(xml) {
758                Unit.call(this, xml);
759                this.urls = [];
760                for (var i = 0, l = this._datas['instance_animation'].length; i < l; i++) {
761                    this.urls.push(this._datas['instance_animation'][i].getAttribute('url').replace('#', ''));
762                }
763            } 
764         });
765         AnimationClip.prototype._childable = ['asset','instance_animation','instance_formula','extra'];
766         var Sampler = enchant.Class.create(Unit, {
767             initialize: function(xml, srcs) {
768                 Unit.call(this, xml);
769                 this.input = [];
770                 this.output = [];
771                 this.lerpedinput =[];
772                 this.lerpedoutput = [];
773                 for (var i = 0, l = this._datas['input'].length; i < l; i++) {
774                     if (this._datas['input'][i].getAttribute('semantic') === 'OUTPUT') {
775                         this.output = srcs[this.getReferenceAttribute(this._datas['input'][i], 'source')];
776                     }
777                     if (this._datas['input'][i].getAttribute('semantic') === 'INPUT') {
778                         this.input = srcs[this.getReferenceAttribute(this._datas['input'][i], 'source')];
779                     }
780                 }
781                 var stride = this.output.length / this.input.length;
782                 var ll = Math.floor(1 + this.input[this.input.length - 1] * enchant.Core.instance.fps);
783                 for (var li = 0; li < ll; li++) {
784                     this.lerpedinput.push(li / enchant.Core.instance.fps);
785                     for (var j = 0; j < stride; j++) {
786                         for (var post = 0, iil = this.input.length; post < iil; post++ ) {
787                             if (li / enchant.Core.instance.fps < this.input[post]) {
788                                 break;
789                             }
790                         }
791                         if (this.output[post * stride + j]) {
792                             this.lerpedoutput.push(this.output[(post - 1) * stride + j] + 1 / (this.input[post] - this.input[post - 1]) * (li / enchant.Core.instance.fps - this.input[post - 1]) * (this.output[post * stride + j] - this.output[(post - 1) * stride + j]));
793                         } else{
794                             this.lerpedoutput.push(this.output[(post - 1) * stride + j]);
795                         }
796                     }
797                 }
798             }
799         });
800         Sampler.prototype._childable = ['input'];
801         var Controller = enchant.Class.create(Unit, {
802             initialize: function(xml) {
803                 Unit.call(this, xml);
804                 for (var i = 0, l = this._datas['skin'].length; i < l; i++) {
805                     this.skin = new Skin(this._datas['skin'][i]);
806                 }
807             }
808         });
809         Controller.prototype._childable = ['asset', 'skin', 'morph', 'extra'];
810         var Skin = enchant.Class.create(Unit, {
811             initialize: function(xml) {
812                 Unit.call(this, xml);
813                 this.source = this.getReferenceAttribute(xml, 'source');
814                 this.sources = [];
815                 var child;
816                 for (var i = 0, l = this._datas['source'].length; i < l; i++) {
817                     var source = this._datas['source'][i];
818                     for (var j = 0, m = source.childNodes.length; j < m; j++) {
819                         child = source.childNodes[j];
820                         if (child.nodeName === 'Name_array') {
821                             this.sources[source.getAttribute('id')] = child.textContent.replace(/^\s+|\s+$/g, "").split(/[\s,]+/);
822                         }
823                         if (child.nodeName === 'float_array') {
824                             this.sources[source.getAttribute('id')] = this.parseFloatArray(child);
825                         }
826                     }
827                 }
828                 this.joints = {};
829                 var joints = this._datas['joints'][0];
830                 for (i = 0, l = joints.childNodes.length; i < l; i++) {
831                     child = joints.childNodes[i];
832                     if (child.nodeName === 'input') {
833                         this.joints[child.getAttribute('semantic')] = this.sources[this.getReferenceAttribute(child, 'source')];
834                     }
835                 }
836                 this.vertex_weights = [];
837                 var vweights = this._datas['vertex_weights'][0];
838                 for (i = 0, l = vweights.childNodes.length; i < l; i++) {
839                     child = vweights.childNodes[i];
840                     if (child.nodeName === 'input') {
841                         this.vertex_weights[child.getAttribute('semantic')] = this.sources[this.getReferenceAttribute(child, 'source')];
842                         this.vertex_weights[child.getAttribute('semantic') + '_offset'] = child.getAttribute('offset');
843                     }
844                     if (child.nodeName === 'vcount' || child.nodeName === 'v') {
845                         this.vertex_weights[child.nodeName] = this.parseFloatArray(child);
846                     }
847                 }
848                 this.bind_shape_matrix = mat4.identity();
849                 if (this._datas['bind_shape_matrix'].length > 0) {
850                     var bind_shape_matrix = this._datas['bind_shape_matrix'][0];
851                     this.bind_shape_matrix = mat4.transpose(this.parseFloatArray(bind_shape_matrix));
852                 }
853             },
854             getProcessedSkinData: function() {
855                 var resultSkin = {};
856                 resultSkin.bind_shape_matrix = this.bind_shape_matrix;
857                 resultSkin.joints = {};
858                 var ids = {};
859                 for (var i = 0, l = this.vertex_weights.JOINT.length; i < l; i++) {
860                     ids[this.vertex_weights.JOINT[i]] = i;
861                 }
862                 resultSkin.ids = ids;
863                 resultSkin.source = this.source;
864                 for (i = 0, l = this.joints['JOINT'].length; i < l; i++) {
865                     resultSkin.joints[this.joints['JOINT'][i]] = [];
866                     for (var j = 0; j < 16; j++) {
867                         var retu = (j - j % 4) / 4;
868                         var gyou = j % 4;
869                         resultSkin.joints[this.joints['JOINT'][i]].push(this.joints['INV_BIND_MATRIX'][i * 16 + gyou * 4 + retu]);
870                     }
871                 }
872                 resultSkin.vertex_weights = [];
873                 var last = 0;
874                 for (i = 0, l = this.vertex_weights['vcount'].length; i < l; i++) {
875                     resultSkin.vertex_weights[i] = [];
876                     for (var vj = 0, vl = this.vertex_weights['vcount'][i]; vj < vl; vj++) {
877                         var weight = this.vertex_weights['WEIGHT'][this.vertex_weights['v'][(last + vj) * 2 + 1]];
878                         resultSkin.vertex_weights[i][this.vertex_weights['JOINT'][this.vertex_weights['v'][(last + vj) * 2]]] = weight;
879                     }
880                     last += this.vertex_weights['vcount'][i];
881                 }
882                 return resultSkin;
883             }
884         });
885         Skin.prototype._childable = ['bind_shape_matrix', 'source', 'joints', 'vertex_weights', 'extra'];
886         var Effect = enchant.Class.create(Unit, {
887             getFieldFromXMLWithDefaultValue: function(defaultValue, elementName, floatArrayName) {
888                 var element = this._datas['profile_COMMON'][0].getElementsByTagName(elementName)[0];
889                 if (element) {
890                     var array = element.getElementsByTagName(floatArrayName)[0];
891                     if (array) {return this.parseFloatArray(array);}
892                 }
893                 return defaultValue;
894             },
895             initialize: function(xml) {
896                 Unit.call(this, xml);
897                 if (this._datas['profile_COMMON'][0].getElementsByTagName('newparam')[0]) {
898                     this.imageSrc = this._datas['profile_COMMON'][0].getElementsByTagName('newparam')[0].getElementsByTagName('surface')[0].getElementsByTagName('init_from')[0].textContent;
899                 }
900                 this.emission = this.getFieldFromXMLWithDefaultValue([0, 0, 0, 1], 'emission', 'color');
901                 this.ambient = this.getFieldFromXMLWithDefaultValue([1, 1, 1, 1], 'ambient', 'color');
902                 this.diffuse = this.getFieldFromXMLWithDefaultValue([1.0, 1.0, 1.0, 1], 'diffuse', 'color');
903                 this.specular = this.getFieldFromXMLWithDefaultValue([1, 1, 1, 1], 'specular', 'color');
904                 this.shininess = this.getFieldFromXMLWithDefaultValue([20], 'shininess', 'float');
905                 this.reflective = this.getFieldFromXMLWithDefaultValue([0, 0, 0, 0], 'reflective', 'color');
906                 this.reflectivity = this.getFieldFromXMLWithDefaultValue([0], 'reflectivity', 'float');
907                 this.transparent = this.getFieldFromXMLWithDefaultValue([0, 0, 0, 0], 'transparent', 'color');
908                 this.transparency = this.getFieldFromXMLWithDefaultValue(0, 'transparency', 'float');
909             }
910         });
911         Effect.prototype._childable = ['asset', 'annotate', 'image', 'newparam', 'profile_CG', 'profile_GLSL', 'profile_COMMON', 'extra'];
912 
913         /**
914          * Exports collada.gl.enchant.js class to enchant.
915          */
916         enchant.gl.collada = {};
917         /**
918          * @scope enchant.gl.collada.ColladaBone.prototype
919          */
920         enchant.gl.collada.ColladaBone = enchant.Class.create(enchant.gl.Bone, {
921             /**
922              * Class to display the status of bones used for collada skinning.
923              * @param {Node} node
924              * @param {vec3} parentpos
925              * @param {quat4} parentrot
926              * @constructs
927              * @extends enchant.gl.Bone
928              */
929             initialize: function(node, parentpos, parentrot) {
930                 var rotation = node.getRotationMatrix();
931                 var translation = node.getTranslationMatrix();
932                 var matrix = node.getnMatrix();
933                 var animationMatrixes = [];
934                 animationMatrixes[0] = [];
935                 animationMatrixes[0][node.sid] = mat4.multiply(translation, rotation, mat4.create());
936                 mat4.multiply(animationMatrixes[0][node.sid],matrix);
937                 var pos = vec3.create([animationMatrixes[0][node.sid][12], animationMatrixes[0][node.sid][13], animationMatrixes[0][node.sid][14]]);
938                 var rotation3x3 = mat4.toMat3(animationMatrixes[0][node.sid], mat3.create());
939                 mat3.transpose(rotation3x3);
940                 var quatanion = mat3.toQuat4(rotation3x3, quat4.create());
941                 var wquatanion, local;
942                 if (parentrot) {
943                     wquatanion = quat4.multiply(parentrot, quatanion, quat4.create());
944                     local = quat4.multiplyVec3(parentrot, vec3.create(pos));
945                 } else {
946                     wquatanion = quatanion;
947                     local = vec3.create(pos);
948                 }
949                 var head = vec3.add(parentpos, local, vec3.create());
950                 enchant.gl.Bone.call(this, node.sid, head, pos, quatanion);
951                 for (var k in node.nodes) {
952                     var child = new enchant.gl.collada.ColladaBone(node.nodes[k], head, wquatanion);
953                     this.addChild(child);
954                 }
955             }
956         });
957         /**
958          * @scope enchant.gl.collada.ColladaSkeleton.prototype
959          */
960         enchant.gl.collada.ColladaSkeleton = enchant.Class.create(enchant.gl.Skeleton, {
961             /**
962              * Class that becomes bone structure route augmented with specific collada information.
963              * @constructs
964              * @extends enchant.gl.Skeleton
965              */
966             initialize: function() {
967                 enchant.gl.Skeleton.call(this);
968             },
969             calculateTableForIds: function(ids) {
970                 this.table = this._calculateFlatTableForIds(this, ids);
971             },
972             _calculateFlatTableForIds: function(skelPart, ids) {
973                 var table = {};
974                 table.pos = [];
975                 table.rot = [];
976                 for (var i = 0; i < 4; i++) {
977                     if (i < 3) {table['pos'][ids[skelPart._name] * 3 + i] = skelPart._globalpos[i];}
978                     table['rot'][ids[skelPart._name] * 4 + i] = skelPart._globalrot[i];
979                 }
980                 for (var x = 0, l = skelPart.childNodes.length; x < l; x++) {
981                     var child = this._calculateFlatTableForIds(skelPart.childNodes[x], ids);
982                     for (var key in child) {
983                         for (var k in child[key]) {
984                             table[key][k] = child[key][k];
985                         }
986                     }
987                 }
988                 return table;
989             }
990         });
991         /**
992          * @scope enchant.gl.collada.ColladaMesh.prototype
993          */
994         enchant.gl.collada.ColladaMesh = enchant.Class.create(enchant.gl.Mesh, {
995             /**
996              * Class to store peak arrays and textures.
997              * Used as a enchant.gl.collada.ColladaSprite3D property.
998              * @param {Triangle} triangles
999              * @constructs
1000              * @extends enchant.gl.Mesh
1001              */
1002             initialize: function(triangles) {
1003                 enchant.gl.Mesh.call(this);
1004                 this.parseMeshFromGeometryMesh(triangles);
1005                 this.colors = [];
1006                 for (var i = 0; i < (this.vertices.length / 3) * 4; i++) {
1007                     this.colors[i] = 1.0;
1008                 }
1009             },
1010             parseMeshFromGeometryMesh: function(triangles) {
1011                 var inputs = triangles.inputs;
1012                 this.vertices = [];
1013                 this.normals = [];
1014                 this.texCoords = [];
1015                 this.indices = [];
1016                 var index;
1017                 for (var k = 0; k < triangles.primitives.length; k += triangles.stride) {
1018                     if (triangles.inputs['POSITIONoffset'] >= 0) {
1019                         index = triangles.primitives[k + triangles.inputs['POSITIONoffset']] * 3;
1020                         this.vertices.push(inputs['POSITION'][index], inputs['POSITION'][index + 1], inputs['POSITION'][index + 2]);
1021                     }
1022                     if (triangles.inputs['NORMALoffset'] >= 0) {
1023                         index = triangles.primitives[k + triangles.inputs['NORMALoffset']] * 3;
1024                         this.normals.push(inputs['NORMAL'][index], inputs['NORMAL'][index + 1], inputs['NORMAL'][index + 2]);
1025                     }
1026                     if (triangles.inputs['TEXCOORDoffset'] >= 0) {
1027                         index = triangles.primitives[k + triangles.inputs['TEXCOORDoffset']] * (2+triangles.inputs['TEXCOORDset']);
1028                         this.texCoords.push(inputs['TEXCOORD'][index], inputs['TEXCOORD'][index + 1]);
1029                     } else {
1030                         this.texCoords.push(0, 0);
1031                     }
1032                 }
1033                 for (k = 0; k < triangles.primitives.length / (triangles.stride); k++) {
1034                     this.indices.push(k);
1035                 }
1036             }
1037         });
1038         /**
1039          * @scope enchant.gl.collada.ColladaSkeletonSpriteMesh.prototype
1040          */
1041         enchant.gl.collada.ColladaSkeletonSpriteMesh = enchant.Class.create(enchant.gl.collada.ColladaMesh, {
1042             /**
1043              * Class to store peak arrays and textures.
1044              * Used as an enchant.gl.collada.ColladaSkeletonSprite3D property.
1045              * @param {Triangle} triangles
1046              * @constructs
1047              * @extends enchant.gl.collada.ColladaMesh
1048              */
1049             initialize: function(triangles) {
1050                 enchant.gl.collada.ColladaMesh.call(this, triangles);
1051                 var vpos1Buffer = new enchant.gl.Buffer(enchant.gl.Buffer.VERTICES);
1052                 var vpos2Buffer = new enchant.gl.Buffer(enchant.gl.Buffer.VERTICES);
1053                 var vpos3Buffer = new enchant.gl.Buffer(enchant.gl.Buffer.VERTICES);
1054                 var bindicesBuffer = new enchant.gl.Buffer(enchant.gl.Buffer.BONE_INDICES);
1055                 var weights1Buffer = new enchant.gl.Buffer(enchant.gl.Buffer.WEIGHTS);
1056                 var weights2Buffer = new enchant.gl.Buffer(enchant.gl.Buffer.WEIGHTS);
1057                 this._addAttribute(vpos1Buffer, 'vpos1');
1058                 this._addAttribute(vpos2Buffer, 'vpos2');
1059                 this._addAttribute(vpos3Buffer, 'vpos3');
1060                 this._addAttribute(bindicesBuffer, 'boneIndices');
1061                 this._addAttribute(weights1Buffer, 'weights1');
1062                 this._addAttribute(weights2Buffer, 'weights2');
1063             },
1064             _addAttribute: function(buffer, prop) {
1065                 this['_' + prop] = buffer;
1066                 Object.defineProperty(this, prop, {
1067                     get: function() {
1068                         return this['_' + prop]._array;
1069                     },
1070                     set: function(array) {
1071                         this['_' + prop]._array = array;
1072                         if (this._appear) {
1073                             this['_' + prop]._bufferData();
1074                         }
1075                     }
1076                 });
1077             }
1078         });
1079         /**
1080          * @scope enchant.gl.collada.AbstractColladaSprite3D.prototype
1081          */
1082         enchant.gl.collada.AbstractColladaSprite3D = enchant.Class.create(enchant.gl.Sprite3D, {
1083             /**
1084              * Base class used for collada Sprite3Ds.
1085              * This class should not be initialized directly.
1086              * @param {Object} lib
1087              * @param {Node} node
1088              * @param {Triangle} triangles
1089              * @constructs
1090              * @extends enchant.gl.Sprite3D
1091              */
1092             initialize: function(lib, node, triangles) {
1093                 enchant.gl.Sprite3D.call(this);
1094                 this.lib = lib;
1095                 if (triangles) {
1096                     this.mesh = this._getMesh(triangles);
1097                     this.initSpriteTexture(node, lib, triangles);
1098                 }
1099             },
1100             _getMesh: function(triangles) {
1101                 return null;
1102             },
1103             getPose: function(poses, length, localframe) {
1104                 var core = enchant.Core.instance;
1105                 var frame = (core.frame) % length;
1106                 if (localframe) {
1107                     frame = localframe;
1108                 }
1109                 var pose = [];
1110                 for (var k in poses) {
1111                     pose[k] = poses[k].getFrame(frame);
1112                 }
1113                 return pose;
1114             },
1115             createPoses: function(node, poses, lib) {
1116                 var matrix = node.getAnimationMatrixesLocal(lib['animations'], lib['animation_clips'], true);
1117                 var length = 0;
1118                 for (var k in matrix) {
1119                     poses[k] = new enchant.gl.KeyFrameManager();
1120                     for (var i in matrix[k]) {
1121                         var pos = vec3.create([matrix[k][i][12], matrix[k][i][13], matrix[k][i][14]]);
1122                         var rotation3x3 = mat4.toMat3(matrix[k][i], mat3.create());
1123                         mat3.transpose(rotation3x3);
1124                         var quatanion = quat4.fromRotationMatrix(rotation3x3, quat4.create());
1125                         poses[k].addFrame(new enchant.gl.Pose(pos, quatanion), parseInt(i, 10));
1126                     }
1127                     length = Math.max(poses[k].length, length);
1128                 }
1129                 return length;
1130             },
1131             createPosesClips: function(node, poseclips, lib) {
1132                 var matrixclips = node.getAnimationMatrixesLocalFromAnimationClips(lib['animations'],lib['animation_clips']);
1133                 var length = [];
1134                 var core = enchant.Core.instance;
1135                 for (var pkey in matrixclips) {
1136                     length[pkey] = 0;
1137                     var matrix = matrixclips[pkey];
1138                     poseclips[pkey] = [];
1139                     for (var mkey in matrix) {
1140                         poseclips[pkey][mkey] = new enchant.gl.KeyFrameManager();
1141                         for (var i in matrix[mkey]) {
1142                             var num = 0;
1143                             if (matrix[mkey].length > parseInt(lib['animation_clips'][pkey].start * core.fps, 10)) {
1144                                 num = i - parseInt(lib['animation_clips'][pkey].start * core.fps, 10);
1145                             } else {
1146                                 num = i;
1147                             }
1148                             var pos = vec3.create([matrix[mkey][i][12], matrix[mkey][i][13], matrix[mkey][i][14]]);
1149                             var rotation3x3 = mat4.toMat3(matrix[mkey][i], mat3.create());
1150                             mat3.transpose(rotation3x3);
1151                             var quatanion = quat4.fromRotationMatrix(rotation3x3, quat4.create());
1152                             if (num >= 0 && i <= parseInt(lib['animation_clips'][pkey].end * core.fps, 10)) {
1153                                 poseclips[pkey][mkey].addFrame(new enchant.gl.Pose(pos, quatanion), parseInt(num, 10));
1154                             }
1155                         }
1156                         length[pkey] = Math.max(poseclips[pkey][mkey].length, length[pkey]);
1157                     }
1158                 }
1159                 return length;
1160             },
1161             initSpriteTexture: function(node, lib, triangles) {
1162                 var libMaterials = lib['materials'];
1163                 var libEffects = lib['effects'];
1164                 var libImages = lib['images'];
1165                 if (node.materialTarget && this.mesh) {
1166                     var material = node.materialTarget[triangles.material];
1167                     if (material) {
1168                         var texture = this.mesh.texture;
1169                         var effect = libEffects[libMaterials[material].effectUrl];
1170                         texture.emission = effect.emission;
1171                         texture.ambient = effect.ambient;
1172                         texture.diffuse = effect.diffuse;
1173                         texture.specular = effect.specular;
1174                         texture.shininess = effect.shininess[0];
1175                         if (effect.imageSrc) {
1176                             texture.src=libImages[effect.imageSrc].initFrom;
1177                         }
1178                     }
1179                 }
1180             },
1181             _getTriangles: function(geometry, index) {
1182                 if (geometry) {
1183                     return geometry.Mesh.triangles[index];
1184                 }
1185                 return null;
1186             },
1187             _getTrianglesLength: function(geometry) {
1188                 if (geometry) {
1189                     return geometry.Mesh.triangles.length;
1190                 }
1191                 return 0;
1192             }
1193         });
1194         /**
1195          * @scope enchant.gl.collada.RootColladaSprite3D.prototype
1196          */
1197         enchant.gl.collada.RootColladaSprite3D = enchant.Class.create(enchant.gl.Sprite3D, {
1198             /**
1199             * Add animation clip.
1200             * Animation clip will be played in the order that it is added.
1201             * @param {String} clipId
1202             */
1203             pushAnimationClip: function(clipId){
1204                 this.childNodes[1].pushAnimationClip(clipId);
1205             },
1206             /**
1207             * Delete added animation clip.
1208             */
1209             clearAnimationClip: function(){
1210                 this.childNodes[1].clearAnimationClip();
1211             },
1212             getAnimationClip: function() {
1213                 return this.childNodes[1].animationClips;
1214             },
1215             loop: {
1216                 set: function(flag) {
1217                     this.childNodes[1].loop = flag;
1218                 },
1219                 get: function() {
1220                     return this.childNodes[1].loop;
1221                 }
1222             }
1223         });
1224         /**
1225          * @scope enchant.gl.collada.ColladaSprite3D.prototype
1226          */
1227         enchant.gl.collada.ColladaSprite3D = enchant.Class.create(enchant.gl.collada.AbstractColladaSprite3D, {
1228             _getMesh: function(triangles) {
1229                 return new enchant.gl.collada.ColladaMesh(triangles);
1230             },
1231             /**
1232              * Class to display Sprite3Ds created from a collada file.
1233              * The ColladaSprite3D class can be used as {@link enchant.gl.Sprite3D}.
1234              * This method is a factory method for ColladaSprite3D class which creates
1235              * a new hirachy of ColladaSprite3D from a collada Node.
1236              * @param {Node} node
1237              * @constructs
1238              * @extends enchant.gl.collada.AbstractColladaSprite3D
1239              */
1240             addColladaSprite3DFromNode: function(node) {
1241                 var geometry = this.lib['geometries'][node.geometryUrl];
1242                 var trianglesLength = this._getTrianglesLength(geometry);
1243                 var currentTriangle = 0;
1244                 var makeEnterframe = function(poses, length, sid, obj) {
1245                     return function() {
1246                          var currentPose = obj.getPose(poses, length)[sid];
1247                          obj.rotation = quat4.toMat4(currentPose._rotation);
1248                     };
1249                 };
1250                 do {
1251                     var sprite = new enchant.gl.collada.ColladaSprite3D(this.lib, node, this._getTriangles(geometry, currentTriangle));
1252                     if (node._datas['translate'][0]) {
1253                         var translate = node.parseFloatArray(node._datas['translate'][0]);
1254                         sprite.x = translate[0];
1255                         sprite.y = translate[1];
1256                         sprite.z = translate[2];
1257                     } else {
1258                         sprite.x = sprite.y = sprite.z = 0;
1259                     }
1260                     var rotation = [];
1261                     var rotateX = [1, 0, 0];
1262                     var rotateY = [0, 1, 0];
1263                     var rotateZ = [0, 0, 1];
1264                     for (var i = 0, l = node._datas['rotate'].length; i < l; i++) {
1265                         var rotate = node._datas['rotate'][i];
1266                         var sid = rotate.getAttribute('sid');
1267                         if (sid === 'rotateZ') {
1268                             rotateZ = node.parseFloatArray(rotate);
1269                         } else if (sid === 'rotateY') {
1270                             rotateY = node.parseFloatArray(rotate);
1271                         } else if (sid === 'rotateX') {
1272                             rotateX = node.parseFloatArray(rotate);
1273                         }
1274                     }
1275                     for (i = 0; i < 3; i++) {
1276                         rotation.push(rotateX[i], rotateY[i], rotateZ[i], 0);
1277                     }
1278                     rotation.push(0, 0, 0, 1);
1279                     sprite.rotation = rotation;
1280                     if (node._datas['matrix'][0]) {
1281                         var matrix = node.parseFloatArray(node._datas['matrix'][0]);
1282                         var transposed = [matrix[0], matrix[4], matrix[8], matrix[12], matrix[1], matrix[5], matrix[9], matrix[13], matrix[2], matrix[6], matrix[10], matrix[14], matrix[3], matrix[7], matrix[11], matrix[15]];
1283                         sprite.matrix = transposed;
1284                     } else {
1285                         sprite.matrix = [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1];
1286                     }
1287                     if (node._datas['scale'][0]) {
1288                         var scale = node.parseFloatArray(node._datas['scale'][0]);
1289                         sprite.scaleX = scale[0];
1290                         sprite.scaleY = scale[1];
1291                         sprite.scaleZ = scale[2];
1292                     } else {
1293                         sprite.scaleX = sprite.scaleY = sprite.scaleZ = 1;
1294                     }
1295                     if (node.nodes) {
1296                         for (var k in node.nodes) {
1297                             sprite.addColladaSprite3DFromNode(node.nodes[k]);
1298                         }
1299                     }
1300                     var poses = [];
1301                     var poselength = this.createPoses(node, poses, this.lib);
1302                     if (poselength > 0) {
1303                         sprite.addEventListener('enterframe', makeEnterframe(poses, poselength, node.sid, sprite));
1304                     }
1305                     //var poselength = this.createPoses(node, poses, this.lib);
1306                     //if (poselength > 0) {
1307                         //sprite.addEventListener('enterframe', function() {
1308                             //var currentPose = this.getPose(poses, poselength)[node.sid];
1309 
1310                             // TODO : USE CURRENT POSE TO MODIFY SPRITE ROTATION / TRANSLATION
1311                             //var mat = new Quat();
1312                             //mat._quat = currentPose._rotation;
1313                             //console.log(currentPose._rotation);
1314                             //sprite.rotationApply(mat);
1315                             //                          var glMat = mat4.create();
1316                             //                          mat.toMat4(glMat);
1317                             //                          mat4.multiply(sprite._rotation, glMat);
1318                             //mat4.rotateX(sprite._rotation, currentPose._rotation[2]);
1319                             //mat4.rotateY(sprite._rotation, currentPose._rotation[1]);
1320                             //mat4.rotateZ(sprite._rotation, currentPose._rotation[0]);
1321                             //sprite._changedRotation = true;
1322                             //sprite.translate(currentPose._position[2],currentPose._position[1],currentPose._position[0]);
1323                         //});
1324                     //}
1325                     this.addChild(sprite);
1326                     currentTriangle++;
1327                 } while (currentTriangle < trianglesLength);
1328             }
1329         });
1330         /**
1331          * @scope enchant.gl.collada.ColladaSprite3D.prototype
1332          */
1333         enchant.gl.collada.ColladaSkeletonSprite3D = enchant.Class.create(enchant.gl.collada.ColladaSprite3D, {
1334             /**
1335              * Base class used for collada Sprite3Ds.
1336              * This class should not be initialized directly.
1337              * @param {Object} lib
1338              * @param {Node} node
1339              * @param {Triangle} triangles
1340              * @constructs
1341              * @extends enchant.gl.collada.ColladaSprite3D
1342              */
1343             initialize: function(lib, node, triangles) {
1344                 enchant.gl.collada.AbstractColladaSprite3D.call(this, lib, node, triangles);
1345                 this.program = enchant.gl.collada.COLLADA_SHADER_PROGRAM;
1346                 this.animation = [];
1347                 this.defaultAnimation = function(){
1348                 };
1349                 this.animationClips = [];
1350                 this.uMVMatrix = mat4.create();
1351                 this.uNMatrix = mat3.create();
1352                 this.applyAnimationChild = function() {
1353                     for (var cni = 0, cnl = this.childNodes.length; cni < cnl; cni++) {
1354                         this.childNodes[cni].defaultAnimation(this.childNodes[cni]);
1355                         this.childNodes[cni].applyAnimationChild();
1356                     }
1357                 };
1358                 this.addEventListener("enterframe", function() {
1359                     this.applyAnimationChild();
1360                     if (this.animation.length === 0) {
1361                     } else {
1362                         var first = this.animation[0];
1363                         first.frame++;
1364                         for(var i = 0, l = this.childNodes.length; i < l; i++) {
1365                             first.animation.enterframe(this.childNodes[i], first.frame);
1366                         }
1367                         if (first.frame >= this.animation[0].animation.length) {
1368                             first = this.animation.shift();
1369                             if (this.loop) {
1370                                 first.frame = 0;
1371                                 this.animation.push(first);
1372                             }
1373                         }
1374                     }
1375                 });
1376             },
1377             pushAnimationClip: function(clipId) {
1378                 if(this.animationClips[clipId]){
1379                     this.animation.push({animation: this.animationClips[clipId], frame: 0});
1380                 }
1381             },
1382             clearAnimationClip: function() {
1383                 this.animation = [];
1384             },
1385             _getMesh: function(triangles) {
1386                 return new enchant.gl.collada.ColladaSkeletonSpriteMesh(triangles);
1387             },
1388             /**
1389              * Class to display Sprite3Ds created from a collada file with a skeletal animation.
1390              * The ColladaSprite3D class can be used as {@link enchant.gl.Sprite3D}.
1391              * This method is a factory method for ColladaSkeletonSprite3D class which creates
1392              * a new hirachy of ColladaSkeletonSprite3D from a collada Node.
1393              * @param {Node} node
1394              * @constructs
1395              * @extends enchant.gl.collada.ColladaSprite3D
1396              */
1397             addColladaSkeletonSprite3DFromNode: function(node, skin, skeleton, maxbonenum) {
1398                 var controller = this.lib['controllers'][node.controllerUrl];
1399                 var geometry = this.lib['geometries'][controller.skin.source];
1400                 var trianglesLength = this._getTrianglesLength(geometry);
1401                 var currentTriangle = 0;
1402                 var makeEnterframe = function(poses, length, skin, maxbonem) {
1403                     return function(obj) {
1404                         skeleton.setPoses(obj.getPose(poses, length));
1405                         skeleton.solveFKs();
1406                         skeleton.calculateTableForIds(skin.ids);
1407                         obj.mesh.udBoneInfo = obj.calculateSkeletonTable(obj.divisioninfo.dividedIndices, skeleton.table, maxbonenum);
1408                     };
1409                 };
1410                 var makeClipEnterframe = function(poses, length, skin, maxbonem, clipId) {
1411                     return function(obj, frame) {
1412                         skeleton.setPoses(obj.getPose(poses, length, frame));
1413                         skeleton.solveFKs();
1414                         skeleton.calculateTableForIds(skin.ids);
1415                         obj.mesh.udBoneInfo = obj.calculateSkeletonTable(obj.divisioninfo.dividedIndices, skeleton.table, maxbonenum);
1416                     };
1417                 };
1418                 do {
1419                     var triangles = this._getTriangles(geometry, currentTriangle);
1420                     var child = new enchant.gl.collada.ColladaSkeletonSprite3D(this.lib, node, triangles);
1421                     var arrays = this.createStaticArrayForShader(triangles, skin, skeleton);
1422                     child.divisioninfo = this.getDivisionInfo(arrays, skeleton.table, maxbonenum);
1423                     this.addChild(child);
1424 
1425                     child.mesh.vpos1 = arrays.vectorFromBone1;
1426                     child.mesh.vpos2 = arrays.vectorFromBone2;
1427                     child.mesh.vpos3 = arrays.vectorFromBone3;
1428 
1429                     child.mesh.weights1 = arrays.weights1;
1430                     child.mesh.weights2 = arrays.weights2;
1431 
1432                     child.mesh.dividedpoints = child.divisioninfo.dividedPoint;
1433                     child.mesh.boneIndices = child.divisioninfo.boneIndices;
1434                     child.mesh.udBoneInfo = child.divisioninfo.skeletontable;
1435                     var poses = [];
1436                     var length = this.createPoses(node, poses, this.lib);
1437                     var posesclips = [];
1438                     var cliplength = this.createPosesClips(node, posesclips, this.lib);
1439                     for (var clk in cliplength) {
1440                         if (cliplength[clk] > 0) {
1441                             var clipenterframe = makeClipEnterframe(posesclips[clk], cliplength[clk], skin, maxbonenum, clk);
1442                             this.animationClips[clk] = {enterframe: clipenterframe, length: cliplength[clk]};
1443                         }
1444                     }
1445                     if (length > 0) {
1446                         child.defaultAnimation = makeEnterframe(poses, length, skin, maxbonenum);
1447                     }
1448                     currentTriangle++;
1449                 } while (currentTriangle < trianglesLength);
1450             },
1451             calculateSkeletonTable: function(dividedIndices, table, n) {
1452                 var skeletontable = [];
1453                 var indices, index, postable, rottable;
1454                 for (var i = 0, l = dividedIndices.length; i < l; i++) {
1455                     indices = dividedIndices[i];
1456                     postable = [];
1457                     rottable = [];
1458                     for (var j = 0, ll = indices.length; j < ll; j++) {
1459                         index = indices[j];
1460                         postable.push(
1461                             table.pos[index * 3],
1462                             table.pos[index * 3 + 1],
1463                             table.pos[index * 3 + 2]
1464                         );
1465                         rottable.push(
1466                             table.rot[index * 4],
1467                             table.rot[index * 4 + 1],
1468                             table.rot[index * 4 + 2],
1469                             table.rot[index * 4 + 3]
1470                         );
1471                     }
1472                     skeletontable.push({
1473                         pos: new Float32Array(postable),
1474                         rot: new Float32Array(rottable)
1475                     });
1476                 }
1477                 return skeletontable;
1478             },
1479             /**
1480              * スキニングの分割描画を行う境界を計算する.
1481              *
1482              */
1483             getDivisionInfo: function(arrays, table, n) {
1484                 var dividedPoint = [0];
1485                 var dividedIndices = [];
1486                 var indices = [];
1487                 var packedIndices = [];
1488                 var count = 0;
1489                 var index, packedIndex, num;
1490                 for (var i = 0, l = arrays.boneIndices.length; i < l; i += 9) {
1491                     for (var j = 0; j < 9; j++) {
1492                         index = arrays.boneIndices[i + j];
1493                         num = indices.indexOf(index);
1494                         if (num === -1) {
1495                             indices.push(index);
1496                             packedIndex = indices.length - 1;
1497                         } else {
1498                             packedIndex = num;
1499                         }
1500                         packedIndices.push(packedIndex);
1501                         if (j === 8) {
1502                             if (indices.length > n) {
1503                                 dividedPoint.push(i + 9);
1504                                 dividedIndices.push(indices);
1505                                 count++;
1506                                 indices = [];
1507                             }
1508                         }
1509                     }
1510                 }
1511                 dividedPoint.push(packedIndices.length);
1512                 dividedIndices.push(indices);
1513                 return {
1514                     dividedPoint: dividedPoint,
1515                     dividedIndices: dividedIndices,
1516                     skeletontable: this.calculateSkeletonTable(dividedIndices, table, n),
1517                     boneIndices: new Uint16Array(packedIndices)
1518                 };
1519             },
1520             createStaticArrayForShader: function(triangles, skin, skeleton) {
1521                 var arraysForShader = {};
1522                 var length = 0;
1523                 var vectorForBones = [
1524                     [],
1525                     [],
1526                     []
1527                 ];
1528                 var weights = [
1529                     [],
1530                     []
1531                 ];
1532                 var boneIndices = [];
1533                 var keys = [];
1534                 var index, vec, rvec;
1535                 rvec = vec3.create();
1536                 for (var i = 0, l = triangles.primitives.length; i < l; i += triangles.stride) {
1537                     if (triangles.inputs['POSITIONoffset'] >= 0) {
1538                         length++;
1539                         index = triangles.primitives[i + triangles.inputs['POSITIONoffset']];
1540                         vec = [
1541                             triangles.inputs['POSITION'][index * 3],
1542                             triangles.inputs['POSITION'][index * 3 + 1],
1543                             triangles.inputs['POSITION'][index * 3 + 2]
1544                         ];
1545                         mat4.multiplyVec3(skin.bind_shape_matrix, vec);
1546                         var count = -1;
1547                         keys.push(skin.vertex_weights[index]);
1548                         for (var key in skin.vertex_weights[index]) {
1549                             count++;
1550                             mat4.multiplyVec3(skin.joints[key], vec, rvec);
1551                             vectorForBones[count].push(rvec[0], rvec[1], rvec[2]);
1552                             boneIndices.push(skin.ids[key]);
1553                             if (count < 2) {
1554                                 weights[count].push(skin.vertex_weights[index][key]);
1555                             } else {
1556                                 break;
1557                             }
1558                         }
1559                         for (var j = count + 1; j < 3; j++) {
1560                             if (j < 2) {
1561                                 weights[j].push(0);
1562                             }
1563                             vectorForBones[j].push(0, 0, 0);
1564                             boneIndices.push(0);
1565                         }
1566                     }
1567                 }
1568                 for (i = 0; i < 3; i++) {
1569                     arraysForShader['vectorFromBone' + (i + 1)] = new Float32Array(vectorForBones[i]);
1570                     if (i < 2) {arraysForShader['weights' + (i + 1)] = new Float32Array(weights[i]);}
1571                 }
1572                 arraysForShader.boneIndices = new Uint16Array(boneIndices);
1573                 arraysForShader.keys = keys;
1574                 return arraysForShader;
1575             },
1576             _render: function(detectTouch) {
1577                 var core = enchant.Core.instance;
1578                 var scene = core.currentScene3D;
1579                 var l = scene.directionalLight;
1580                 var detect = (detectTouch === 'detect') ? 1.0 : 0.0;
1581                 mat4.multiply(scene._camera.mat, this.tmpMat, this.uMVMatrix);
1582                 mat4.toInverseMat3(this.tmpMat, this.uNMatrix);
1583                 mat3.transpose(this.uNMatrix);
1584                 core.GL.currentProgram.setAttributes({
1585                     aVertexPosition: this.mesh._vertices,
1586                     aVertexNormal: this.mesh._normals,
1587                     aTextureCoord: this.mesh._texCoords,
1588                     aBoneIndices: this.mesh._boneIndices,
1589                     aVectorFromBone1: this.mesh._vpos1,
1590                     aVectorFromBone2: this.mesh._vpos2,
1591                     aVectorFromBone3: this.mesh._vpos3,
1592                     aBoneWeight1: this.mesh._weights1,
1593                     aBoneWeight2: this.mesh._weights2
1594                 });
1595                 core.GL.currentProgram.setUniforms({
1596                     uUseDirectionalLight: scene.useDirectionalLight,
1597                     uLightColor: l.color,
1598                     uDetectTouch: detect,
1599                     uAmbientLightColor: scene.ambientLight.color,
1600                     uPMatrix: scene._camera.projMat,
1601                     uMVMatrix: this.uMVMatrix,
1602                     uNMatrix: this.uNMatrix,
1603                     uLightDirection: [
1604                         l.directionX, l.directionY, l.directionZ
1605                     ],
1606                     uLookVec: [
1607                         scene._camera._centerX - scene._camera._x,
1608                         scene._camera._centerY - scene._camera._y,
1609                         scene._camera._centerZ - scene._camera._z
1610                     ]
1611                 });
1612                 var u = {
1613                     uDetectColor: this.detectColor,
1614                     uSpecular: this.mesh.texture.specular,
1615                     uDiffuse: this.mesh.texture.diffuse,
1616                     uEmission: this.mesh.texture.emission,
1617                     uAmbient: this.mesh.texture.ambient,
1618                     uShininess: this.mesh.texture.shininess
1619                 };
1620 
1621                 if (this.mesh.texture._image) {
1622                     u.uUseTexture = 1;
1623                     u.uSampler = this.mesh.texture;
1624                 } else {
1625                     u.uUseTexture = 0;
1626                     u.uSampler = 0;
1627                 }
1628                 core.GL.currentProgram.setUniforms(u);
1629                 for (var i = 0; i < this.mesh.dividedpoints.length - 1; i++) {
1630                     core.GL.currentProgram.setUniforms({
1631                         uBonePos: this.mesh.udBoneInfo[i]['pos'],
1632                         uBoneRot: this.mesh.udBoneInfo[i]['rot']
1633                     });
1634                     enchant.Core.instance.GL.renderElements(this.mesh._indices, this.mesh.dividedpoints[i] / 3 * 2, this.mesh.dividedpoints[i + 1] / 3 - this.mesh.dividedpoints[i] / 3);
1635                 }
1636             }
1637         });
1638         var bufferProto = Object.getPrototypeOf(enchant.gl.Buffer);
1639         bufferProto.MORPHS = bufferProto.NORMALS;
1640         bufferProto.QUATS = bufferProto.COLORS;
1641         bufferProto.BONE_INDICES = {
1642             size: 3,
1643             type: 5123,
1644             normed: false,
1645             stride: 0,
1646             offset: 0,
1647             btype: 34962,
1648             usage: 35044,
1649             Atype: Uint16Array
1650         };
1651         bufferProto.WEIGHTS = {
1652             size: 1,
1653             type: 5126,
1654             norm: false,
1655             stride: 0,
1656             offset: 0,
1657             btype: 34962,
1658             usage: 35044,
1659             Atype: Float32Array
1660         };
1661         var COLLADA_VERTEX_SHADER_SOURCE = '\n\
1662             uniform mat4 uMVMatrix;\n\
1663             uniform mat4 uPMatrix;\n\
1664             uniform mat3 uNMatrix;\n\
1665             \n\
1666             uniform vec3 uBonePos[55];\n\
1667             uniform vec4 uBoneRot[55];\n\
1668             \n\
1669             attribute vec3 aBoneIndices;\n\
1670             attribute vec3 aVertexNormal;\n\
1671             attribute vec2 aTextureCoord;\n\
1672             \n\
1673             attribute float aBoneWeight1;\n\
1674             attribute float aBoneWeight2;\n\
1675             attribute vec3 aVectorFromBone1;\n\
1676             attribute vec3 aVectorFromBone2;\n\
1677             attribute vec3 aVectorFromBone3;\n\
1678             \n\
1679             varying vec3 vNormal;\n\
1680             varying vec4 vColor;\n\
1681             varying vec2 vTextureCoord;\n\
1682             \n\
1683             \n\
1684             vec3 qtransform(vec4 q, vec3 v) {\n\
1685                 return v + 2.0 * cross(cross(v, q.xyz) - q.w*v, q.xyz);\n\
1686             }\n\
1687             \n\
1688             void main() {\n\
1689                 int  index[3];\n\
1690                 index[0] = int(aBoneIndices.x);\n\
1691                 index[1] = int(aBoneIndices.y);\n\
1692                 index[2] = int(aBoneIndices.z);\n\
1693                 \n\
1694                 float w3 = 1.0 - aBoneWeight1 - aBoneWeight2;\n\
1695                 \n\
1696                 vec3 position = (qtransform(uBoneRot[index[0]], aVectorFromBone1) + uBonePos[index[0]]) * aBoneWeight1\n\
1697                         + (qtransform(uBoneRot[index[1]], aVectorFromBone2) + uBonePos[index[1]]) * aBoneWeight2\n\
1698                         + (qtransform(uBoneRot[index[2]], aVectorFromBone3) + uBonePos[index[2]]) * w3;\n\
1699                 vec3 normal = qtransform(uBoneRot[index[0]], aVertexNormal) * aBoneWeight1\n\
1700                         + qtransform(uBoneRot[index[1]], aVertexNormal) * aBoneWeight2\n\
1701                         + qtransform(uBoneRot[index[2]], aVertexNormal) * w3;\n\
1702                 \n\
1703                 gl_Position = uPMatrix * uMVMatrix * vec4(position, 1.0);\n\
1704                 vColor = vec4(1,1,1,1);\n\
1705                 vTextureCoord = aTextureCoord;\n\
1706                 vNormal = normalize(uNMatrix * normal);\n\
1707                 vNormal = normalize(uNMatrix * aVertexNormal);\n\
1708             }\n\
1709             ';
1710 
1711         enchant.gl.Core.prototype._original_start = enchant.gl.Core.prototype.start;
1712         enchant.gl.Core.prototype.start = function() {
1713             enchant.gl.collada.COLLADA_SHADER_PROGRAM = new enchant.gl.Shader(COLLADA_VERTEX_SHADER_SOURCE, enchant.Core.instance.GL.defaultProgram._fShaderSource);
1714             this._original_start();
1715         };
1716         var ColladaLibraryLoader = enchant.Class.create({
1717             initialize: function(libraryName, libraryPropertyName, className) {
1718                 this.libraryName = libraryName;
1719                 this.libraryPropertyName = libraryPropertyName;
1720                 this.className = className;
1721                 this.library = {};
1722             },
1723             loadLibraryFromXML: function(colladaRootElement, url) {
1724                 var libraries = colladaRootElement.getElementsByTagName('library_' + this.libraryName);
1725                 this.library = {};
1726                 var props = [];
1727                 if (libraries.length > 0) {
1728                     for (var ci = 0, cl = libraries[0].childNodes.length; ci < cl; ci++) {
1729                         if (libraries[0].childNodes[ci].nodeName === this.libraryPropertyName) {
1730                             props.push(libraries[0].childNodes[ci]);
1731                         }
1732                     }
1733                 }
1734                 var childNodes=colladaRootElement.childNodes;
1735                 if (this.libraryPropertyName === "node") {
1736                     props = colladaRootElement.getElementsByTagName(this.libraryPropertyName);
1737                 }
1738                 for (var i = 0, l = props.length; i < l; i++) {
1739                     var child = props[i];
1740                     this.library[child.getAttribute('id')] = new this.className(child, url);
1741                     if (this.libraryPropertyName === "animation_clip") {
1742                         this.library[child.getAttribute('id')].start=child.getAttribute('start');
1743                         this.library[child.getAttribute('id')].end=child.getAttribute('end');
1744                     }
1745                 }
1746                 return this.library;
1747             }
1748         });
1749         var availableLibraryFeatures = [new ColladaLibraryLoader('images', 'image', Image), new ColladaLibraryLoader('geometries', 'geometry', Geometry), new ColladaLibraryLoader('nodes', 'node', Node), new ColladaLibraryLoader('visual_scenes', 'visual_scene', VisualScene), new ColladaLibraryLoader('materials', 'material', Material), new ColladaLibraryLoader('effects', 'effect', Effect), new ColladaLibraryLoader('controllers', 'controller', Controller), new ColladaLibraryLoader('animations', 'animation', Animation), new ColladaLibraryLoader('animation_clips', 'animation_clip', AnimationClip)];
1750     }());
1751 }
1752