'use strict'; exports.template = /* html */`
`; exports.style = /* css */` .preview-section { margin-top: 0px; } .preview { border-top: 1px solid var(--color-normal-border); } .preview > .info { padding: 4px 4px 0 4px; } .preview > .info > ui-label { margin-right: 6px; } .preview > .image { height: var(--inspector-footer-preview-height, 200px); overflow: hidden; display: flex; flex: 1; } .preview >.image > .canvas { flex: 1; } .select-box { float: right; } .preview-channel { display: none; } .preview-channel.show { display: inline-block; } `; exports.$ = { container: '.preview', vertices: '.vertices', triangles: '.triangles', image: '.image', canvas: '.canvas', minPosLabel: '.minPosLabel', maxPosLabel: '.maxPosLabel', previewType: '.preview-type', previewChannel: '.preview-channel', }; async function callMeshPreviewFunction(funcName, ...args) { return await Editor.Message.request('scene', 'call-preview-function', 'scene:mesh-preview', funcName, ...args); } const previewSelectType = { shaded: 'Shaded', uv: 'UV Layout', }; const Elements = { preview: { ready() { const panel = this; let _isPreviewDataDirty = false; Object.defineProperty(panel, 'isPreviewDataDirty', { get() { return _isPreviewDataDirty; }, set(value) { if (value !== _isPreviewDataDirty) { _isPreviewDataDirty = value; value && panel.refreshPreview(); } }, }); panel.$.previewType.innerHTML = Object.values(previewSelectType).map(v => ``).join(''); panel.$.previewType.value = previewSelectType.shaded; panel.$.previewType.addEventListener('confirm', (event) => { const value = event.target.value; if (value === previewSelectType.uv) { panel.$.previewChannel.classList.add('show'); panel.$.previewChannel.value = 0; } else { panel.$.previewChannel.classList.remove('show'); } panel.isPreviewDataDirty = true; }); panel.$.previewChannel.addEventListener('confirm', () => { panel.isPreviewDataDirty = true; }); panel.$.canvas.addEventListener('mousedown', async (event) => { // Non-model previews do not respond to events if (panel.$.previewType.value !== previewSelectType.shaded) { return; } await callMeshPreviewFunction('onMouseDown', { x: event.x, y: event.y, button: event.button }); async function mousemove(event) { await callMeshPreviewFunction('onMouseMove', { movementX: event.movementX, movementY: event.movementY, }); panel.isPreviewDataDirty = true; } async function mouseup(event) { await callMeshPreviewFunction('onMouseUp', { x: event.x, y: event.y, }); document.removeEventListener('mousemove', mousemove); document.removeEventListener('mouseup', mouseup); panel.isPreviewDataDirty = false; } document.addEventListener('mousemove', mousemove); document.addEventListener('mouseup', mouseup); panel.isPreviewDataDirty = true; }); panel.$.canvas.addEventListener('wheel', async (event) => { // Non-model previews do not respond to events if (panel.$.previewType.value !== previewSelectType.shaded) { return; } await callMeshPreviewFunction('onMouseWheel', { wheelDeltaY: event.wheelDeltaY, wheelDeltaX: event.wheelDeltaX, }); panel.isPreviewDataDirty = true; }); const GlPreview = Editor._Module.require('PreviewExtends').default; panel.glPreview = new GlPreview('scene:mesh-preview', 'query-mesh-preview-data'); function observer() { panel.isPreviewDataDirty = true; } panel.resizeObserver = new window.ResizeObserver(observer); panel.resizeObserver.observe(panel.$.image); observer(); }, async update() { const panel = this; if (!panel.$.canvas) { return; } await panel.glPreview.init({ width: panel.$.canvas.clientWidth, height: panel.$.canvas.clientHeight }); // reset panel.$.previewType.value = previewSelectType.shaded; panel.$.previewChannel.value = 0; panel.$.previewChannel.classList.remove('show'); const info = await callMeshPreviewFunction('setMesh', panel.asset.uuid); panel.previewUVs = await callMeshPreviewFunction('getModelUVs', panel.asset.uuid); if (panel.previewUVs.length === 0) { panel.$.previewType.innerHTML = ``; } else { panel.$.previewType.innerHTML = Object.values(previewSelectType).map(v => ``).join(''); panel.$.previewChannel.innerHTML = panel.previewUVs.map((_, i) => ``).join(''); } panel.infoUpdate(info); panel.isPreviewDataDirty = true; }, close() { const panel = this; panel.resizeObserver.unobserve(panel.$.image); panel.previewUVs = []; }, }, info: { ready() { const panel = this; panel.infoUpdate = Elements.info.update.bind(panel); }, update(info) { if (!info) { return; } const panel = this; panel.$.vertices.value = 'Vertices: ' + info.vertices; panel.$.triangles.value = 'Triangles: ' + info.polygons; panel.$.minPosLabel.value = ''; if (info.minPosition) { const pos = info.minPosition; panel.$.minPosLabel.value = `MinPos: (${pos.x.toFixed(3)}, ${pos.y.toFixed(3)}, ${pos.z.toFixed(3)})`; } panel.$.maxPosLabel.value = ''; if (info.maxPosition) { const pos = info.maxPosition; panel.$.maxPosLabel.value = `MaxPos: (${pos.x.toFixed(3)}, ${pos.y.toFixed(3)}, ${pos.z.toFixed(3)})`; } panel.isPreviewDataDirty = true; }, close() { callMeshPreviewFunction('hide'); }, }, }; exports.methods = { async refreshPreview() { const panel = this; // After await, the panel no longer exists if (!panel.$.canvas) { return; } const doDraw = async function() { try { const canvas = panel.$.canvas; const { clientWidth: width, clientHeight: height } = panel.$.image; if (canvas.width !== width || canvas.height !== height) { canvas.width = width; canvas.height = height; await panel.glPreview.initGL(canvas, { width, height }); await panel.glPreview.resizeGL(width, height); } let info; if (panel.$.previewType.value === previewSelectType.shaded) { info = await panel.glPreview.queryPreviewData({ width: canvas.width, height: canvas.height, }); } if (panel.$.previewType.value === previewSelectType.uv) { info = panel.glPreview.computedUV(panel.previewUVs[panel.$.previewChannel.value], canvas.width, canvas.height); } panel.glPreview.drawGL(info); } catch (e) { console.warn(e); } }; requestAnimationFrame(async () => { await doDraw(); panel.isPreviewDataDirty = false; }); }, }; exports.ready = function() { for (const prop in Elements) { const element = Elements[prop]; if (element.ready) { element.ready.call(this); } } }; exports.update = function(assetList, metaList) { this.assetList = assetList; this.metaList = metaList; this.asset = assetList[0]; this.meta = metaList[0]; // 如何多选就隐藏预览 if (assetList.length > 1) { this.$.container.style.display = 'none'; } else { this.$.container.style.display = 'block'; } for (const prop in Elements) { const element = Elements[prop]; if (element.update) { element.update.call(this); } } }; exports.close = function() { for (const prop in Elements) { const element = Elements[prop]; if (element.close) { element.close.call(this); } } };