Begin glTF parser
This commit is contained in:
parent
ef7654024a
commit
1355c4e342
@ -11,7 +11,7 @@ export type Camera = OrthographicCamera | PerspectiveCamera;
|
|||||||
export interface OrthographicCameraProps {
|
export interface OrthographicCameraProps {
|
||||||
readonly name?: string;
|
readonly name?: string;
|
||||||
|
|
||||||
readonly verticalSize: number;
|
readonly halfVerticalSize: number;
|
||||||
readonly nearPlane: number;
|
readonly nearPlane: number;
|
||||||
readonly farPlane: number;
|
readonly farPlane: number;
|
||||||
}
|
}
|
||||||
@ -39,13 +39,13 @@ export class OrthographicCamera {
|
|||||||
|
|
||||||
constructor({
|
constructor({
|
||||||
name = "",
|
name = "",
|
||||||
verticalSize,
|
halfVerticalSize,
|
||||||
nearPlane,
|
nearPlane,
|
||||||
farPlane,
|
farPlane,
|
||||||
}: OrthographicCameraProps) {
|
}: OrthographicCameraProps) {
|
||||||
this._name = name;
|
this._name = name;
|
||||||
|
|
||||||
this._halfVerticalSize = verticalSize;
|
this._halfVerticalSize = halfVerticalSize;
|
||||||
this._nearPlane = nearPlane;
|
this._nearPlane = nearPlane;
|
||||||
this._farPlane = farPlane;
|
this._farPlane = farPlane;
|
||||||
|
|
||||||
|
@ -1,3 +1,9 @@
|
|||||||
|
/*!
|
||||||
|
* This Source Code Form is subject to the terms of the Mozilla Public License,
|
||||||
|
* v. 2.0. If a copy of the MPL was not distributed with this file, You can
|
||||||
|
* obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
|
*/
|
||||||
|
|
||||||
export function degToRad(angleDeg: number): number {
|
export function degToRad(angleDeg: number): number {
|
||||||
return angleDeg * Math.PI / 180;
|
return angleDeg * Math.PI / 180;
|
||||||
}
|
}
|
||||||
|
510
src/gltf.ts
Normal file
510
src/gltf.ts
Normal file
@ -0,0 +1,510 @@
|
|||||||
|
/*!
|
||||||
|
* This Source Code Form is subject to the terms of the Mozilla Public License,
|
||||||
|
* v. 2.0. If a copy of the MPL was not distributed with this file, You can
|
||||||
|
* obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as data from "./data";
|
||||||
|
|
||||||
|
/* INITIAL SUPPORT PLAN
|
||||||
|
*
|
||||||
|
* Basic properties:
|
||||||
|
* - extensionsRequired:
|
||||||
|
* - issues error when any extension not supported at least partially
|
||||||
|
* - extensionsUsed: ignored
|
||||||
|
* - accessors: used indirectly
|
||||||
|
* - read when converting mesh
|
||||||
|
* - sparse: no support
|
||||||
|
* - issues error
|
||||||
|
* - animations: no support
|
||||||
|
* - issues warning
|
||||||
|
* - no animations emitted
|
||||||
|
* - asset:
|
||||||
|
* - version: verified
|
||||||
|
* - rest: ignored
|
||||||
|
* - buffers: used indirectly
|
||||||
|
* - read when converting mesh
|
||||||
|
* - uri: no support
|
||||||
|
* - issues error
|
||||||
|
* - bufferViews: used indirectly
|
||||||
|
* - read when converting mesh
|
||||||
|
* - cameras:
|
||||||
|
* - orthographic:
|
||||||
|
* - xmag: ignored
|
||||||
|
* - ymag: converted to halfVerticalSize
|
||||||
|
* - perspective:
|
||||||
|
* - aspectRatio: ignored
|
||||||
|
* - issues warning when provided
|
||||||
|
* - images:
|
||||||
|
* - uri: no support
|
||||||
|
* - issues error
|
||||||
|
* - materials:
|
||||||
|
* - name: full support
|
||||||
|
* - pbrMetallicRoughness:
|
||||||
|
* - baseColorFactor: full support
|
||||||
|
* - baseColorTexture: partial support
|
||||||
|
* - forced texCoord 0
|
||||||
|
* - issues error when different provided
|
||||||
|
* - metallicFactor: full support
|
||||||
|
* - roughnessFactor: full support
|
||||||
|
* - metallicRoughnessTexture: partial support
|
||||||
|
* - forced texCoord 0
|
||||||
|
* - issues error when different provided
|
||||||
|
* - normalTexture: partial support
|
||||||
|
* - scale: full support
|
||||||
|
* - forced texCoord 0
|
||||||
|
* - issues error when different provided
|
||||||
|
* - occlusionTexture: partial support
|
||||||
|
* - strength: full support
|
||||||
|
* - forced texCoord 1
|
||||||
|
* - issues error when different provided
|
||||||
|
* - emissiveTexture: partial support
|
||||||
|
* - forced texCoord 0
|
||||||
|
* - issues error when different provided
|
||||||
|
* - emissiveFactor: full support
|
||||||
|
* - alphaMode:
|
||||||
|
* - OPAQUE: full support
|
||||||
|
* - MASK: no support
|
||||||
|
* - issues error
|
||||||
|
* - BLEND: partial support
|
||||||
|
* - decoded, but not implemented
|
||||||
|
* - doubleSided: prtial support
|
||||||
|
* - decoded, but not implemented
|
||||||
|
*
|
||||||
|
* Extensions:
|
||||||
|
* - KHR_lights_punctual
|
||||||
|
* - name: full support
|
||||||
|
* - color/intensity: full support
|
||||||
|
* - converted to color = color * intensity
|
||||||
|
* - type:
|
||||||
|
* - directional: full support
|
||||||
|
* - point: full support
|
||||||
|
* - spot: no support
|
||||||
|
* - issues error
|
||||||
|
* - range: no support
|
||||||
|
* - issues warning
|
||||||
|
* - always infite range
|
||||||
|
* - KHR_materials_emissive_strength: full support
|
||||||
|
* - converted to emissive = emissive * strength
|
||||||
|
* - KHR_materials_ior: full support
|
||||||
|
* - when not provided, glTF's default is used (1.5) intead of oktaeder's (1.45)
|
||||||
|
* - KHR_materials_ior: full support
|
||||||
|
* - probably
|
||||||
|
*/
|
||||||
|
|
||||||
|
export interface ParseResult {
|
||||||
|
readonly cameras: readonly data.Camera[];
|
||||||
|
readonly materials: readonly data.Material[];
|
||||||
|
readonly lights: readonly data.Light[];
|
||||||
|
readonly scenes: readonly data.Scene[];
|
||||||
|
readonly scene: data.Scene | null;
|
||||||
|
|
||||||
|
readonly warnings: readonly ParseError[];
|
||||||
|
readonly errors: readonly ParseError[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ParseErrorProps {
|
||||||
|
message: string;
|
||||||
|
position?: JsonPosition | undefined;
|
||||||
|
severity: ParseErrorSeverity;
|
||||||
|
options?: ErrorOptions | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ParseErrorSeverity =
|
||||||
|
| "warning"
|
||||||
|
| "error"
|
||||||
|
;
|
||||||
|
|
||||||
|
export class ParseError extends Error {
|
||||||
|
|
||||||
|
override message: string;
|
||||||
|
position: JsonPosition | undefined;
|
||||||
|
severity: ParseErrorSeverity;
|
||||||
|
|
||||||
|
constructor({
|
||||||
|
message,
|
||||||
|
position,
|
||||||
|
severity,
|
||||||
|
options,
|
||||||
|
}: ParseErrorProps) {
|
||||||
|
super(message, options);
|
||||||
|
|
||||||
|
this.message = message;
|
||||||
|
this.position = position;
|
||||||
|
this.severity = severity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface JsonPosition {
|
||||||
|
readonly line: number;
|
||||||
|
readonly column: number;
|
||||||
|
readonly path: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ParseOptions {
|
||||||
|
/**
|
||||||
|
* When `true`, the parser will throw with a `ParseError` on the first error
|
||||||
|
* encountered. This includes warnings when `treatWarningsAsErrors` is
|
||||||
|
* `true`. When `false`, the parser will always return `ParseResult` and is
|
||||||
|
* never expected to throw. Failures are then communicated with the
|
||||||
|
* `ParseResult.errors` array.
|
||||||
|
*
|
||||||
|
* When this option is `true`, `stopOnFirstError` has no effect.
|
||||||
|
* @default true
|
||||||
|
*/
|
||||||
|
readonly throwOnError?: boolean;
|
||||||
|
/**
|
||||||
|
* When `true`, the parser will stop processing on the first error
|
||||||
|
* encountered. This includes warnings when `treatWarningsAsErrors` is
|
||||||
|
* `true`. When `false`, the parser will continue processing when it
|
||||||
|
* encounters an error that it consideres recoverable.
|
||||||
|
*
|
||||||
|
* This option has no effect when `throwOnError` is `true`.
|
||||||
|
* @default true
|
||||||
|
*/
|
||||||
|
readonly stopOnFirstError?: boolean;
|
||||||
|
/**
|
||||||
|
* When `true`, the parser will treat any encountered warning as a failure
|
||||||
|
* for the purpose of the other options. Note that regardless of this
|
||||||
|
* option, the warnings will always be returned in the
|
||||||
|
* `ParseResult.warnings` array and they will always have their `severity`
|
||||||
|
* property equal to `"warning"`.
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
|
readonly treatWarningsAsErrors?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function parse(gltf: ArrayBufferView, {
|
||||||
|
throwOnError = true,
|
||||||
|
stopOnFirstError = true,
|
||||||
|
treatWarningsAsErrors = false,
|
||||||
|
}: ParseOptions = {}): Promise<ParseResult> {
|
||||||
|
|
||||||
|
const cameras: data.Camera[] = [];
|
||||||
|
const materials: data.Material[] = [];
|
||||||
|
const lights: data.Light[] = [];
|
||||||
|
const scenes: data.Scene[] = [];
|
||||||
|
const scene: data.Scene | null = null;
|
||||||
|
|
||||||
|
const warnings: ParseError[] = [];
|
||||||
|
const errors: ParseError[] = [];
|
||||||
|
|
||||||
|
function makeParseResult(): ParseResult {
|
||||||
|
return Object.freeze({
|
||||||
|
cameras: Object.freeze(cameras),
|
||||||
|
materials: Object.freeze(materials),
|
||||||
|
lights: Object.freeze(lights),
|
||||||
|
scenes: Object.freeze(scenes),
|
||||||
|
scene,
|
||||||
|
|
||||||
|
warnings: Object.freeze(warnings),
|
||||||
|
errors: Object.freeze(errors),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let gltfDataView = new DataView(gltf.buffer, gltf.byteOffset, gltf.byteLength);
|
||||||
|
|
||||||
|
// --- GLB HEADER ----------------------------------------------------------
|
||||||
|
|
||||||
|
if (gltfDataView.byteLength < 12) {
|
||||||
|
const message = `glTF buffer view is too short to be a valid binary glTF container. Binary glTF begins with a 12-byte header, but the provided buffer view has byte length of ${gltf.byteLength}`;
|
||||||
|
const error = new ParseError({ message, severity: "error" });
|
||||||
|
if (throwOnError) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
errors.push(error);
|
||||||
|
// unrecoverable error
|
||||||
|
return makeParseResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
const magic = gltfDataView.getUint32(0, true);
|
||||||
|
const version = gltfDataView.getUint32(4, true);
|
||||||
|
let length = gltfDataView.getUint32(8, true);
|
||||||
|
|
||||||
|
if (magic !== 0x46546C67) {
|
||||||
|
const message = `glTF container has invalid magic bytes. The first four bytes must have a value of 0x46546C67 when read as little endian unsigned integer, but in the provided buffer view they have the value of ${u32toHexString(magic)}`;
|
||||||
|
const error = new ParseError({ message, severity: "error" });
|
||||||
|
if (throwOnError) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
errors.push(error);
|
||||||
|
/* NOTE This error is considered unrecoverable, because it is very
|
||||||
|
* likely that when the magic bytes are wrong, the provided buffer
|
||||||
|
* view points to a completely different format or garbage data and
|
||||||
|
* it would be pointless to continue parsing in this case.
|
||||||
|
*/
|
||||||
|
return makeParseResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (version !== 2) {
|
||||||
|
const message = `Unsupported binary glTF container format. The bytes 4-8 define the binary glTF conatiner format version when read as little endian unsigned integer. Only version 2 is supported, but in the provided buffer they have the value of ${version}`;
|
||||||
|
const error = new ParseError({ message, severity: "error" });
|
||||||
|
if (throwOnError) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
errors.push(error);
|
||||||
|
// unrecoverable error
|
||||||
|
return makeParseResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (length !== gltf.byteLength) {
|
||||||
|
const message = `Invalid glTF container length. The bytes 8-12 define the length in bytes of the entirety of the binary glTF container when read as little endian unsigned integer. The container byte length is defined as ${length}, but the provided buffer view has byte length of ${gltf.byteLength}`;
|
||||||
|
const error = new ParseError({ message, severity: "error" });
|
||||||
|
if (throwOnError) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
errors.push(error);
|
||||||
|
// recovery: use the lower length value and pretend its the actual length
|
||||||
|
length = Math.min(length, gltf.byteLength);
|
||||||
|
gltfDataView = new DataView(gltf.buffer, gltf.byteOffset, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
let rest = new DataView(gltf.buffer, gltf.byteOffset + 12, gltf.byteLength - 12);
|
||||||
|
|
||||||
|
// --- JSON CHUNK ----------------------------------------------------------
|
||||||
|
|
||||||
|
throw new Error("TODO");
|
||||||
|
|
||||||
|
// --- BIN CHUNK -----------------------------------------------------------
|
||||||
|
}
|
||||||
|
|
||||||
|
function u32toHexString(value: number) {
|
||||||
|
return "0x" + ("00000000" + value.toString(16)).slice(-8);
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- GLTF DATA STRUCTURES ----------------------------------------------------
|
||||||
|
|
||||||
|
export interface Gltf {
|
||||||
|
extensionsRequired?: [string, ...string[]];
|
||||||
|
accessors?: [Accessor, ...Accessor[]];
|
||||||
|
asset: Asset;
|
||||||
|
buffers?: [Buffer, ...Buffer[]];
|
||||||
|
bufferViews?: [BufferView, ...BufferView[]];
|
||||||
|
cameras?: [Camera, ...Camera[]];
|
||||||
|
images?: [Image, ...Image[]];
|
||||||
|
materials?: [Material, ...Material[]];
|
||||||
|
meshes?: [Mesh, ...Mesh[]];
|
||||||
|
nodes?: [Node, ...Node[]];
|
||||||
|
samplers?: [Sampler, ...Sampler[]];
|
||||||
|
scene?: number;
|
||||||
|
scenes?: [number, ...number[]];
|
||||||
|
textures?: [Texture, ...Texture[]];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Accessor {
|
||||||
|
bufferView?: number;
|
||||||
|
/** @default 0 */
|
||||||
|
byteOffset?: number;
|
||||||
|
componentType: ComponentType;
|
||||||
|
/** @default false */
|
||||||
|
normalized?: boolean;
|
||||||
|
count: number;
|
||||||
|
type: AccessorType;
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum ComponentType {
|
||||||
|
Byte = 5120,
|
||||||
|
UnsignedByte = 5121,
|
||||||
|
Short = 5122,
|
||||||
|
UnsignedShort = 5123,
|
||||||
|
UnsignedInt = 5125,
|
||||||
|
Float = 5126,
|
||||||
|
}
|
||||||
|
|
||||||
|
export type AccessorType =
|
||||||
|
| "SCALAR"
|
||||||
|
| "VEC2"
|
||||||
|
| "VEC3"
|
||||||
|
| "VEC4"
|
||||||
|
| "MAT2"
|
||||||
|
| "MAT3"
|
||||||
|
| "MAT4"
|
||||||
|
;
|
||||||
|
|
||||||
|
export interface Asset {
|
||||||
|
version: `${string}.${string}`;
|
||||||
|
minVersion: `${string}.${string}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Buffer {
|
||||||
|
uri?: string;
|
||||||
|
byteLength: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface BufferView {
|
||||||
|
buffer: number;
|
||||||
|
/** @default 0 */
|
||||||
|
byteOffset?: number;
|
||||||
|
byteLength: number;
|
||||||
|
byteStride?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type Camera =
|
||||||
|
| CameraOrthographic
|
||||||
|
| CameraPerspective
|
||||||
|
;
|
||||||
|
|
||||||
|
export interface CameraOrthographic {
|
||||||
|
orthographic: Orthographic;
|
||||||
|
type: "orthographic";
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CameraPerspective {
|
||||||
|
orthographic: Perspective;
|
||||||
|
type: "perspective";
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Orthographic {
|
||||||
|
xmag: number;
|
||||||
|
ymag: number;
|
||||||
|
zfar: number;
|
||||||
|
znear: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Perspective {
|
||||||
|
aspectRatio?: number;
|
||||||
|
yfov: number;
|
||||||
|
zfar?: number;
|
||||||
|
znear: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Image {
|
||||||
|
uri?: string;
|
||||||
|
mimeType?: ImageMimeType;
|
||||||
|
bufferView?: number;
|
||||||
|
name?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ImageMimeType =
|
||||||
|
| "image/jpeg"
|
||||||
|
| "image/png"
|
||||||
|
;
|
||||||
|
|
||||||
|
export interface Material {
|
||||||
|
name?: string;
|
||||||
|
extensions?: MaterialExtensions;
|
||||||
|
pbrMetallicRoughness?: MaterialPbrMetallicRoughness;
|
||||||
|
normalTexture?: NormalTextureInfo;
|
||||||
|
occlusionTexture?: OcclusionTextureInfo;
|
||||||
|
emissiveTexture?: TextureInfo;
|
||||||
|
emissiveFactor?: [r: number, g: number, b: number];
|
||||||
|
alphaMode?: AlphaMode;
|
||||||
|
/** @default false */
|
||||||
|
doubleSided?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface MaterialExtensions {
|
||||||
|
KHR_materials_emissive_strength?: KHR_materials_emissive_strength;
|
||||||
|
KHR_materials_ior?: KHR_materials_ior;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface KHR_materials_emissive_strength {
|
||||||
|
/** @default 1 */
|
||||||
|
emissiveStrength?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface KHR_materials_ior {
|
||||||
|
/** @default 1.5 */
|
||||||
|
ior?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface MaterialPbrMetallicRoughness {
|
||||||
|
/** @default [1, 1, 1, 1] */
|
||||||
|
baseColorFactor?: [r: number, b: number, g: number, partialCoverage: number];
|
||||||
|
baseColorTexture?: TextureInfo;
|
||||||
|
/** @default 1 */
|
||||||
|
metallicFactor?: number;
|
||||||
|
/** @default 1 */
|
||||||
|
roughnessFactor?: number;
|
||||||
|
metallicRoughnessTexture?: TextureInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TextureInfo {
|
||||||
|
index: number;
|
||||||
|
/** @default 0 */
|
||||||
|
texCoord?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface NormalTextureInfo extends TextureInfo {
|
||||||
|
/** @default 1 */
|
||||||
|
scale?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface OcclusionTextureInfo extends TextureInfo {
|
||||||
|
/** @default 1 */
|
||||||
|
strength?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type AlphaMode =
|
||||||
|
| "OPAQUE"
|
||||||
|
| "MASK"
|
||||||
|
| "BLEND"
|
||||||
|
;
|
||||||
|
|
||||||
|
export interface Mesh {
|
||||||
|
primitives: [Primitive, ...Primitive[]];
|
||||||
|
name?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Primitive {
|
||||||
|
attributes: {
|
||||||
|
POSITION?: number,
|
||||||
|
NORMAL?: number,
|
||||||
|
TANGENT?: number,
|
||||||
|
TEXCOORD_0?: number,
|
||||||
|
TEXCOORD_1?: number,
|
||||||
|
};
|
||||||
|
indices?: number;
|
||||||
|
material?: number;
|
||||||
|
/** @default PrimitiveMode.Triangles */
|
||||||
|
mode?: PrimitiveMode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum PrimitiveMode {
|
||||||
|
Points = 0,
|
||||||
|
Lines = 1,
|
||||||
|
LineLoop = 2,
|
||||||
|
LineStrip = 3,
|
||||||
|
Triangles = 4,
|
||||||
|
TriangleStrip = 5,
|
||||||
|
TriangleFan = 6,
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Sampler {
|
||||||
|
magFilter?: MagFilter;
|
||||||
|
minFilter?: MinFilter;
|
||||||
|
/** @default WrappingMode.Repeat */
|
||||||
|
wrapS?: WrappingMode;
|
||||||
|
/** @default WrappingMode.Repeat */
|
||||||
|
wrapT?: WrappingMode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum MagFilter {
|
||||||
|
Nearest = 9728,
|
||||||
|
Linear = 9729,
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum MinFilter {
|
||||||
|
Nearest = 9728,
|
||||||
|
Linear = 9729,
|
||||||
|
NearestMipmapNearest = 9984,
|
||||||
|
LinearMipmapNearest = 9985,
|
||||||
|
NearestMipmapLinear = 9986,
|
||||||
|
LinearMipmapLinear = 9987,
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum WrappingMode {
|
||||||
|
ClampToEdge = 33071,
|
||||||
|
MirroredRepeat = 33648,
|
||||||
|
Repeat = 10497,
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Texture {
|
||||||
|
sampler?: number;
|
||||||
|
source?: number;
|
||||||
|
name?: string;
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user