UNPKG

threepipe

Version:

A 3D viewer framework built on top of three.js in TypeScript with a focus on quality rendering, modularity and extensibility.

230 lines (204 loc) 8.16 kB
import {GLTFExporter, GLTFExporterPlugin} from 'three/examples/jsm/exporters/GLTFExporter.js' import {IExportParser} from '../IExporter' import {GLTFWriter2} from './GLTFWriter2' import {AnimationClip, Object3D} from 'three' import {ThreeViewer} from '../../viewer' import { glbEncryptionProcessor, GLTFLightExtrasExtension, GLTFMaterialExtrasExtension, GLTFMaterialsAlphaMapExtension, GLTFMaterialsBumpMapExtension, GLTFMaterialsDisplacementMapExtension, GLTFMaterialsLightMapExtension, GLTFObject3DExtrasExtension, GLTFViewerConfigExtension, } from '../gltf' import {GLTFMeshGpuInstancingExporter} from '../../three/utils/gpu-instancing' export interface GLTFExporter2Options { /** * embed images in glb even when remote url is available * @default false */ embedUrlImages?: boolean, /** * Embed previews of images in glb * @default false */ embedUrlImagePreviews?: boolean, /** * export viewer config (scene settings) */ viewerConfig?: boolean, /** * Extension to export to, default for object/scene = glb */ exportExt?: string, preserveUUIDs?: boolean, /** * see GLTFDracoExporter and {@link GLTFMaterialExtrasExtension} */ externalImagesInExtras?: boolean, /** * see GLTFViewerExport->processViewer * @default false */ encodeUint16Rgbe?: boolean /** * Number of spaces to use when exporting to json * @default 2 */ jsonSpaces?: number, /** * Encrypt the exported file in a GLB container using {@link encryptKey} * @default false. * Works only for glb export. */ encrypt?: boolean, /** * Encryption key, if not provided, will be prompted * @default undefined. * Works only for glb export. */ encryptKey?: string|Uint8Array, // From GLTFExporter /** * Export position, rotation and scale instead of matrix per node. * Default is false */ trs?: boolean; /** * Export only visible objects. * Default is false. */ onlyVisible?: boolean; /** * Export just the attributes within the drawRange, if defined, instead of exporting the whole array. * Default is true. */ truncateDrawRange?: boolean; /** * Restricts the image maximum size (both width and height) to the given value. This option works only if embedImages is true. * Default is Infinity. */ maxTextureSize?: number; /** * List of animations to be included in the export. */ animations?: AnimationClip[]; /** * Generate indices for non-index geometry and export with them. * Default is false. */ forceIndices?: boolean; /** * Export custom glTF extensions defined on an object's userData.gltfExtensions property. * Default is true. */ includeCustomExtensions?: boolean; [key: string]: any } export class GLTFExporter2 extends GLTFExporter implements IExportParser { constructor() { super() this.processors.push(glbEncryptionProcessor) } register(callback: (writer: GLTFWriter2)=>GLTFExporterPlugin): this { return super.register(callback as any) } processors: ((obj: ArrayBuffer|any|Blob, options: GLTFExporter2Options) => Promise<ArrayBuffer|any|Blob>)[] = [] async parseAsync(obj: ArrayBuffer|any, options: GLTFExporter2Options): Promise<Blob> { if (!obj) throw new Error('No object to export') let gltf = !obj.__isGLTFOutput && (Array.isArray(obj) || obj.isObject3D) ? await new Promise((resolve, reject) => this.parse(obj, resolve, reject, options)) : obj for (const processor of this.processors) { gltf = await processor(gltf, options) } if (gltf && gltf instanceof Blob) return gltf if (gltf && typeof gltf === 'object' && !gltf.byteLength) { // byteLength is for ArrayBuffer return new Blob([JSON.stringify(gltf, (k, v)=> k.startsWith('__') ? undefined : v, options.jsonSpaces ?? 2)], {type: 'model/gltf+json'}) } else if (gltf) { return new Blob([gltf as ArrayBuffer], {type: 'model/gltf+binary'}) } else { throw new Error('GLTFExporter2.parse() failed') } } parse( input: Object3D | Object3D[], onDone: (gltf: ArrayBuffer | {[key: string]: any}) => void, onError: (error: ErrorEvent) => void, options: GLTFExporter2Options = {}, ): void { const gltfOptions: GLTFWriter2['options'] = { // default options binary: false, trs: options.trs ?? false, onlyVisible: options.onlyVisible ?? false, truncateDrawRange: options.truncateDrawRange ?? true, externalImagesInExtras: !options.embedUrlImages && options.externalImagesInExtras || false, // this is handled in gltfMaterialExtrasWriter, also see GLTFDracoExporter maxTextureSize: options.maxTextureSize ?? Infinity, animations: options.animations ?? [], includeCustomExtensions: options.includeCustomExtensions ?? true, forceIndices: options.forceIndices ?? false, exporterOptions: options, } if (options.exportExt === 'glb') { gltfOptions.binary = true } if (options.preserveUUIDs !== false) { // default true (Array.isArray(input) ? input : [input]).forEach((obj: Object3D) => obj.traverse((obj1: Object3D) => { if (obj1.uuid) obj1.userData.gltfUUID = obj1.uuid })) } // animations (Array.isArray(input) ? input : [input]).forEach((obj: Object3D) => obj.traverse((obj1: Object3D) => { if (obj1.animations) { for (const animation of obj1.animations) { if ((animation as any).__gltfExport !== false && !gltfOptions.animations!.includes(animation)) { gltfOptions.animations!.push(...obj1.animations) } } } })) return super.parse(input, (o: any)=> { if (options.preserveUUIDs !== false) { // default true (Array.isArray(input) ? input : [input]).forEach((obj: Object3D) => obj.traverse((obj1: Object3D) => { delete obj1.userData.gltfUUID })) } // eslint-disable-next-line @typescript-eslint/naming-convention onDone(Object.assign(o, {__isGLTFOutput: true})) }, onError, gltfOptions, new GLTFWriter2()) } static ExportExtensions: ((parser: GLTFWriter2) => GLTFExporterPlugin)[] = [ GLTFMaterialExtrasExtension.Export, GLTFObject3DExtrasExtension.Export, GLTFLightExtrasExtension.Export, GLTFMaterialsBumpMapExtension.Export, GLTFMaterialsDisplacementMapExtension.Export, GLTFMaterialsLightMapExtension.Export, GLTFMaterialsAlphaMapExtension.Export, (w)=>new GLTFMeshGpuInstancingExporter(w), ] setup(viewer: ThreeViewer, extraExtensions?: ((parser: GLTFWriter2) => GLTFExporterPlugin)[]): this { for (const ext of GLTFExporter2.ExportExtensions) this.register(ext) if (extraExtensions) for (const ext of extraExtensions) this.register(ext) // should be last this.register(this.gltfViewerWriter(viewer)) return this } gltfViewerWriter(viewer: ThreeViewer): (parser: GLTFWriter2) => GLTFExporterPlugin { return (writer: GLTFWriter2) => ({ afterParse: (input: any)=>{ input = Array.isArray(input) ? input[0] : input if (!input?.userData?.rootSceneModelRoot || writer.options?.exporterOptions?.viewerConfig === false || input?.userData?.__exportViewerConfig === false ) return GLTFViewerConfigExtension.ExportViewerConfig(viewer, writer) }, }) } }