WebGLPrograms.js 7.2 KB
Newer Older
M
Mr.doob 已提交
1
THREE.WebGLPrograms = function ( renderer, capabilities ) {
G
gero3 已提交
2

3
	var programs = [];
G
gero3 已提交
4

G
gero3 已提交
5 6 7 8 9 10
	var shaderIDs = {
		MeshDepthMaterial: 'depth',
		MeshNormalMaterial: 'normal',
		MeshBasicMaterial: 'basic',
		MeshLambertMaterial: 'lambert',
		MeshPhongMaterial: 'phong',
11
		MeshStandardMaterial: 'standard',
G
gero3 已提交
12 13
		LineBasicMaterial: 'basic',
		LineDashedMaterial: 'dashed',
14
		PointsMaterial: 'points'
G
gero3 已提交
15
	};
G
gero3 已提交
16

M
Mr.doob 已提交
17
	var parameterNames = [
18 19
		"precision", "supportsVertexTextures", "map", "mapEncoding", "envMap", "envMapMode", "envMapEncoding",
		"lightMap", "aoMap", "emissiveMap", "emissiveMapEncoding", "bumpMap", "normalMap", "displacementMap", "specularMap",
20
		"roughnessMap", "metalnessMap",
M
Mr.doob 已提交
21 22 23
		"alphaMap", "combine", "vertexColors", "fog", "useFog", "fogExp",
		"flatShading", "sizeAttenuation", "logarithmicDepthBuffer", "skinning",
		"maxBones", "useVertexTexture", "morphTargets", "morphNormals",
24
		"maxMorphTargets", "maxMorphNormals", "premultipliedAlpha",
M
Mr.doob 已提交
25
		"numDirLights", "numPointLights", "numSpotLights", "numHemiLights",
26
		"shadowMapEnabled", "pointLightShadows", "toneMapping", 'physicallyCorrectLights',
M
Mr.doob 已提交
27
		"shadowMapType",
M
Mr.doob 已提交
28
		"alphaTest", "doubleSided", "flipSided"
M
Mr.doob 已提交
29
	];
G
gero3 已提交
30 31


G
gero3 已提交
32 33
	function allocateBones ( object ) {

G
gero3 已提交
34
		if ( capabilities.floatVertexTextures && object && object.skeleton && object.skeleton.useVertexTexture ) {
G
gero3 已提交
35 36 37 38 39 40 41 42 43 44 45 46

			return 1024;

		} else {

			// default for when object is not specified
			// ( for example when prebuilding shader to be used with multiple objects )
			//
			//  - leave some extra space for other uniforms
			//  - limit here is ANGLE's 254 max uniform vectors
			//    (up to 54 should be safe)

G
gero3 已提交
47
			var nVertexUniforms = capabilities.maxVertexUniforms;
G
gero3 已提交
48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68
			var nVertexMatrices = Math.floor( ( nVertexUniforms - 20 ) / 4 );

			var maxBones = nVertexMatrices;

			if ( object !== undefined && object instanceof THREE.SkinnedMesh ) {

				maxBones = Math.min( object.skeleton.bones.length, maxBones );

				if ( maxBones < object.skeleton.bones.length ) {

					console.warn( 'WebGLRenderer: too many bones - ' + object.skeleton.bones.length + ', this GPU supports just ' + maxBones + ' (try OpenGL instead of ANGLE)' );

				}

			}

			return maxBones;

		}

	}
G
gero3 已提交
69

70
	function getTextureEncodingFromMap( map, gammaOverrideLinear ) {
71 72

		var encoding;
M
Mr.doob 已提交
73 74

		if ( ! map ) {
M
Mr.doob 已提交
75

76
			encoding = THREE.LinearEncoding;
M
Mr.doob 已提交
77

78
		} else if ( map instanceof THREE.Texture ) {
M
Mr.doob 已提交
79

80
			encoding = map.encoding;
M
Mr.doob 已提交
81

82
		} else if ( map instanceof THREE.WebGLRenderTarget ) {
M
Mr.doob 已提交
83

84
			encoding = map.texture.encoding;
M
Mr.doob 已提交
85

86 87 88
		}

		// add backwards compatibility for WebGLRenderer.gammaInput/gammaOutput parameter, should probably be removed at some point.
M
Mr.doob 已提交
89
		if ( encoding === THREE.LinearEncoding && gammaOverrideLinear ) {
M
Mr.doob 已提交
90

91
			encoding = THREE.GammaEncoding;
M
Mr.doob 已提交
92

93 94 95
		}

		return encoding;
M
Mr.doob 已提交
96

97 98
	}

M
Mr.doob 已提交
99
	this.getParameters = function ( material, lights, fog, object ) {
M
Mr.doob 已提交
100

G
gero3 已提交
101
		var shaderID = shaderIDs[ material.type ];
M
Mr.doob 已提交
102

G
gero3 已提交
103
		// heuristics to create shader parameters according to lights in the scene
G
gero3 已提交
104 105 106
		// (not to blow over maxLights budget)

		var maxBones = allocateBones( object );
M
Mr.doob 已提交
107
		var precision = renderer.getPrecision();
G
gero3 已提交
108 109 110

		if ( material.precision !== null ) {

G
gero3 已提交
111
			precision = capabilities.getMaxPrecision( material.precision );
G
gero3 已提交
112 113 114

			if ( precision !== material.precision ) {

115
				console.warn( 'THREE.WebGLProgram.getParameters:', material.precision, 'not supported, using', precision, 'instead.' );
G
gero3 已提交
116 117 118 119

			}

		}
M
Mr.doob 已提交
120

G
gero3 已提交
121
		var parameters = {
G
gero3 已提交
122 123

			shaderID: shaderID,
G
gero3 已提交
124 125

			precision: precision,
G
gero3 已提交
126
			supportsVertexTextures: capabilities.vertexTextures,
127
			outputEncoding: getTextureEncodingFromMap( renderer.getCurrentRenderTarget(), renderer.gammaOutput ),
G
gero3 已提交
128
			map: !! material.map,
129
			mapEncoding: getTextureEncodingFromMap( material.map, renderer.gammaInput ),
G
gero3 已提交
130 131
			envMap: !! material.envMap,
			envMapMode: material.envMap && material.envMap.mapping,
132
			envMapEncoding: getTextureEncodingFromMap( material.envMap, renderer.gammaInput ),
M
Mr.doob 已提交
133
			envMapCubeUV: ( !! material.envMap ) && ( ( material.envMap.mapping === THREE.CubeUVReflectionMapping ) || ( material.envMap.mapping === THREE.CubeUVRefractionMapping ) ),
G
gero3 已提交
134 135 136
			lightMap: !! material.lightMap,
			aoMap: !! material.aoMap,
			emissiveMap: !! material.emissiveMap,
137
			emissiveMapEncoding: getTextureEncodingFromMap( material.emissiveMap, renderer.gammaInput ),
G
gero3 已提交
138 139
			bumpMap: !! material.bumpMap,
			normalMap: !! material.normalMap,
140
			displacementMap: !! material.displacementMap,
W
WestLangley 已提交
141 142
			roughnessMap: !! material.roughnessMap,
			metalnessMap: !! material.metalnessMap,
G
gero3 已提交
143 144 145 146 147 148 149 150 151 152 153 154 155 156
			specularMap: !! material.specularMap,
			alphaMap: !! material.alphaMap,

			combine: material.combine,

			vertexColors: material.vertexColors,

			fog: fog,
			useFog: material.fog,
			fogExp: fog instanceof THREE.FogExp2,

			flatShading: material.shading === THREE.FlatShading,

			sizeAttenuation: material.sizeAttenuation,
157
			logarithmicDepthBuffer: capabilities.logarithmicDepthBuffer,
G
gero3 已提交
158 159 160

			skinning: material.skinning,
			maxBones: maxBones,
G
gero3 已提交
161
			useVertexTexture: capabilities.floatVertexTextures && object && object.skeleton && object.skeleton.useVertexTexture,
G
gero3 已提交
162 163 164

			morphTargets: material.morphTargets,
			morphNormals: material.morphNormals,
M
Mr.doob 已提交
165 166
			maxMorphTargets: renderer.maxMorphTargets,
			maxMorphNormals: renderer.maxMorphNormals,
G
gero3 已提交
167

168 169 170 171
			numDirLights: lights.directional.length,
			numPointLights: lights.point.length,
			numSpotLights: lights.spot.length,
			numHemiLights: lights.hemi.length,
172 173

			pointLightShadows: lights.shadowsPointLight,
G
gero3 已提交
174

175
			shadowMapEnabled: renderer.shadowMap.enabled && object.receiveShadow && lights.shadows > 0,
M
Mr.doob 已提交
176
			shadowMapType: renderer.shadowMap.type,
G
gero3 已提交
177

B
Ben Houston 已提交
178
			toneMapping: renderer.toneMapping,
179
			physicallyCorrectLights: renderer.physicallyCorrectLights,
B
Ben Houston 已提交
180

181 182
			premultipliedAlpha: material.premultipliedAlpha,

G
gero3 已提交
183 184 185 186 187
			alphaTest: material.alphaTest,
			doubleSided: material.side === THREE.DoubleSide,
			flipSided: material.side === THREE.BackSide

		};
188

G
gero3 已提交
189
		return parameters;
G
gero3 已提交
190

G
gero3 已提交
191
	};
G
gero3 已提交
192

M
Mr.doob 已提交
193
	this.getProgramCode = function ( material, parameters ) {
G
gero3 已提交
194

G
gero3 已提交
195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218
		var chunks = [];

		if ( parameters.shaderID ) {

			chunks.push( parameters.shaderID );

		} else {

			chunks.push( material.fragmentShader );
			chunks.push( material.vertexShader );

		}

		if ( material.defines !== undefined ) {

			for ( var name in material.defines ) {

				chunks.push( name );
				chunks.push( material.defines[ name ] );

			}

		}

G
gero3 已提交
219 220 221
		for ( var i = 0; i < parameterNames.length; i ++ ) {

			var parameterName = parameterNames[ i ];
G
gero3 已提交
222 223 224 225 226 227
			chunks.push( parameterName );
			chunks.push( parameters[ parameterName ] );

		}

		return chunks.join();
G
gero3 已提交
228 229 230

	};

231
	this.acquireProgram = function ( material, parameters, code ) {
G
gero3 已提交
232 233 234 235

		var program;

		// Check if code has been already compiled
G
gero3 已提交
236 237 238 239 240 241 242
		for ( var p = 0, pl = programs.length; p < pl; p ++ ) {

			var programInfo = programs[ p ];

			if ( programInfo.code === code ) {

				program = programInfo;
243
				++ program.usedTimes;
G
gero3 已提交
244 245 246 247 248 249 250 251

				break;

			}

		}

		if ( program === undefined ) {
G
gero3 已提交
252

M
Mr.doob 已提交
253
			program = new THREE.WebGLProgram( renderer, code, material, parameters );
G
gero3 已提交
254
			programs.push( program );
G
gero3 已提交
255

G
gero3 已提交
256
		}
G
gero3 已提交
257

258
		return program;
G
gero3 已提交
259

260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276
	};

	this.releaseProgram = function( program ) {

		if ( -- program.usedTimes === 0 ) {

			// Remove from unordered set
			var i = programs.indexOf( program );
			programs[ i ] = programs[ programs.length - 1 ];
			programs.pop();

			// Free WebGL resources
			program.destroy();

		}

	};
G
gero3 已提交
277

278 279 280
	// Exposed for resource monitoring & error feedback via renderer.info:
	this.programs = programs;

G
gero3 已提交
281
};