Skip to content

Commit 0c77d33

Browse files
committedNov 25, 2023
feat: add gltfExporter composable
1 parent 712f940 commit 0c77d33

File tree

4 files changed

+128
-1
lines changed

4 files changed

+128
-1
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
<script setup lang="ts">
2+
import { TresCanvas, useRenderLoop } from '@tresjs/core'
3+
import { CameraControls, useTweakPane, useGLTFExporter } from '@tresjs/cientos'
4+
import { MeshStandardMaterial, TorusGeometry } from 'three'
5+
import { shallowRef } from 'vue'
6+
7+
const donutMaterial = new MeshStandardMaterial({})
8+
const donutGeometry = new TorusGeometry(1, 0.5, 16, 32)
9+
const boxRef = shallowRef()
10+
11+
const { pane } = useTweakPane()
12+
13+
const btn1 = pane.addButton({
14+
title: 'Scene',
15+
label: 'Download',
16+
})
17+
18+
btn1.on('click', () => useGLTFExporter(boxRef.value.parent))
19+
20+
const btn2 = pane.addButton({
21+
title: 'Cube',
22+
label: 'Download', // optional
23+
})
24+
25+
btn2.on('click', () => useGLTFExporter(boxRef.value, { fileName: 'cube', binary: true }))
26+
27+
const { onLoop } = useRenderLoop()
28+
onLoop(({ elapsed }) => {
29+
if (boxRef.value) {
30+
boxRef.value.rotation.x = elapsed
31+
boxRef.value.rotation.y = elapsed
32+
}
33+
})
34+
</script>
35+
36+
<template>
37+
<TresCanvas
38+
clear-color="#82DBC5"
39+
window-size
40+
>
41+
<TresPerspectiveCamera :position="[0, 2, 5]" />
42+
<TresMesh
43+
v-for="i in 20"
44+
:key="i"
45+
:geometry="donutGeometry"
46+
:material="donutMaterial"
47+
:position="[0.5 + (i * (Math.random() - 0.5) * 5),
48+
0.5 + (i * (Math.random() - 0.5) * 5),
49+
0.5 + (i * (Math.random() - 0.5) * 5)]"
50+
:scale="(i * 0.5) + 1"
51+
/>
52+
<TresMesh
53+
ref="boxRef"
54+
:position-z="30"
55+
:scale="10"
56+
>
57+
<TresBoxGeometry :args="[1, 1, 1]" />
58+
<TresMeshStandardMaterial :color="0x00ff00" />
59+
</TresMesh>
60+
<CameraControls :distance="75" />
61+
<TresDirectionalLight :position="[0, 10, 10]" />
62+
</TresCanvas>
63+
</template>

‎playground/src/router/routes/misc.ts

+5
Original file line numberDiff line numberDiff line change
@@ -19,4 +19,9 @@ export const miscRoutes = [
1919
name: 'Stats',
2020
component: () => import('../../pages/misc/StatsDemo.vue'),
2121
},
22+
{
23+
path: '/misc/gltfExporter',
24+
name: 'GLTFExporter',
25+
component: () => import('../../pages/misc/GLTFExporterDemo.vue'),
26+
},
2227
]

‎src/core/misc/index.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,5 +3,6 @@ import { useTweakPane } from './useTweakPane'
33
import { Stats } from './Stats'
44
import { StatsGl } from './StatsGl'
55
import Html from './html/HTML.vue'
6+
import { useGLTFExporter } from './useGLTFExporter'
67

7-
export { useTweakPane, Html, Stats, StatsGl }
8+
export { useTweakPane, Html, Stats, StatsGl, useGLTFExporter }

‎src/core/misc/useGLTFExporter.ts

+58
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
import { GLTFExporter } from 'three/addons/exporters/GLTFExporter.js'
2+
import { ref } from 'vue'
3+
import type { AnimationClip, Object3D } from 'three'
4+
import { useLogger } from '@tresjs/core'
5+
6+
const { logError } = useLogger()
7+
8+
interface gltfExporterOptions {
9+
fileName?: string
10+
trs?: boolean
11+
onlyVisible?: boolean
12+
binary?: boolean
13+
maxTextureSize?: number
14+
animations?: AnimationClip[]
15+
includeCustomExtensions?: boolean
16+
}
17+
18+
export async function useGLTFExporter(object3D: Object3D, options?: gltfExporterOptions ) {
19+
const exporter = new GLTFExporter()
20+
const objectToDownload = ref(object3D)
21+
const name = options?.fileName || object3D.name || 'scene'
22+
23+
exporter.parse(
24+
objectToDownload.value,
25+
( gltf: Object3D ) => {
26+
if ( gltf instanceof ArrayBuffer ) {
27+
saveArrayBuffer( gltf, `${name}.glb` )
28+
}
29+
else {
30+
const output = JSON.stringify( gltf, null, 2 )
31+
saveString( output, `${name}.gltf` )
32+
}
33+
},
34+
( error: any ) => {
35+
logError( 'An error happened', error )
36+
},
37+
options,
38+
)
39+
return true
40+
}
41+
42+
function saveString( text: string, filename: string ) {
43+
save( new Blob( [ text ], { type: 'text/plain' } ), filename )
44+
}
45+
46+
function saveArrayBuffer( buffer: any, filename: string ) {
47+
save( new Blob( [ buffer ], { type: 'application/octet-stream' } ), filename )
48+
}
49+
50+
function save( blob: Blob, filename: string ) {
51+
const link = document.createElement( 'a' )
52+
link.style.display = 'none'
53+
document.body.appendChild( link )
54+
link.href = URL.createObjectURL( blob )
55+
link.download = filename
56+
link.click()
57+
link.remove()
58+
}

0 commit comments

Comments
 (0)
Please sign in to comment.