diff --git a/editor/js/Sidebar.Project.Image.js b/editor/js/Sidebar.Project.Image.js index f8ada3822486e2..8f6023368fb225 100644 --- a/editor/js/Sidebar.Project.Image.js +++ b/editor/js/Sidebar.Project.Image.js @@ -2,7 +2,7 @@ import * as THREE from 'three'; import { UIBreak, UIButton, UIInteger, UIPanel, UIRow, UISelect, UIText } from './libs/ui.js'; -// import { ViewportPathtracer } from './Viewport.Pathtracer.js'; +import { ViewportPathtracer } from './Viewport.Pathtracer.js'; function SidebarProjectImage( editor ) { @@ -19,17 +19,34 @@ function SidebarProjectImage( editor ) { // Shading const shadingRow = new UIRow(); - // container.add( shadingRow ); + container.add( shadingRow ); shadingRow.add( new UIText( strings.getKey( 'sidebar/project/shading' ) ).setClass( 'Label' ) ); const shadingTypeSelect = new UISelect().setOptions( { - 0: 'Solid', - 1: 'Realistic' - } ).setWidth( '125px' ); - shadingTypeSelect.setValue( 0 ); + 0: 'SOLID / WebGLRenderer', + 1: 'REALISTIC / WebGLPathTracer' + } ).setWidth( '170px' ).setTextTransform( 'unset' ).onChange( refreshShadingRow ).setValue( 0 ); shadingRow.add( shadingTypeSelect ); + const pathTracerMinSamples = 3; + const pathTracerMaxSamples = 65536; + const samplesNumber = new UIInteger( 16 ).setRange( pathTracerMinSamples, pathTracerMaxSamples ); + + const samplesRow = new UIRow(); + samplesRow.add( new UIText( 'Sample Count' ).setClass( 'Label' ) ); // TODO: l10n + samplesRow.add( samplesNumber ); + + container.add( samplesRow ); + + function refreshShadingRow() { + + samplesRow.setHidden( shadingTypeSelect.getValue() !== '1' ); + + } + + refreshShadingRow(); + // Resolution const resolutionRow = new UIRow(); @@ -108,7 +125,7 @@ function SidebarProjectImage( editor ) { renderer.dispose(); break; - /* + case 1: // REALISTIC const status = document.createElement( 'div' ); @@ -120,26 +137,41 @@ function SidebarProjectImage( editor ) { status.style.fontSize = '12px'; output.document.body.appendChild( status ); - const pathtracer = new ViewportPathtracer( renderer ); - pathtracer.init( scene, camera ); - pathtracer.setSize( imageWidth.getValue(), imageHeight.getValue()); + const pathTracer = new ViewportPathtracer( renderer ); + pathTracer.init( scene, camera ); + pathTracer.setSize( imageWidth.getValue(), imageHeight.getValue() ); + + const maxSamples = Math.max( pathTracerMinSamples, Math.min( pathTracerMaxSamples, samplesNumber.getValue() ) ); function animate() { if ( output.closed === true ) return; - requestAnimationFrame( animate ); + const samples = Math.floor( pathTracer.getSamples() ) + 1; + + if ( samples < maxSamples ) { + + requestAnimationFrame( animate ); + + } + + pathTracer.update(); + + const progress = Math.floor( samples / maxSamples * 100 ); + + status.textContent = `${ samples } / ${ maxSamples } ( ${ progress }% )`; + + if ( progress === 100 ) { - pathtracer.update(); + status.textContent += ' ✓'; - // status.textContent = Math.floor( samples ); + } } animate(); break; - */ } diff --git a/editor/js/Viewport.Pathtracer.js b/editor/js/Viewport.Pathtracer.js index 63a983967a1978..7e78fd38c6525e 100644 --- a/editor/js/Viewport.Pathtracer.js +++ b/editor/js/Viewport.Pathtracer.js @@ -67,6 +67,14 @@ function ViewportPathtracer( renderer ) { } + function getSamples() { + + if ( pathTracer === null ) return; + + return pathTracer.samples; + + } + return { init: init, setSize: setSize, @@ -74,7 +82,8 @@ function ViewportPathtracer( renderer ) { setEnvironment: setEnvironment, updateMaterials: updateMaterials, update: update, - reset: reset + reset: reset, + getSamples: getSamples }; }