diff --git a/CHANGES.md b/CHANGES.md index cd1d736cbe81..721374cf53b6 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -3,8 +3,10 @@ Change Log ### 1.57 - 2019-05-01 -##### Fixes :wrench: +##### Additions :tada: +* Added support for the `KHR_texture_transform` glTF extension. [#7549](https://github.com/AnalyticalGraphicsInc/cesium/pull/7549) +##### Fixes :wrench: * Fixed an error where `clampToHeightMostDetailed` or `sampleHeightMostDetailed` would crash if entities were created when the promise resolved. [#7690](https://github.com/AnalyticalGraphicsInc/cesium/pull/7690) ### 1.56.1 - 2019-04-02 diff --git a/Source/Scene/Model.js b/Source/Scene/Model.js index 09ae462f5652..d32bdf8cfea5 100644 --- a/Source/Scene/Model.js +++ b/Source/Scene/Model.js @@ -252,6 +252,8 @@ define([ * {@link https://github.com/KhronosGroup/glTF/tree/master/extensions/2.0/Khronos/KHR_materials_unlit/README.md|KHR_materials_unlit} *
  • * {@link https://github.com/KhronosGroup/glTF/tree/master/extensions/2.0/Khronos/KHR_materials_pbrSpecularGlossiness/README.md|KHR_materials_pbrSpecularGlossiness} + *
  • + * {@link https://github.com/KhronosGroup/glTF/blob/master/extensions/2.0/Khronos/KHR_texture_transform/README.md|KHR_texture_transform} *
  • * *

    @@ -1296,6 +1298,8 @@ define([ * {@link https://github.com/KhronosGroup/glTF/tree/master/extensions/2.0/Khronos/KHR_materials_unlit/README.md|KHR_materials_unlit} *
  • * {@link https://github.com/KhronosGroup/glTF/tree/master/extensions/2.0/Khronos/KHR_materials_pbrSpecularGlossiness/README.md|KHR_materials_pbrSpecularGlossiness} + *
  • + * {@link https://github.com/KhronosGroup/glTF/blob/master/extensions/2.0/Khronos/KHR_texture_transform/README.md|KHR_texture_transform} *
  • * *

    @@ -2379,24 +2383,67 @@ define([ var rendererSamplers = model._rendererResources.samplers; var sampler = rendererSamplers[texture.sampler]; - sampler = defaultValue(sampler, new Sampler({ - wrapS : TextureWrap.REPEAT, - wrapT : TextureWrap.REPEAT - })); + if (!defined(sampler)) { + sampler = new Sampler({ + wrapS : TextureWrap.REPEAT, + wrapT : TextureWrap.REPEAT + }); + } + + var usesTextureTransform = false; + var materials = model.gltf.materials; + var materialsLength = materials.length; + for (var i = 0; i < materialsLength; ++i) { + var material = materials[i]; + if (defined(material.extensions) && defined(material.extensions.KHR_techniques_webgl)) { + var values = material.extensions.KHR_techniques_webgl.values; + for (var valueName in values) { + if (values.hasOwnProperty(valueName) && valueName.indexOf('Texture') !== -1) { + var value = values[valueName]; + if (value.index === gltfTexture.id && defined(value.extensions) && defined(value.extensions.KHR_texture_transform)) { + usesTextureTransform = true; + break; + } + } + } + } + if (usesTextureTransform) { + break; + } + } + + var wrapS = sampler.wrapS; + var wrapT = sampler.wrapT; + var minFilter = sampler.minificationFilter; + + if (usesTextureTransform && minFilter !== TextureMinificationFilter.LINEAR && minFilter !== TextureMinificationFilter.NEAREST) { + if (minFilter === TextureMinificationFilter.NEAREST_MIPMAP_NEAREST || minFilter === TextureMinificationFilter.NEAREST_MIPMAP_LINEAR) { + minFilter = TextureMinificationFilter.NEAREST; + } else { + minFilter = TextureMinificationFilter.LINEAR; + } + + sampler = new Sampler({ + wrapS : sampler.wrapS, + wrapT : sampler.wrapT, + textureMinificationFilter : minFilter, + textureMagnificationFilter : sampler.magnificationFilter + }); + } var internalFormat = gltfTexture.internalFormat; var mipmap = (!(defined(internalFormat) && PixelFormat.isCompressedFormat(internalFormat))) && - ((sampler.minificationFilter === TextureMinificationFilter.NEAREST_MIPMAP_NEAREST) || - (sampler.minificationFilter === TextureMinificationFilter.NEAREST_MIPMAP_LINEAR) || - (sampler.minificationFilter === TextureMinificationFilter.LINEAR_MIPMAP_NEAREST) || - (sampler.minificationFilter === TextureMinificationFilter.LINEAR_MIPMAP_LINEAR)); + ((minFilter === TextureMinificationFilter.NEAREST_MIPMAP_NEAREST) || + (minFilter === TextureMinificationFilter.NEAREST_MIPMAP_LINEAR) || + (minFilter === TextureMinificationFilter.LINEAR_MIPMAP_NEAREST) || + (minFilter === TextureMinificationFilter.LINEAR_MIPMAP_LINEAR)); var requiresNpot = mipmap || - (sampler.wrapS === TextureWrap.REPEAT) || - (sampler.wrapS === TextureWrap.MIRRORED_REPEAT) || - (sampler.wrapT === TextureWrap.REPEAT) || - (sampler.wrapT === TextureWrap.MIRRORED_REPEAT); + (wrapS === TextureWrap.REPEAT) || + (wrapS === TextureWrap.MIRRORED_REPEAT) || + (wrapT === TextureWrap.REPEAT) || + (wrapT === TextureWrap.MIRRORED_REPEAT); var tx; var source = gltfTexture.image; diff --git a/Source/Scene/ModelUtility.js b/Source/Scene/ModelUtility.js index 391ce8ee8efd..53a517e95c67 100644 --- a/Source/Scene/ModelUtility.js +++ b/Source/Scene/ModelUtility.js @@ -493,6 +493,7 @@ define([ 'KHR_techniques_webgl' : true, 'KHR_materials_unlit' : true, 'KHR_materials_pbrSpecularGlossiness' : true, + 'KHR_texture_transform' : true, 'WEB3D_quantized_attributes' : true, 'EXT_texture_webp' : true }; diff --git a/Source/Scene/processPbrMaterials.js b/Source/Scene/processPbrMaterials.js index ca2f77f6943f..d772999bc0ab 100644 --- a/Source/Scene/processPbrMaterials.js +++ b/Source/Scene/processPbrMaterials.js @@ -87,6 +87,43 @@ define([ defined(material.extensions.KHR_materials_pbrSpecularGlossiness); } + function addTextureCoordinates(gltf, textureName, generatedMaterialValues, defaultTexCoord, result) { + var texCoord; + if (defined(generatedMaterialValues[textureName + 'Offset'])) { + var textureIndex = generatedMaterialValues[textureName].index; + var sampler = gltf.samplers[gltf.textures[textureIndex].sampler]; + + var repeatS = sampler.wrapS === WebGLConstants.REPEAT ? 'true' : 'false'; + var repeatT = sampler.wrapT === WebGLConstants.REPEAT ? 'true' : 'false'; + + texCoord = textureName + 'Coord'; + result.fragmentShaderMain += ' vec2 ' + texCoord + ' = computeTexCoord(' + defaultTexCoord + ', ' + textureName + 'Offset, ' + textureName + 'Rotation, ' + textureName + 'Scale, ' + repeatS + ', ' + repeatT + ');\n'; + } else { + texCoord = defaultTexCoord; + } + return texCoord; + } + + var DEFAULT_TEXTURE_OFFSET = [0.0, 0.0]; + var DEFAULT_TEXTURE_ROTATION = [0.0]; + var DEFAULT_TEXTURE_SCALE = [1.0, 1.0]; + + function handleKHRTextureTransform(parameterName, value, generatedMaterialValues) { + if (parameterName.indexOf('Texture') === -1 || !defined(value.extensions) || !defined(value.extensions.KHR_texture_transform)) { + return; + } + + var uniformName = 'u_' + parameterName; + var extension = value.extensions.KHR_texture_transform; + generatedMaterialValues[uniformName + 'Offset'] = defaultValue(extension.offset, DEFAULT_TEXTURE_OFFSET); + generatedMaterialValues[uniformName + 'Rotation'] = defaultValue(extension.rotation, DEFAULT_TEXTURE_ROTATION); + generatedMaterialValues[uniformName + 'Scale'] = defaultValue(extension.scale, DEFAULT_TEXTURE_SCALE); + + if (defined(value.texCoord) && defined(extension.texCoord)) { + generatedMaterialValues[uniformName].texCoord = extension.texCoord; + } + } + function generateTechnique(gltf, material, materialIndex, generatedMaterialValues, primitiveByMaterial, options) { var addBatchIdToGeneratedShaders = defaultValue(options.addBatchIdToGeneratedShaders, false); @@ -99,12 +136,15 @@ define([ var uniformName; var parameterName; + var value; var pbrMetallicRoughness = material.pbrMetallicRoughness; if (defined(pbrMetallicRoughness) && !useSpecGloss) { for (parameterName in pbrMetallicRoughness) { if (pbrMetallicRoughness.hasOwnProperty(parameterName)) { + value = pbrMetallicRoughness[parameterName]; uniformName = 'u_' + parameterName; - generatedMaterialValues[uniformName] = pbrMetallicRoughness[parameterName]; + generatedMaterialValues[uniformName] = value; + handleKHRTextureTransform(parameterName, value, generatedMaterialValues); } } } @@ -113,16 +153,20 @@ define([ var pbrSpecularGlossiness = material.extensions.KHR_materials_pbrSpecularGlossiness; for (parameterName in pbrSpecularGlossiness) { if (pbrSpecularGlossiness.hasOwnProperty(parameterName)) { + value = pbrSpecularGlossiness[parameterName]; uniformName = 'u_' + parameterName; - generatedMaterialValues[uniformName] = pbrSpecularGlossiness[parameterName]; + generatedMaterialValues[uniformName] = value; + handleKHRTextureTransform(parameterName, value, generatedMaterialValues); } } } for (var additional in material) { if (material.hasOwnProperty(additional) && ((additional.indexOf('Texture') >= 0) || additional.indexOf('Factor') >= 0)) { + value = material[additional]; uniformName = 'u_' + additional; - generatedMaterialValues[uniformName] = material[additional]; + generatedMaterialValues[uniformName] = value; + handleKHRTextureTransform(additional, value, generatedMaterialValues); } } @@ -371,19 +415,42 @@ define([ fragmentShader += 'varying vec4 v_tangent;\n'; } + var fragmentShaderMain = ''; + // Add texture coordinates if the material uses them - var v_texcoord; + var v_texCoord; + var normalTexCoord; + var baseColorTexCoord; + var specularGlossinessTexCoord; + var diffuseTexCoord; + var metallicRoughnessTexCoord; + var occlusionTexCoord; + var emissiveTexCoord; + if (hasTexCoords) { techniqueAttributes.a_texcoord_0 = { semantic : 'TEXCOORD_0' }; - v_texcoord = 'v_texcoord_0'; + v_texCoord = 'v_texcoord_0'; vertexShader += 'attribute vec2 a_texcoord_0;\n'; - vertexShader += 'varying vec2 ' + v_texcoord + ';\n'; - vertexShaderMain += ' ' + v_texcoord + ' = a_texcoord_0;\n'; + vertexShader += 'varying vec2 ' + v_texCoord + ';\n'; + vertexShaderMain += ' ' + v_texCoord + ' = a_texcoord_0;\n'; + + fragmentShader += 'varying vec2 ' + v_texCoord + ';\n'; + + var result = { + fragmentShaderMain : fragmentShaderMain + }; + normalTexCoord = addTextureCoordinates(gltf, 'u_normalTexture', generatedMaterialValues, v_texCoord, result); + baseColorTexCoord = addTextureCoordinates(gltf, 'u_baseColorTexture', generatedMaterialValues, v_texCoord, result); + specularGlossinessTexCoord = addTextureCoordinates(gltf, 'u_specularGlossinessTexture', generatedMaterialValues, v_texCoord, result); + diffuseTexCoord = addTextureCoordinates(gltf, 'u_diffuseTexture', generatedMaterialValues, v_texCoord, result); + metallicRoughnessTexCoord = addTextureCoordinates(gltf, 'u_metallicRoughnessTexture', generatedMaterialValues, v_texCoord, result); + occlusionTexCoord = addTextureCoordinates(gltf, 'u_occlusionTexture', generatedMaterialValues, v_texCoord, result); + emissiveTexCoord = addTextureCoordinates(gltf, 'u_emmissiveTexture', generatedMaterialValues, v_texCoord, result); - fragmentShader += 'varying vec2 ' + v_texcoord + ';\n'; + fragmentShaderMain = result.fragmentShaderMain; } // Add skinning information if available @@ -488,6 +555,20 @@ define([ '#endif \n' + '}\n\n'; + fragmentShader += + 'vec2 computeTexCoord(vec2 texCoords, vec2 offset, float rotation, vec2 scale, bool repeatS, bool repeatT) \n' + + '{\n' + + ' rotation = -rotation; \n' + + ' mat3 transform = mat3(\n' + + ' cos(rotation) * scale.x, sin(rotation) * scale.x, 0.0, \n' + + ' -sin(rotation) * scale.y, cos(rotation) * scale.y, 0.0, \n' + + ' offset.x, offset.y, 1.0); \n' + + ' vec2 transformedTexCoords = (transform * vec3(fract(texCoords), 1.0)).xy; \n' + + ' transformedTexCoords.x = repeatS ? fract(transformedTexCoords.x) : clamp(transformedTexCoords.x, 0.0, 1.0); \n' + + ' transformedTexCoords.y = repeatT ? fract(transformedTexCoords.y) : clamp(transformedTexCoords.y, 0.0, 1.0); \n' + + ' return transformedTexCoords; \n' + + '}\n\n'; + fragmentShader += '#ifdef USE_IBL_LIGHTING \n'; fragmentShader += 'uniform vec2 gltf_iblFactor; \n'; fragmentShader += '#endif \n'; @@ -496,6 +577,7 @@ define([ fragmentShader += '#endif \n'; fragmentShader += 'void main(void) \n{\n'; + fragmentShader += fragmentShaderMain; // Add normal mapping to fragment shader if (hasNormals) { @@ -506,7 +588,7 @@ define([ fragmentShader += ' vec3 t = normalize(v_tangent.xyz);\n'; fragmentShader += ' vec3 b = normalize(cross(ng, t) * v_tangent.w);\n'; fragmentShader += ' mat3 tbn = mat3(t, b, ng);\n'; - fragmentShader += ' vec3 n = texture2D(u_normalTexture, ' + v_texcoord + ').rgb;\n'; + fragmentShader += ' vec3 n = texture2D(u_normalTexture, ' + normalTexCoord + ').rgb;\n'; fragmentShader += ' n = normalize(tbn * (2.0 * n - 1.0));\n'; } else { // Add standard derivatives extension @@ -518,13 +600,13 @@ define([ fragmentShader += '#ifdef GL_OES_standard_derivatives\n'; fragmentShader += ' vec3 pos_dx = dFdx(v_positionEC);\n'; fragmentShader += ' vec3 pos_dy = dFdy(v_positionEC);\n'; - fragmentShader += ' vec3 tex_dx = dFdx(vec3(' + v_texcoord + ',0.0));\n'; - fragmentShader += ' vec3 tex_dy = dFdy(vec3(' + v_texcoord + ',0.0));\n'; + fragmentShader += ' vec3 tex_dx = dFdx(vec3(' + normalTexCoord + ',0.0));\n'; + fragmentShader += ' vec3 tex_dy = dFdy(vec3(' + normalTexCoord + ',0.0));\n'; fragmentShader += ' vec3 t = (tex_dy.t * pos_dx - tex_dx.t * pos_dy) / (tex_dx.s * tex_dy.t - tex_dy.s * tex_dx.t);\n'; fragmentShader += ' t = normalize(t - ng * dot(ng, t));\n'; fragmentShader += ' vec3 b = normalize(cross(ng, t));\n'; fragmentShader += ' mat3 tbn = mat3(t, b, ng);\n'; - fragmentShader += ' vec3 n = texture2D(u_normalTexture, ' + v_texcoord + ').rgb;\n'; + fragmentShader += ' vec3 n = texture2D(u_normalTexture, ' + normalTexCoord + ').rgb;\n'; fragmentShader += ' n = normalize(tbn * (2.0 * n - 1.0));\n'; fragmentShader += '#else\n'; fragmentShader += ' vec3 n = ng;\n'; @@ -543,7 +625,7 @@ define([ // Add base color to fragment shader if (defined(generatedMaterialValues.u_baseColorTexture)) { - fragmentShader += ' vec4 baseColorWithAlpha = SRGBtoLINEAR4(texture2D(u_baseColorTexture, ' + v_texcoord + '));\n'; + fragmentShader += ' vec4 baseColorWithAlpha = SRGBtoLINEAR4(texture2D(u_baseColorTexture, ' + baseColorTexCoord + '));\n'; if (defined(generatedMaterialValues.u_baseColorFactor)) { fragmentShader += ' baseColorWithAlpha *= u_baseColorFactor;\n'; } @@ -562,7 +644,7 @@ define([ if (hasNormals) { if (useSpecGloss) { if (defined(generatedMaterialValues.u_specularGlossinessTexture)) { - fragmentShader += ' vec4 specularGlossiness = SRGBtoLINEAR4(texture2D(u_specularGlossinessTexture, ' + v_texcoord + '));\n'; + fragmentShader += ' vec4 specularGlossiness = SRGBtoLINEAR4(texture2D(u_specularGlossinessTexture, ' + specularGlossinessTexCoord + '));\n'; fragmentShader += ' vec3 specular = specularGlossiness.rgb;\n'; fragmentShader += ' float glossiness = specularGlossiness.a;\n'; if (defined(generatedMaterialValues.u_specularFactor)) { @@ -584,7 +666,7 @@ define([ } } if (defined(generatedMaterialValues.u_diffuseTexture)) { - fragmentShader += ' vec4 diffuse = SRGBtoLINEAR4(texture2D(u_diffuseTexture, ' + v_texcoord + '));\n'; + fragmentShader += ' vec4 diffuse = SRGBtoLINEAR4(texture2D(u_diffuseTexture, ' + diffuseTexCoord + '));\n'; if (defined(generatedMaterialValues.u_diffuseFactor)) { fragmentShader += ' diffuse *= u_diffuseFactor;\n'; } @@ -594,7 +676,7 @@ define([ fragmentShader += ' vec4 diffuse = vec4(1.0);\n'; } } else if (defined(generatedMaterialValues.u_metallicRoughnessTexture)) { - fragmentShader += ' vec3 metallicRoughness = texture2D(u_metallicRoughnessTexture, ' + v_texcoord + ').rgb;\n'; + fragmentShader += ' vec3 metallicRoughness = texture2D(u_metallicRoughnessTexture, ' + metallicRoughnessTexCoord + ').rgb;\n'; fragmentShader += ' float metalness = clamp(metallicRoughness.b, 0.0, 1.0);\n'; fragmentShader += ' float roughness = clamp(metallicRoughness.g, 0.04, 1.0);\n'; if (defined(generatedMaterialValues.u_metallicFactor)) { @@ -763,10 +845,10 @@ define([ // Ignore occlusion and emissive when unlit if (!isUnlit) { if (defined(generatedMaterialValues.u_occlusionTexture)) { - fragmentShader += ' color *= texture2D(u_occlusionTexture, ' + v_texcoord + ').r;\n'; + fragmentShader += ' color *= texture2D(u_occlusionTexture, ' + occlusionTexCoord + ').r;\n'; } if (defined(generatedMaterialValues.u_emissiveTexture)) { - fragmentShader += ' vec3 emissive = SRGBtoLINEAR3(texture2D(u_emissiveTexture, ' + v_texcoord + ').rgb);\n'; + fragmentShader += ' vec3 emissive = SRGBtoLINEAR3(texture2D(u_emissiveTexture, ' + emissiveTexCoord + ').rgb);\n'; if (defined(generatedMaterialValues.u_emissiveFactor)) { fragmentShader += ' emissive *= u_emissiveFactor;\n'; } @@ -831,6 +913,16 @@ define([ } function getPBRValueType(paramName) { + if (paramName.indexOf('Offset') !== -1) { + return WebGLConstants.FLOAT_VEC2; + } else if (paramName.indexOf('Rotation') !== -1) { + return WebGLConstants.FLOAT; + } else if (paramName.indexOf('Scale') !== -1) { + return WebGLConstants.FLOAT_VEC2; + } else if (paramName.indexOf('Texture') !== -1) { + return WebGLConstants.SAMPLER_2D; + } + switch (paramName) { case 'u_baseColorFactor': return WebGLConstants.FLOAT_VEC4; @@ -838,16 +930,6 @@ define([ return WebGLConstants.FLOAT; case 'u_roughnessFactor': return WebGLConstants.FLOAT; - case 'u_baseColorTexture': - return WebGLConstants.SAMPLER_2D; - case 'u_metallicRoughnessTexture': - return WebGLConstants.SAMPLER_2D; - case 'u_normalTexture': - return WebGLConstants.SAMPLER_2D; - case 'u_occlusionTexture': - return WebGLConstants.SAMPLER_2D; - case 'u_emissiveTexture': - return WebGLConstants.SAMPLER_2D; case 'u_emissiveFactor': return WebGLConstants.FLOAT_VEC3; // Specular Glossiness Types @@ -857,10 +939,6 @@ define([ return WebGLConstants.FLOAT_VEC3; case 'u_glossinessFactor': return WebGLConstants.FLOAT; - case 'u_diffuseTexture': - return WebGLConstants.SAMPLER_2D; - case 'u_specularGlossinessTexture': - return WebGLConstants.SAMPLER_2D; } } diff --git a/Specs/Data/Models/Box-Texture-Transform/CesiumTexturedBoxTest.gltf b/Specs/Data/Models/Box-Texture-Transform/CesiumTexturedBoxTest.gltf new file mode 100644 index 000000000000..c16c964666b9 --- /dev/null +++ b/Specs/Data/Models/Box-Texture-Transform/CesiumTexturedBoxTest.gltf @@ -0,0 +1,188 @@ +{ + "asset": { + "generator": "COLLADA2GLTF", + "version": "2.0" + }, + "scenes": [ + { + "nodes": [ + 0 + ] + } + ], + "scene": 0, + "nodes": [ + { + "children": [ + 1 + ], + "matrix": [ + 1.0, + 0.0, + 0.0, + 0.0, + 0.0, + 0.0, + -1.0, + 0.0, + 0.0, + 1.0, + 0.0, + 0.0, + 0.0, + 0.0, + 0.0, + 1.0 + ], + "name": "Z_UP" + }, + { + "mesh": 0, + "name": "Mesh" + } + ], + "meshes": [ + { + "primitives": [ + { + "attributes": { + "NORMAL": 1, + "POSITION": 2, + "TEXCOORD_0": 3 + }, + "indices": 0, + "mode": 4, + "material": 0 + } + ], + "name": "Mesh" + } + ], + "accessors": [ + { + "bufferView": 0, + "byteOffset": 0, + "componentType": 5123, + "count": 36, + "max": [ + 23 + ], + "min": [ + 0 + ], + "type": "SCALAR" + }, + { + "bufferView": 1, + "byteOffset": 0, + "componentType": 5126, + "count": 24, + "max": [ + 1.0, + 1.0, + 1.0 + ], + "min": [ + -1.0, + -1.0, + -1.0 + ], + "type": "VEC3" + }, + { + "bufferView": 1, + "byteOffset": 288, + "componentType": 5126, + "count": 24, + "max": [ + 0.5, + 0.5, + 0.5 + ], + "min": [ + -0.5, + -0.5, + -0.5 + ], + "type": "VEC3" + }, + { + "bufferView": 2, + "byteOffset": 0, + "componentType": 5126, + "count": 24, + "max": [ + 6.0, + 1.0 + ], + "min": [ + 0.0, + 0.0 + ], + "type": "VEC2" + } + ], + "materials": [ + { + "pbrMetallicRoughness": { + "baseColorTexture": { + "index": 0, + "extensions": { + "KHR_texture_transform": { + "scale": [5.0, 3.0] + } + } + }, + "metallicFactor": 0.0 + }, + "name": "Texture" + } + ], + "textures": [ + { + "sampler": 0, + "source": 0 + } + ], + "images": [ + { + "uri": "" + } + ], + "samplers": [ + { + "magFilter": 9729, + "minFilter": 9986, + "wrapS": 10497, + "wrapT": 10497 + } + ], + "bufferViews": [ + { + "buffer": 0, + "byteOffset": 768, + "byteLength": 72, + "target": 34963 + }, + { + "buffer": 0, + "byteOffset": 0, + "byteLength": 576, + "byteStride": 12, + "target": 34962 + }, + { + "buffer": 0, + "byteOffset": 576, + "byteLength": 192, + "byteStride": 8, + "target": 34962 + } + ], + "buffers": [ + { + "byteLength": 840, + "uri": "data:application/octet-stream;base64,AAAAAAAAAAAAAIA/AAAAAAAAAAAAAIA/AAAAAAAAAAAAAIA/AAAAAAAAAAAAAIA/AACAPwAAAAAAAAAAAACAPwAAAAAAAAAAAACAPwAAAAAAAAAAAACAPwAAAAAAAAAAAAAAAAAAgD8AAAAAAAAAAAAAgD8AAAAAAAAAAAAAgD8AAAAAAAAAAAAAgD8AAAAAAAAAAAAAgL8AAAAAAAAAAAAAgL8AAAAAAAAAAAAAgL8AAAAAAAAAAAAAgL8AAAAAAACAvwAAAAAAAAAAAACAvwAAAAAAAAAAAACAvwAAAAAAAAAAAACAvwAAAAAAAAAAAAAAAAAAAAAAAIC/AAAAAAAAAAAAAIC/AAAAAAAAAAAAAIC/AAAAAAAAAAAAAIC/AAAAvwAAAL8AAAA/AAAAPwAAAL8AAAA/AAAAvwAAAD8AAAA/AAAAPwAAAD8AAAA/AAAAPwAAAD8AAAA/AAAAPwAAAL8AAAA/AAAAPwAAAD8AAAC/AAAAPwAAAL8AAAC/AAAAvwAAAD8AAAA/AAAAPwAAAD8AAAA/AAAAvwAAAD8AAAC/AAAAPwAAAD8AAAC/AAAAPwAAAL8AAAA/AAAAvwAAAL8AAAA/AAAAPwAAAL8AAAC/AAAAvwAAAL8AAAC/AAAAvwAAAL8AAAA/AAAAvwAAAD8AAAA/AAAAvwAAAL8AAAC/AAAAvwAAAD8AAAC/AAAAvwAAAL8AAAC/AAAAvwAAAD8AAAC/AAAAPwAAAL8AAAC/AAAAPwAAAD8AAAC/AADAQAAAAAAAAKBAAAAAAAAAwED+/38/AACgQP7/fz8AAIBAAAAAAAAAoEAAAAAAAACAQAAAgD8AAKBAAACAPwAAAEAAAAAAAACAPwAAAAAAAABAAACAPwAAgD8AAIA/AABAQAAAAAAAAIBAAAAAAAAAQEAAAIA/AACAQAAAgD8AAEBAAAAAAAAAAEAAAAAAAABAQAAAgD8AAABAAACAPwAAAAAAAAAAAAAAAP7/fz8AAIA/AAAAAAAAgD/+/38/AAABAAIAAwACAAEABAAFAAYABwAGAAUACAAJAAoACwAKAAkADAANAA4ADwAOAA0AEAARABIAEwASABEAFAAVABYAFwAWABUA" + } + ] +} diff --git a/Specs/Scene/ModelSpec.js b/Specs/Scene/ModelSpec.js index 890c5c3db4ea..7beafda1a432 100644 --- a/Specs/Scene/ModelSpec.js +++ b/Specs/Scene/ModelSpec.js @@ -90,6 +90,7 @@ defineSuite([ var texturedBoxCRNEmbeddedUrl = './Data/Models/Box-Textured-CRN-Embedded/CesiumTexturedBoxTest.gltf'; var texturedBoxCustomUrl = './Data/Models/Box-Textured-Custom/CesiumTexturedBoxTest.gltf'; var texturedBoxKhrBinaryUrl = './Data/Models/Box-Textured-Binary/CesiumTexturedBoxTest.glb'; + var texturedBoxTextureTransformUrl = './Data/Models/Box-Texture-Transform/CesiumTexturedBoxTest.gltf'; var texturedBoxWebpUrl = './Data/Models/Box-Textured-Webp/CesiumBoxWebp.gltf'; var boxRtcUrl = './Data/Models/Box-RTC/Box.gltf'; var boxEcefUrl = './Data/Models/Box-ECEF/ecef.gltf'; @@ -1208,6 +1209,16 @@ defineSuite([ }); }); + it('renders textured box with KHR_texture_transform extension', function() { + return loadModel(texturedBoxTextureTransformUrl, { + incrementallyLoadTextures : false + }).then(function(m) { + verifyRender(m); + expect(Object.keys(m._rendererResources.textures).length).toBe(1); + primitives.remove(m); + }); + }); + it('renders textured box with WebP texture', function() { if (!FeatureDetection.supportsWebP()) { return;