Viewer.js 21 KB

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