API Docs for: 0.5.5
Show:

File: kick/core/BuiltInResourceProvider.js

define(["./Util", "kick/mesh/MeshDataFactory", "kick/material/GLSLConstants", "./Constants"], function (Util, MeshDataFactory, GLSLConstants, Constants) {
    "use strict";

    var ASSERT = Constants._ASSERT;

    /**
     * Responsible for providing the built-in resources (such as textures, shaders and mesh data).
     * All build-in resources have the prefix kickjs
     * @class BuiltInResourceProvider
     * @namespace kick.core
     * @constructor
     * @extends kick.core.ResourceProvider
     * @param {kick.core.Engine} engine
     * @private
     */
    return function (engine) {
        var gl = engine.gl,
            thisObj = this;
        Object.defineProperties(this, {
            /**
             * Returns kickjs
             * @property protocol
             * @type String
             * @final
             */
            protocol: {
                value: "kickjs://"
            }
        });

        /**
         * <ul>
         * <li><b>Triangle</b> Url: kickjs://mesh/triangle/</li>
         * <li><b>Plane</b> Url: kickjs://mesh/plane/<br></li>
         * <li><b>Disc</b> Url: kickjs://mesh/disc/?slides=20<br></li>
         * <li><b>UVSphere</b> Url: kickjs://mesh/uvsphere/?slides=20&stacks=10&radius=1.0<br>Note that the parameters is optional</li>
         * <li><b>Cube</b> Url: kickjs://mesh/cube/?length=1.0<br>Note that the parameters is optional</li>
         * <li><b>Point</b> Url: kickjs://mesh/point/</li>
         * </ul>
         * @method getMeshData
         * @param {String} url
         * @param {kick.mesh.Mesh} meshDestination
         * @param {ResourceTracker} [resourceTracker]
         */
        this.getMeshData = function (url, meshDestination, resourceTracker) {
            var meshDataObj,
                slices,
                stacks,
                radius,
                length,
                getParameterInt = Util.getParameterInt,
                getParameterFloat = Util.getParameterFloat;
            if (resourceTracker && resourceTracker.resourceLoadingStarted){
                resourceTracker.resourceLoadingStarted(url, meshDestination);
            }
            if (url.indexOf("kickjs://mesh/triangle/") === 0) {
                meshDataObj = MeshDataFactory.createTriangleData();
            } else if (url.indexOf("kickjs://mesh/plane/") === 0) {
                meshDataObj = MeshDataFactory.createPlaneData();
            } else if (url.indexOf("kickjs://mesh/disc/") === 0) {
                slices = getParameterInt(url, "slices");
                meshDataObj = MeshDataFactory.createDiscData(slices);
            } else if (url.indexOf("kickjs://mesh/uvsphere/") === 0) {
                slices = getParameterInt(url, "slices");
                stacks = getParameterInt(url, "stacks");
                radius = getParameterFloat(url, "radius");
                meshDataObj = MeshDataFactory.createUVSphereData(slices, stacks, radius);
            } else if (url.indexOf("kickjs://mesh/cube/") === 0) {
                length = getParameterFloat(url, "length");
                meshDataObj = MeshDataFactory.createCubeData(length);
            } else if (url.indexOf("kickjs://mesh/point/") === 0) {
                meshDataObj = MeshDataFactory.createPointData();
            } else {
                Util.fail("No meshdata found for " + url);
                if (resourceTracker && resourceTracker.resourceLoadingStarted){
                    resourceTracker.resourceLoadingFailed(url, meshDestination);
                }
                return;
            }

            meshDestination.meshData = meshDataObj;
            if (resourceTracker && resourceTracker.resourceLoadingStarted){
                resourceTracker.resourceLoadingFinished(url, meshDestination);
            }
        };

        /**
         * Create a default shader config based on a URL<br>
         * The following shaders are available:
         *  <ul>
         *  <li><b>Default</b> Url: kickjs://shader/default/</li>
         *  <li><b>Specular</b> Url: kickjs://shader/specular/</li>
         *  <li><b>Diffuse</b> Url: kickjs://shader/diffuse/</li>
         *  <li><b>Unlit</b> Url: kickjs://shader/unlit/</li>
         *  <li><b>Unlit_no_depth</b> Url: kickjs://shader/unlit_no_depth/</li>
         *  <li><b>Bumped Specular</b> Url: kickjs://shader/bumped\_specular/</li>
         *  <li><b>Transparent Point Unlit</b> Url: kickjs://shader/point\_transparent\_unlit/</li>
         *  <li><b>Transparent Specular</b> Url: kickjs://shader/transparent\_specular/</li>
         *  <li><b>Transparent Unlit</b> Url: kickjs://shader/transparent\_unlit/</li>
         *  <li><b>Particles</b> Url: kickjs://shader/particles/</li>
         *  <li><b>Skybox</b> Url: kickjs://shader/skybox/</li>
         *  <li><b>Shadowmap</b> Url: kickjs://shader/\_\_shadowmap/</li>
         *  <li><b>Pick</b> Url: kickjs://shader/\_\_pick/</li>
         *  <li><b>Error</b> Url: kickjs://shader/\_\_error/<br></li>
         *  </ul>
         * @method getShaderData
         * @param {String} url
         * @param {kick.material.Shader} shaderDestination
         */
        this.getShaderData = function (url, shaderDestination, resourceTracker) {
            var i, config,
                vertexShaderSrc,
                fragmentShaderSrc,
                blend = false,
                polygonOffsetEnabled = false,
                depthMask = true,
                faceCulling = Constants.GL_BACK,
                zTest = Constants.GL_LESS,
                renderOrder = 1000,
                glslConstants = GLSLConstants,
                defaultUniforms = {},
                compareAndSetShader = function (shaderName) {
                    var res = url.indexOf("kickjs://shader/" + shaderName + "/") === 0;
                    if (res) {
                        vertexShaderSrc = glslConstants[shaderName + "_vs.glsl"];
                        fragmentShaderSrc = glslConstants[shaderName + "_fs.glsl"];
                        if (shaderName.indexOf("transparent_") === 0) {
                            blend = true;
                            depthMask = false;
                            renderOrder = 2000;
                        }

                        else if (shaderName === "__shadowmap") {
                            polygonOffsetEnabled = true;
                        }
                        else if (shaderName === "specular" || shaderName === "transparent_specular") {
                            defaultUniforms = {
                                mainColor: [1, 1, 1, 1],
                                mainTexture: engine.project.load(engine.project.ENGINE_TEXTURE_WHITE),
                                specularColor: [1, 1, 1, 1],
                                specularExponent: 50
                            };
                        }
                        else if (shaderName === "diffuse" ||
                                shaderName === "transparent_diffuse" ||
                                shaderName === "unlit" ||
                                shaderName === "unlit_vertex_color" ||
                                shaderName === "transparent_unlit") {
                            defaultUniforms = {
                                mainColor: [1, 1, 1, 1],
                                mainTexture: engine.project.load(engine.project.ENGINE_TEXTURE_WHITE)
                            };
                        }
                        else if (shaderName === "bumped_specular"){
                            defaultUniforms = {
                                mainColor: [1, 1, 1, 1],
                                mainTexture: engine.project.load(engine.project.ENGINE_TEXTURE_WHITE),
                                normalMap: engine.project.load(engine.project.ENGINE_TEXTURE_DEFAULT_NORMAL),
                                specularColor: [1, 1, 1, 1],
                                specularExponent: 50
                            };
                        }
                        else if (shaderName === "transparent_point_sprite"){
                            defaultUniforms = {
                                mainColor: [1, 1, 1, 1],
                                mainTexture: engine.project.load(engine.project.ENGINE_TEXTURE_WHITE),
                                pointSize: [50]
                            };
                        }
                        else if (shaderName === "particles"){
                            defaultUniforms = {
                                pointSize: [50]
                            };
                            blend = true;
                            depthMask = false;
                            renderOrder = 2000;
                        } else if (shaderName === "skybox"){
                            defaultUniforms = {
                                mainColor: [1, 1, 1, 1],
                                mainTexture: engine.project.load(engine.project.ENGINE_TEXTURE_CUBEMAP_WHITE)
                            };
                            faceCulling = Constants.GL_FRONT;
                            zTest = Constants.GL_LEQUAL;
                            renderOrder = 1999;
                        }

                    }
                    return res;
                },
                shaderTypes = [
                    "specular",
                    "diffuse",
                    "__shadowmap",
                    "__error",
                    "__pick",
                    "__pick_normal",
                    "__pick_uv",
                    "transparent_specular",
                    "transparent_diffuse",
                    "unlit",
                    "unlit_no_depth",
                    "unlit_vertex_color",
                    "bumped_specular",
                    "transparent_point_sprite",
                    "transparent_unlit",
                    "particles",
                    "skybox"
                ];
            if (url === "kickjs://shader/default/") {
                url = "kickjs://shader/diffuse/";
            }
            if (url ==="unlit_no_depth"){
                url = "kickjs://shader/unlit/";
            }
            for (i = 0; i < shaderTypes.length; i++) {
                if (compareAndSetShader(shaderTypes[i])) {
                    break;
                }
            }
            if (ASSERT) {
                if (!vertexShaderSrc) {
                    Util.fail("Cannot find shader url '" + url + "'");
                }
            }

            config = {
                blend: blend,
                depthMask: depthMask,
                renderOrder: renderOrder,
                polygonOffsetEnabled: polygonOffsetEnabled,
                vertexShaderSrc: vertexShaderSrc,
                fragmentShaderSrc: fragmentShaderSrc,
                defaultUniforms: defaultUniforms,
                faceCulling: faceCulling,
                zTest: zTest
            };

            Util.applyConfig(shaderDestination, config);
            shaderDestination.apply();
        };

        /**
         * Create a default texture based on a URL.<br>
         * The following default textures exists:
         *  <ul>
         *  <li><b>Black</b> Url: kickjs://texture/black/</li>
         *  <li><b>White</b> Url: kickjs://texture/white/<br></li>
         *  <li><b>Gray</b>  Url: kickjs://texture/gray/<br></li>
         *  <li><b>Default normal</b>  Url: kickjs://texture/default\_normal/<br></li>
         *  <li><b>Checkerboard</b>  Url: kickjs://texture/checkerboard/<br></li>
         *  <li><b>KickJS logo</b>  Url: kickjs://texture/logo/<br></li>
         *  </ul>
         * @method getImageData
         * @param uri
         * @param textureDestination
         */
        this.getImageData = function (uri, textureDestination, resourceTracker) {
            if (resourceTracker && resourceTracker.resourceLoadingStarted){
                resourceTracker.resourceLoadingStarted(uri, textureDestination);
            }
            var data,
                resourceLoadingFailed = function(){
                    if (resourceTracker && resourceTracker.resourceLoadingFailed){
                        resourceTracker.resourceLoadingFailed(uri, textureDestination);
                    }
                },
                img,
                logoResource = "";

            if (uri.indexOf("kickjs://texture/black/") === 0) {
                data = new Uint8Array([0, 0, 0, 255,
                    0,   0,   0, 255,
                    0,   0,   0, 255,
                    0,   0,   0, 255]);
            } else if (uri.indexOf("kickjs://texture/white/") === 0) {
                data = new Uint8Array([255, 255, 255, 255,
                    255,   255,   255, 255,
                    255,   255,   255, 255,
                    255,   255,   255, 255]);
            } else if (uri.indexOf("kickjs://texture/default_normal/") === 0) {
                data = new Uint8Array([127, 127, 255, 255,
                    127,   127,   255, 255,
                    127,   127,   255, 255,
                    127,   127,   255, 255]);
            } else if (uri.indexOf("kickjs://texture/gray/") === 0) {
                data = new Uint8Array([127, 127, 127, 255,
                    127,   127,   127, 255,
                    127,   127,   127, 255,
                    127,   127,   127, 255]);
            } else if (uri.indexOf("kickjs://texture/checkerboard/") === 0) {
                data = new Uint8Array([255, 255, 255, 255,
                    0,   0,   0, 255,
                    0,   0,   0, 255,
                    255, 255, 255, 255]);
            } else if (uri.indexOf("kickjs://texture/logo/") === 0) {
                textureDestination.setTemporaryTexture();
                img = document.createElement("img");
                img.onload = function () {
                    textureDestination.generateMipmaps = true;
                    textureDestination.internalFormat = Constants.GL_RGBA;
                    textureDestination.magFilter = Constants.GL_LINEAR;
                    textureDestination.minFilter = Constants.GL_LINEAR_MIPMAP_LINEAR;
                    textureDestination.setImage(img, uri);
                    if (resourceTracker && resourceTracker.resourceLoadingStarted){
                        resourceTracker.resourceLoadingFinished(uri, textureDestination);
                    }
                };
                img.onerror = function () {
                    resourceLoadingFailed();
                };
                img.src = logoResource;
                return;
            } else {
                Util.fail("Unknown uri " + uri);
                resourceLoadingFailed();
                return null;
            }
            textureDestination.setImageData(2, 2, 0, Constants.GL_UNSIGNED_BYTE, data, uri);
            if (resourceTracker && resourceTracker.resourceLoadingStarted){
                resourceTracker.resourceLoadingFinished(uri, textureDestination);
            }
        };
    };
});