Viewer.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809
  1. import { Tab } from '../ui/Tab.js';
  2. import { List } from '../ui/List.js';
  3. import { Item } from '../ui/Item.js';
  4. import { splitPath, splitCamelCase } from '../ui/utils.js';
  5. import { RendererUtils, NoToneMapping, LinearSRGBColorSpace, QuadMesh, NodeMaterial, CanvasTarget, Vector2 } from 'three/webgpu';
  6. import { renderOutput, vec2, vec3, vec4, Fn, screenUV, step, OnMaterialUpdate, uniform, float } from 'three/tsl';
  7. const _size = /*@__PURE__*/ new Vector2();
  8. const aspectRatioUV = /*@__PURE__*/ Fn( ( [ uv, textureNode, canvasAspect ] ) => {
  9. const textureAspect = uniform( 0 );
  10. OnMaterialUpdate( () => {
  11. const { width, height } = textureNode.value;
  12. textureAspect.value = width / height;
  13. } );
  14. const ratio = canvasAspect.div( textureAspect );
  15. const centered = uv.sub( 0.5 );
  16. // If canvasAspect > textureAspect:
  17. const uvWide = vec2( centered.x.mul( ratio ), centered.y ).add( 0.5 );
  18. // If canvasAspect <= textureAspect:
  19. const uvTall = vec2( centered.x, centered.y.div( ratio ) ).add( 0.5 );
  20. const finalUV = canvasAspect.greaterThan( textureAspect ).select( uvWide, uvTall );
  21. const inBounds = step( 0.0, finalUV.x ).mul( step( finalUV.x, 1.0 ) ).mul( step( 0.0, finalUV.y ) ).mul( step( finalUV.y, 1.0 ) );
  22. return vec3( finalUV, inBounds );
  23. } );
  24. class Viewer extends Tab {
  25. constructor( options = {} ) {
  26. super( 'Viewer', options );
  27. this.content.style.overflow = 'hidden';
  28. this.maximizedByFullscreenButton = false;
  29. // Toolbar
  30. const toolbar = document.createElement( 'div' );
  31. toolbar.className = 'toolbar';
  32. const backBtn = document.createElement( 'button' );
  33. backBtn.className = 'viewer-back-btn';
  34. backBtn.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"><line x1="19" y1="12" x2="5" y2="12"></line><polyline points="12 19 5 12 12 5"></polyline></svg>';
  35. backBtn.title = 'Back to list';
  36. backBtn.style.display = 'none';
  37. toolbar.appendChild( backBtn );
  38. const label = document.createElement( 'span' );
  39. label.textContent = 'View:';
  40. toolbar.appendChild( label );
  41. const select = document.createElement( 'select' );
  42. select.className = 'select';
  43. select.style.width = '200px';
  44. const defaultOption = document.createElement( 'option' );
  45. defaultOption.value = 'list';
  46. defaultOption.textContent = 'List';
  47. select.appendChild( defaultOption );
  48. toolbar.appendChild( select );
  49. this.content.appendChild( toolbar );
  50. const nodeList = new List( 'Viewer', 'Name' );
  51. nodeList.setGridStyle( '150px minmax(200px, 2fr)' );
  52. nodeList.domElement.style.minWidth = '400px';
  53. const scrollWrapper = document.createElement( 'div' );
  54. scrollWrapper.className = 'list-scroll-wrapper';
  55. scrollWrapper.style.flexGrow = '1';
  56. scrollWrapper.style.overflowY = 'auto';
  57. scrollWrapper.style.minHeight = '0';
  58. scrollWrapper.appendChild( nodeList.domElement );
  59. this.content.appendChild( scrollWrapper );
  60. // Container for full screen view
  61. const fullViewerContainer = document.createElement( 'div' );
  62. fullViewerContainer.className = 'full-viewer-container';
  63. fullViewerContainer.style.touchAction = 'none';
  64. this.content.appendChild( fullViewerContainer );
  65. const nodes = new Item( 'User Defined' );
  66. nodeList.add( nodes );
  67. //
  68. this.itemLibrary = new Map();
  69. this.folderLibrary = new Map();
  70. this.canvasNodes = new Map();
  71. this.currentDataList = [];
  72. this.nodeList = nodeList;
  73. this.nodes = nodes;
  74. this.scrollWrapper = scrollWrapper;
  75. this.fullViewerContainer = fullViewerContainer;
  76. this.select = select;
  77. this.backBtn = backBtn;
  78. this.activeFullNodeId = null;
  79. backBtn.addEventListener( 'click', () => {
  80. select.value = 'list';
  81. this.showListView();
  82. if ( this.maximizedByFullscreenButton ) {
  83. if ( this.profiler && this.profiler.panel.classList.contains( 'maximized' ) ) {
  84. this.profiler.toggleMaximize();
  85. }
  86. this.maximizedByFullscreenButton = false;
  87. }
  88. } );
  89. select.addEventListener( 'change', () => {
  90. const val = select.value;
  91. if ( val === 'list' ) {
  92. this.showListView();
  93. } else {
  94. this.showNodeView( val );
  95. }
  96. } );
  97. // Event forwarding setup for OrbitControls
  98. this.isDraggingThumbnail = false;
  99. this.activeSourceCanvas = null;
  100. this.activePointerIds = new Set();
  101. const handleGlobalPointer = ( e ) => {
  102. if ( ! this.isDraggingThumbnail || ! this.activeSourceCanvas ) return;
  103. const renderer = this.inspector.getRenderer();
  104. if ( ! renderer || ! renderer.domElement ) return;
  105. if ( e.isForwarded ) return;
  106. // Block native event from reaching other document-level listeners (OrbitControls)
  107. e.stopImmediatePropagation();
  108. e.preventDefault();
  109. // Project and dispatch forwarded event
  110. this.forwardEvent( e, this.activeSourceCanvas, renderer.domElement );
  111. if ( e.type === 'pointerup' || e.type === 'pointercancel' ) {
  112. this.activePointerIds.delete( e.pointerId );
  113. if ( this.activePointerIds.size === 0 ) {
  114. this.isDraggingThumbnail = false;
  115. this.activeSourceCanvas = null;
  116. }
  117. }
  118. };
  119. window.addEventListener( 'pointermove', handleGlobalPointer, true );
  120. window.addEventListener( 'pointerup', handleGlobalPointer, true );
  121. window.addEventListener( 'pointercancel', handleGlobalPointer, true );
  122. }
  123. getFolder( name ) {
  124. let folder = this.folderLibrary.get( name );
  125. if ( folder === undefined ) {
  126. folder = new Item( name );
  127. this.folderLibrary.set( name, folder );
  128. this.nodeList.add( folder );
  129. }
  130. return folder;
  131. }
  132. hide() {
  133. super.hide();
  134. this.maximizedByFullscreenButton = false;
  135. this.isDraggingThumbnail = false;
  136. this.activeSourceCanvas = null;
  137. this.activePointerIds.clear();
  138. }
  139. addNodeItem( canvasData ) {
  140. let item = this.itemLibrary.get( canvasData.id );
  141. if ( item === undefined ) {
  142. const name = canvasData.name;
  143. const domElement = canvasData.canvasTarget.domElement;
  144. // Create wrapper
  145. const wrapper = document.createElement( 'div' );
  146. wrapper.className = 'node-canvas-wrapper';
  147. wrapper.style.position = 'relative';
  148. wrapper.style.display = 'inline-block';
  149. wrapper.style.width = '140px';
  150. wrapper.style.height = '140px';
  151. wrapper.style.touchAction = 'none';
  152. // View full screen button
  153. const viewBtn = document.createElement( 'button' );
  154. viewBtn.className = 'node-canvas-detach-btn';
  155. viewBtn.title = 'View full size';
  156. viewBtn.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="7" y1="17" x2="17" y2="7"></line><polyline points="7 7 17 7 17 17"></polyline></svg>';
  157. viewBtn.onclick = ( e ) => {
  158. e.stopPropagation();
  159. this.select.value = canvasData.id;
  160. this.showNodeView( canvasData.id );
  161. };
  162. // Fullscreen and maximize button
  163. const fullscreenBtn = document.createElement( 'button' );
  164. fullscreenBtn.className = 'node-canvas-fullscreen-btn';
  165. fullscreenBtn.title = 'Fullscreen view';
  166. fullscreenBtn.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M8 3H5a2 2 0 0 0-2 2v3m18 0V5a2 2 0 0 0-2-2h-3m0 18h3a2 2 0 0 0 2-2v-3M3 16v3a2 2 0 0 0 2 2h3"/></svg>';
  167. fullscreenBtn.onclick = ( e ) => {
  168. e.stopPropagation();
  169. this.select.value = canvasData.id;
  170. this.showNodeView( canvasData.id );
  171. if ( this.profiler && ! this.profiler.panel.classList.contains( 'maximized' ) ) {
  172. this.profiler.toggleMaximize();
  173. this.maximizedByFullscreenButton = true;
  174. if ( ! this._maximizeListenerAdded && this.profiler.maximizeBtn ) {
  175. this.profiler.maximizeBtn.addEventListener( 'click', () => {
  176. this.maximizedByFullscreenButton = false;
  177. } );
  178. this._maximizeListenerAdded = true;
  179. }
  180. }
  181. };
  182. wrapper.appendChild( domElement );
  183. wrapper.appendChild( viewBtn );
  184. wrapper.appendChild( fullscreenBtn );
  185. this.setupEventForwarding( domElement );
  186. // Store elements in canvasData for access
  187. canvasData.domElement = domElement;
  188. canvasData.wrapperElement = wrapper;
  189. item = new Item( wrapper, name );
  190. item.itemRow.children[ 1 ].style[ 'justify-content' ] = 'flex-start';
  191. this.itemLibrary.set( canvasData.id, item );
  192. }
  193. return item;
  194. }
  195. setupEventForwarding( sourceCanvas ) {
  196. sourceCanvas.style.touchAction = 'none';
  197. const onPointerDown = ( e ) => {
  198. const renderer = this.inspector.getRenderer();
  199. if ( ! renderer || ! renderer.domElement ) return;
  200. const targetCanvas = renderer.domElement;
  201. this.isDraggingThumbnail = true;
  202. this.activeSourceCanvas = sourceCanvas;
  203. this.activePointerIds.add( e.pointerId );
  204. // Project and dispatch pointerdown
  205. this.forwardEvent( e, sourceCanvas, targetCanvas );
  206. };
  207. sourceCanvas.addEventListener( 'pointerdown', onPointerDown );
  208. // Wheel event support for zooming
  209. const onWheel = ( e ) => {
  210. const renderer = this.inspector.getRenderer();
  211. if ( ! renderer || ! renderer.domElement ) return;
  212. e.preventDefault();
  213. e.stopPropagation();
  214. this.forwardEvent( e, sourceCanvas, renderer.domElement );
  215. };
  216. sourceCanvas.addEventListener( 'wheel', onWheel, { passive: false } );
  217. // Click, dblclick, contextmenu
  218. const onMouseShortcut = ( e ) => {
  219. const renderer = this.inspector.getRenderer();
  220. if ( ! renderer || ! renderer.domElement ) return;
  221. e.stopPropagation();
  222. if ( e.type === 'contextmenu' ) {
  223. e.preventDefault();
  224. }
  225. this.forwardEvent( e, sourceCanvas, renderer.domElement );
  226. };
  227. sourceCanvas.addEventListener( 'click', onMouseShortcut );
  228. sourceCanvas.addEventListener( 'dblclick', onMouseShortcut );
  229. sourceCanvas.addEventListener( 'contextmenu', onMouseShortcut );
  230. }
  231. forwardEvent( event, sourceCanvas, targetCanvas ) {
  232. const sourceRect = sourceCanvas.getBoundingClientRect();
  233. const targetRect = targetCanvas.getBoundingClientRect();
  234. const localX = ( event.clientX - sourceRect.left ) / sourceRect.width;
  235. const localY = ( event.clientY - sourceRect.top ) / sourceRect.height;
  236. const targetClientX = targetRect.left + localX * targetRect.width;
  237. const targetClientY = targetRect.top + localY * targetRect.height;
  238. const targetPageX = targetClientX + window.scrollX;
  239. const targetPageY = targetClientY + window.scrollY;
  240. let newEvent;
  241. const eventInit = {
  242. bubbles: true,
  243. cancelable: true,
  244. view: window,
  245. clientX: targetClientX,
  246. clientY: targetClientY,
  247. screenX: targetClientX + window.screenX,
  248. screenY: targetClientY + window.screenY,
  249. pageX: targetPageX,
  250. pageY: targetPageY,
  251. ctrlKey: event.ctrlKey,
  252. shiftKey: event.shiftKey,
  253. altKey: event.altKey,
  254. metaKey: event.metaKey,
  255. buttons: event.buttons,
  256. button: event.button
  257. };
  258. if ( event instanceof WheelEvent ) {
  259. newEvent = new WheelEvent( event.type, {
  260. ...eventInit,
  261. deltaX: event.deltaX,
  262. deltaY: event.deltaY,
  263. deltaZ: event.deltaZ,
  264. deltaMode: event.deltaMode
  265. } );
  266. } else if ( window.PointerEvent && event instanceof PointerEvent ) {
  267. newEvent = new PointerEvent( event.type, {
  268. ...eventInit,
  269. pointerId: event.pointerId,
  270. width: event.width,
  271. height: event.height,
  272. pressure: event.pressure,
  273. tiltX: event.tiltX,
  274. tiltY: event.tiltY,
  275. pointerType: event.pointerType,
  276. isPrimary: event.isPrimary
  277. } );
  278. } else {
  279. newEvent = new MouseEvent( event.type, eventInit );
  280. }
  281. newEvent.isForwarded = true;
  282. targetCanvas.dispatchEvent( newEvent );
  283. }
  284. showListView() {
  285. if ( this.activeFullNodeId ) {
  286. const canvasData = Array.from( this.canvasNodes.values() ).find( data => String( data.id ) === String( this.activeFullNodeId ) );
  287. if ( canvasData ) {
  288. // Move canvas back to wrapper
  289. canvasData.wrapperElement.appendChild( canvasData.domElement );
  290. // Reset size
  291. canvasData.domElement.style.width = '';
  292. canvasData.domElement.style.height = '';
  293. canvasData.canvasTarget.setSize( 140, 140 );
  294. const renderer = this.inspector.getRenderer();
  295. renderer.backend.delete( canvasData.canvasTarget );
  296. }
  297. this.activeFullNodeId = null;
  298. }
  299. this.scrollWrapper.style.display = '';
  300. this.fullViewerContainer.style.display = 'none';
  301. this.backBtn.style.display = 'none';
  302. }
  303. showNodeView( nodeId ) {
  304. // First restore previous full screen node if any
  305. if ( this.activeFullNodeId && String( this.activeFullNodeId ) !== String( nodeId ) ) {
  306. this.showListView();
  307. }
  308. const canvasData = Array.from( this.canvasNodes.values() ).find( data => String( data.id ) === String( nodeId ) );
  309. if ( canvasData ) {
  310. this.activeFullNodeId = nodeId;
  311. this.backBtn.style.display = 'flex';
  312. // Hide list, show full screen container
  313. this.scrollWrapper.style.display = 'none';
  314. this.fullViewerContainer.style.display = 'flex';
  315. // Move canvas to the full viewer container
  316. this.fullViewerContainer.appendChild( canvasData.domElement );
  317. canvasData.domElement.style.width = '100%';
  318. canvasData.domElement.style.height = '100%';
  319. // Resize canvas to fit full viewer container
  320. const rect = this.fullViewerContainer.getBoundingClientRect();
  321. const contentWidth = rect.width || this.content.clientWidth;
  322. const contentHeight = rect.height || ( this.content.clientHeight - 38 ); // minus toolbar
  323. canvasData.canvasTarget.setSize( contentWidth, contentHeight );
  324. const renderer = this.inspector.getRenderer();
  325. renderer.backend.delete( canvasData.canvasTarget );
  326. }
  327. }
  328. getCanvasDataByNode( renderer, node ) {
  329. let canvasData = this.canvasNodes.get( node );
  330. if ( canvasData === undefined ) {
  331. const canvas = document.createElement( 'canvas' );
  332. const canvasTarget = new CanvasTarget( canvas );
  333. canvasTarget.setPixelRatio( window.devicePixelRatio );
  334. canvasTarget.setSize( 140, 140 );
  335. const id = node.id;
  336. const { path, name } = splitPath( splitCamelCase( node.getName() || '(unnamed)' ) );
  337. const canvasAspect = uniform( 1 );
  338. const mask = float( 1 );
  339. const target = node.context( { getUV: ( textureNode ) => {
  340. const uvData = aspectRatioUV( screenUV, textureNode, canvasAspect );
  341. const correctedUV = uvData.xy;
  342. mask.assign( uvData.z );
  343. return correctedUV;
  344. } } );
  345. let output = vec4( vec3( target ), 1 ).mul( mask );
  346. output = renderOutput( output, NoToneMapping, renderer.outputColorSpace );
  347. output = output.context( { inspector: true } );
  348. const material = new NodeMaterial();
  349. material.outputNode = output;
  350. const quad = new QuadMesh( material );
  351. quad.name = 'Viewer - ' + name;
  352. canvasData = {
  353. id,
  354. name,
  355. path,
  356. node,
  357. quad,
  358. canvasTarget,
  359. material,
  360. canvasAspect
  361. };
  362. this.canvasNodes.set( node, canvasData );
  363. }
  364. return canvasData;
  365. }
  366. update( inspector ) {
  367. const renderer = inspector.getRenderer();
  368. const nodes = inspector.getNodes();
  369. if ( nodes.length > 0 ) {
  370. if ( ! renderer.backend.isWebGPUBackend ) {
  371. inspector.resolveConsoleOnce( 'warn', 'Inspector: Viewer is only available with WebGPU.' );
  372. return;
  373. }
  374. if ( ! this.isVisible ) {
  375. this.show();
  376. }
  377. }
  378. if ( ! this.isActive ) return;
  379. const canvasDataList = nodes.map( node => this.getCanvasDataByNode( renderer, node ) );
  380. // Check if the list of nodes has changed
  381. let nodesChanged = canvasDataList.length !== this.currentDataList.length;
  382. if ( ! nodesChanged ) {
  383. for ( let i = 0; i < canvasDataList.length; i ++ ) {
  384. if ( canvasDataList[ i ].id !== this.currentDataList[ i ].id ) {
  385. nodesChanged = true;
  386. break;
  387. }
  388. }
  389. }
  390. if ( nodesChanged ) {
  391. const currentSelectedValue = this.select.value;
  392. // Clear options except the first one ('list')
  393. while ( this.select.options.length > 1 ) {
  394. this.select.remove( 1 );
  395. }
  396. // Add options for each node in canvasDataList
  397. for ( const canvasData of canvasDataList ) {
  398. const option = document.createElement( 'option' );
  399. option.value = canvasData.id;
  400. option.textContent = canvasData.path ? `${ canvasData.path } / ${ canvasData.name }` : canvasData.name;
  401. this.select.appendChild( option );
  402. }
  403. // Restore selection if still valid
  404. let hasSelectedValue = false;
  405. for ( let i = 0; i < this.select.options.length; i ++ ) {
  406. if ( this.select.options[ i ].value === currentSelectedValue ) {
  407. this.select.selectedIndex = i;
  408. hasSelectedValue = true;
  409. break;
  410. }
  411. }
  412. if ( ! hasSelectedValue ) {
  413. this.select.value = 'list';
  414. this.showListView();
  415. }
  416. }
  417. // Real-time resize of active full-screen node canvas target
  418. if ( this.activeFullNodeId ) {
  419. const canvasData = canvasDataList.find( data => String( data.id ) === String( this.activeFullNodeId ) );
  420. if ( canvasData ) {
  421. const rect = this.fullViewerContainer.getBoundingClientRect();
  422. const contentWidth = rect.width || this.content.clientWidth;
  423. const contentHeight = rect.height || ( this.content.clientHeight - 38 );
  424. if ( canvasData.canvasTarget.domElement.width !== contentWidth || canvasData.canvasTarget.domElement.height !== contentHeight ) {
  425. canvasData.canvasTarget.setSize( contentWidth, contentHeight );
  426. renderer.backend.delete( canvasData.canvasTarget );
  427. }
  428. }
  429. }
  430. //
  431. const previousDataList = [ ...this.currentDataList ];
  432. // remove old
  433. for ( const canvasData of previousDataList ) {
  434. if ( this.itemLibrary.has( canvasData.id ) && canvasDataList.indexOf( canvasData ) === - 1 ) {
  435. const item = this.itemLibrary.get( canvasData.id );
  436. const parent = item.parent;
  437. parent.remove( item );
  438. if ( this.folderLibrary.has( parent.data[ 0 ] ) && parent.children.length === 0 ) {
  439. parent.parent.remove( parent );
  440. this.folderLibrary.delete( parent.data[ 0 ] );
  441. }
  442. this.itemLibrary.delete( canvasData.id );
  443. }
  444. }
  445. //
  446. const indexes = {};
  447. for ( const canvasData of canvasDataList ) {
  448. const item = this.addNodeItem( canvasData );
  449. const previousCanvasTarget = renderer.getCanvasTarget();
  450. const path = canvasData.path;
  451. if ( path ) {
  452. const folder = this.getFolder( path );
  453. if ( indexes[ path ] === undefined ) {
  454. indexes[ path ] = 0;
  455. }
  456. if ( folder.parent === null || item.parent !== folder || folder.children.indexOf( item ) !== indexes[ path ] ) {
  457. folder.add( item );
  458. }
  459. indexes[ path ] ++;
  460. } else {
  461. if ( ! item.parent ) {
  462. this.nodes.add( item );
  463. }
  464. }
  465. const rttNodes = [];
  466. const mainSize = previousCanvasTarget.getDrawingBufferSize( _size );
  467. canvasData.node.traverse( ( child ) => {
  468. if ( child.isRTTNode && child.autoResize === true ) {
  469. const oldWidth = child.width;
  470. const oldHeight = child.height;
  471. child.width = mainSize.width;
  472. child.height = mainSize.height;
  473. child.setSize( mainSize.width, mainSize.height );
  474. rttNodes.push( {
  475. node: child,
  476. oldWidth,
  477. oldHeight
  478. } );
  479. }
  480. } );
  481. const state = RendererUtils.resetRendererState( renderer );
  482. renderer.toneMapping = NoToneMapping;
  483. renderer.outputColorSpace = LinearSRGBColorSpace;
  484. renderer.setCanvasTarget( canvasData.canvasTarget );
  485. if ( canvasData.canvasAspect ) {
  486. canvasData.canvasAspect.value = canvasData.canvasTarget.domElement.width / canvasData.canvasTarget.domElement.height;
  487. }
  488. canvasData.quad.render( renderer );
  489. renderer.setCanvasTarget( previousCanvasTarget );
  490. RendererUtils.restoreRendererState( renderer, state );
  491. for ( const rtt of rttNodes ) {
  492. rtt.node.width = rtt.oldWidth;
  493. rtt.node.height = rtt.oldHeight;
  494. }
  495. }
  496. this.currentDataList = canvasDataList;
  497. }
  498. setActive( isActive ) {
  499. super.setActive( isActive );
  500. }
  501. }
  502. export { Viewer };
粤ICP备19079148号