Skip to content

Commit

Permalink
Fix previous TS conversion
Browse files Browse the repository at this point in the history
Fixes the facet normal / vertex text interpolation bug from the TS conversion and guards against positionAttribute-less meshes
  • Loading branch information
stockhuman committed Dec 13, 2023
1 parent f35f76e commit 2f37ad0
Show file tree
Hide file tree
Showing 3 changed files with 194 additions and 187 deletions.
21 changes: 0 additions & 21 deletions src/exporters/STLExporter.d.ts

This file was deleted.

166 changes: 0 additions & 166 deletions src/exporters/STLExporter.js

This file was deleted.

194 changes: 194 additions & 0 deletions src/exporters/STLExporter.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,194 @@
import {
BufferAttribute,
InterleavedBufferAttribute,
Mesh,
Object3D,
PlaneGeometry,
SkinnedMesh,
Vector3,
} from 'three'

interface STLExporterOptions {
binary?: boolean
}

/**
* The official threejs build breaks on empty BufferGeometries,
* while the three-stdlib has a broken non-binary export.
* This implementation is derived from the latter, with fixes.
*/
export class STLExporter {
private binary

private output: string | DataView
private offset

private objects: { object3d: Object3D; geometry: PlaneGeometry }[]
private triangles

private vA
private vB
private vC
private cb
private ab
private normal

constructor() {
this.binary = false

this.output = ''
this.offset = 80 // skip header

this.objects = []
this.triangles = 0

this.vA = new Vector3()
this.vB = new Vector3()
this.vC = new Vector3()
this.cb = new Vector3()
this.ab = new Vector3()
this.normal = new Vector3()
}

parse(scene: Object3D, options: { binary: true }): DataView
parse(scene: Object3D, options: { binary: false }): string

public parse(scene: Object3D, options?: STLExporterOptions): string | DataView {
this.binary = options?.binary !== undefined ? options?.binary : false

scene.traverse((object: Object3D) => {
if (object instanceof Mesh && object.isMesh) {
const geometry = object.geometry

if (!geometry.isBufferGeometry) {
throw new Error('THREE.STLExporter: Geometry is not of type THREE.BufferGeometry.')
}

const index = geometry.index
const positionAttribute = geometry.getAttribute('position') || null
if (!positionAttribute) return

this.triangles += index !== null ? index.count / 3 : positionAttribute.count / 3

this.objects.push({
object3d: object,
geometry: geometry,
})
}
})

if (this.binary) {
const bufferLength = this.triangles * 2 + this.triangles * 3 * 4 * 4 + 80 + 4
const arrayBuffer = new ArrayBuffer(bufferLength)
this.output = new DataView(arrayBuffer)
this.output.setUint32(this.offset, this.triangles, true)
this.offset += 4
} else {
this.output = ''
this.output += 'solid exported\n'
}

for (let i = 0, il = this.objects.length; i < il; i++) {
const object = this.objects[i].object3d
const geometry = this.objects[i].geometry

const index = geometry.index
const positionAttribute = geometry.getAttribute('position')

if (index !== null) {
// indexed geometry
for (let j = 0; j < index.count; j += 3) {
const a = index.getX(j + 0)
const b = index.getX(j + 1)
const c = index.getX(j + 2)

this.writeFace(a, b, c, positionAttribute, object as SkinnedMesh)
}
} else {
// non-indexed geometry
for (let j = 0; j < positionAttribute.count; j += 3) {
const a = j + 0
const b = j + 1
const c = j + 2

this.writeFace(a, b, c, positionAttribute, object as SkinnedMesh)
}
}
}

if (!this.binary) {
this.output += 'endsolid exported\n'
}

return this.output
}

private writeFace(
a: number,
b: number,
c: number,
positionAttribute: BufferAttribute | InterleavedBufferAttribute,
object: SkinnedMesh,
): void {
this.vA.fromBufferAttribute(positionAttribute, a)
this.vB.fromBufferAttribute(positionAttribute, b)
this.vC.fromBufferAttribute(positionAttribute, c)

if (object.isSkinnedMesh) {
object.applyBoneTransform(a, this.vA)
object.applyBoneTransform(b, this.vB)
object.applyBoneTransform(c, this.vC)
}

this.vA.applyMatrix4(object.matrixWorld)
this.vB.applyMatrix4(object.matrixWorld)
this.vC.applyMatrix4(object.matrixWorld)

this.writeNormal(this.vA, this.vB, this.vC)

this.writeVertex(this.vA)
this.writeVertex(this.vB)
this.writeVertex(this.vC)

if (this.binary && this.output instanceof DataView) {
this.output.setUint16(this.offset, 0, true)
this.offset += 2
} else {
this.output += '\t\tendloop\n'
this.output += '\tendfacet\n'
}
}

private writeNormal(vA: Vector3, vB: Vector3, vC: Vector3): void {
this.cb.subVectors(vC, vB)
this.ab.subVectors(vA, vB)
this.cb.cross(this.ab).normalize()

this.normal.copy(this.cb).normalize()

if (this.binary && this.output instanceof DataView) {
this.output.setFloat32(this.offset, this.normal.x, true)
this.offset += 4
this.output.setFloat32(this.offset, this.normal.y, true)
this.offset += 4
this.output.setFloat32(this.offset, this.normal.z, true)
this.offset += 4
} else {
this.output += `\tfacet normal ${this.normal.x} ${this.normal.y} ${this.normal.z}\n`
this.output += '\t\touter loop\n'
}
}

private writeVertex(vertex: Vector3): void {
if (this.binary && this.output instanceof DataView) {
this.output.setFloat32(this.offset, vertex.x, true)
this.offset += 4
this.output.setFloat32(this.offset, vertex.y, true)
this.offset += 4
this.output.setFloat32(this.offset, vertex.z, true)
this.offset += 4
} else {
this.output += `\t\t\tvertex ${vertex.x} ${vertex.y} ${vertex.z}\n`
}
}
}

0 comments on commit 2f37ad0

Please sign in to comment.