WebGLMultiview.js 4.8 KB
Newer Older
1 2
/**
 * @author fernandojsg / http://fernandojsg.com
3
 * @author Takahiro https://github.com/takahirox
4 5
 */

6
import { WebGLMultiviewRenderTarget } from '../WebGLMultiviewRenderTarget.js';
7 8
import { Matrix3 } from '../../math/Matrix3.js';
import { Matrix4 } from '../../math/Matrix4.js';
9
import { Vector2 } from '../../math/Vector2.js';
10

11
function WebGLMultiview( renderer, gl ) {
12

13
	var DEFAULT_NUMVIEWS = 2;
14

15
	var extensions = renderer.extensions;
16
	var properties = renderer.properties;
17

18
	var renderTarget, currentRenderTarget;
19
	var mat3, mat4, cameraArray, renderSize;
20

21 22
	var available;
	var maxNumViews = 0;
23

24
	//
25

26
	function isAvailable() {
27

28
		if ( available === undefined ) {
29

30
			var extension = extensions.get( 'OVR_multiview2' );
31

32 33 34 35 36 37 38 39 40 41
			available = extension !== null && gl.getContextAttributes().antialias === false;

			if ( available ) {

				maxNumViews = gl.getParameter( extension.MAX_VIEWS_OVR );
				renderTarget = new WebGLMultiviewRenderTarget( 0, 0, DEFAULT_NUMVIEWS );

				renderSize = new Vector2();
				mat4 = [];
				mat3 = [];
42
				cameraArray = [];
43

44
				for ( var i = 0; i < maxNumViews; i ++ ) {
45 46 47 48 49 50 51 52 53 54 55 56 57

					mat4[ i ] = new Matrix4();
					mat3[ i ] = new Matrix3();

				}

			}

		}

		return available;

	}
58

59 60 61 62 63 64 65 66 67 68
	function getCameraArray( camera ) {

		if ( camera.isArrayCamera ) return camera.cameras;

		cameraArray[ 0 ] = camera;

		return cameraArray;

	}

69
	function updateCameraProjectionMatricesUniform( camera, uniforms ) {
70

71
		var cameras = getCameraArray( camera );
72

73
		for ( var i = 0; i < cameras.length; i ++ ) {
74

75
			mat4[ i ].copy( cameras[ i ].projectionMatrix );
76 77 78

		}

79
		uniforms.setValue( gl, 'projectionMatrices', mat4 );
80

81
	}
82

83
	function updateCameraViewMatricesUniform( camera, uniforms ) {
84

85
		var cameras = getCameraArray( camera );
86

87
		for ( var i = 0; i < cameras.length; i ++ ) {
88

89
			mat4[ i ].copy( cameras[ i ].matrixWorldInverse );
90 91 92

		}

93
		uniforms.setValue( gl, 'viewMatrices', mat4 );
94

95
	}
96

97
	function updateObjectMatricesUniforms( object, camera, uniforms ) {
98

99
		var cameras = getCameraArray( camera );
100

101
		for ( var i = 0; i < cameras.length; i ++ ) {
102

103 104
			mat4[ i ].multiplyMatrices( cameras[ i ].matrixWorldInverse, object.matrixWorld );
			mat3[ i ].getNormalMatrix( mat4[ i ] );
105 106 107

		}

108 109
		uniforms.setValue( gl, 'modelViewMatrices', mat4 );
		uniforms.setValue( gl, 'normalMatrices', mat3 );
110

111 112
	}

113 114
	function isMultiviewCompatible( camera ) {

115 116
		if ( camera.isArrayCamera === undefined ) return true;

117 118 119 120 121 122
		var cameras = camera.cameras;

		if ( cameras.length > maxNumViews ) return false;

		for ( var i = 1, il = cameras.length; i < il; i ++ ) {

123 124
			if ( cameras[ 0 ].viewport.z !== cameras[ i ].viewport.z ||
				cameras[ 0 ].viewport.w !== cameras[ i ].viewport.w ) return false;
125 126 127 128 129 130

		}

		return true;

	}
131

132
	function resizeRenderTarget( camera ) {
133

134 135 136 137 138 139 140
		if ( currentRenderTarget ) {

			renderSize.set( currentRenderTarget.width, currentRenderTarget.height );

		} else {

			renderer.getDrawingBufferSize( renderSize );
141

142
		}
F
Fernando Serrano 已提交
143

144 145 146
		if ( camera.isArrayCamera ) {

			var viewport = camera.cameras[ 0 ].viewport;
147

148 149 150 151 152 153 154 155 156
			renderTarget.setSize( viewport.z, viewport.w );
			renderTarget.setNumViews( camera.cameras.length );

		} else {

			renderTarget.setSize( renderSize.x, renderSize.y );
			renderTarget.setNumViews( DEFAULT_NUMVIEWS );

		}
F
Fernando Serrano 已提交
157

158 159
	}

160
	function attachCamera( camera ) {
161

162
		if ( isMultiviewCompatible( camera ) === false ) return;
163

164 165
		currentRenderTarget = renderer.getRenderTarget();
		resizeRenderTarget( camera );
166
		renderer.setRenderTarget( renderTarget );
167

168 169
	}

F
Fernando Serrano 已提交
170
	function detachCamera( camera ) {
171

F
Fernando Serrano 已提交
172
		if ( renderTarget !== renderer.getRenderTarget() ) return;
173

174
		renderer.setRenderTarget( currentRenderTarget );
175

176
		flush( camera );
177

178
	}
179

180
	function flush( camera ) {
181

182 183 184 185 186 187 188
		var srcRenderTarget = renderTarget;
		var numViews = srcRenderTarget.numViews;

		var srcFramebuffers = properties.get( srcRenderTarget ).__webglViewFramebuffers;

		var viewWidth = srcRenderTarget.width;
		var viewHeight = srcRenderTarget.height;
189

190 191 192
		if ( camera.isArrayCamera ) {

			for ( var i = 0; i < numViews; i ++ ) {
193

194
				var viewport = camera.cameras[ i ].viewport;
195

196 197 198 199 200 201 202 203 204 205 206
				var x1 = viewport.x;
				var y1 = viewport.y;
				var x2 = x1 + viewport.z;
				var y2 = y1 + viewport.w;

				gl.bindFramebuffer( gl.READ_FRAMEBUFFER, srcFramebuffers[ i ] );
				gl.blitFramebuffer( 0, 0, viewWidth, viewHeight, x1, y1, x2, y2, gl.COLOR_BUFFER_BIT, gl.NEAREST );

			}

		} else {
207

208 209
			gl.bindFramebuffer( gl.READ_FRAMEBUFFER, srcFramebuffers[ 0 ] );
			gl.blitFramebuffer( 0, 0, viewWidth, viewHeight, 0, 0, renderSize.x, renderSize.y, gl.COLOR_BUFFER_BIT, gl.NEAREST );
210 211 212

		}

F
Fernando Serrano 已提交
213
	}
214

215
	this.isAvailable = isAvailable;
216
	this.attachCamera = attachCamera;
F
Fernando Serrano 已提交
217
	this.detachCamera = detachCamera;
218 219 220 221
	this.updateCameraProjectionMatricesUniform = updateCameraProjectionMatricesUniform;
	this.updateCameraViewMatricesUniform = updateCameraViewMatricesUniform;
	this.updateObjectMatricesUniforms = updateObjectMatricesUniforms;

222 223 224
}

export { WebGLMultiview };