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;