Skip to content
This repository has been archived by the owner on Aug 21, 2024. It is now read-only.

Resource State #9632

Merged
merged 88 commits into from
Feb 14, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
88 commits
Select commit Hold shift + click to select a range
ef60ca0
Start of Resource State
MichaelEstes Jan 16, 2024
1054824
licenses
MichaelEstes Jan 16, 2024
b49b8b5
Start of resource manager hooks
MichaelEstes Jan 17, 2024
4226e4e
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 17, 2024
5e301f7
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 17, 2024
3c8243c
Resource hook refactoring
MichaelEstes Jan 17, 2024
4c47a5a
useTexture for SkyboxComponent
MichaelEstes Jan 17, 2024
2655b3f
Start of unloading logic
MichaelEstes Jan 17, 2024
2be8cdd
Include entity in useTexture
MichaelEstes Jan 17, 2024
2d9f5ed
Seperate logic by resource type
MichaelEstes Jan 18, 2024
e5ff4d2
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 18, 2024
3cd761b
Update Three cache
MichaelEstes Jan 18, 2024
284203c
Avatar loading system to resource manager
MichaelEstes Jan 18, 2024
9cf9fd3
Particle system to resource manager
MichaelEstes Jan 18, 2024
a23ea3e
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 18, 2024
7bc8a6d
Per type metadata
MichaelEstes Jan 18, 2024
531fb95
Include entity
MichaelEstes Jan 18, 2024
038331d
Unload assets when changed in editor, onUnload callback
MichaelEstes Jan 19, 2024
d2820de
Image component resource manager hooks
MichaelEstes Jan 19, 2024
dba2b8d
Don't use unloaded callback when the component is being unmounted
MichaelEstes Jan 19, 2024
ec4036c
Batch loader, loopAnimationComponent, AvatarAnimationSystem resource …
MichaelEstes Jan 19, 2024
df76a2c
Spawn point component to resource hooks, make sure unload only is cal…
MichaelEstes Jan 19, 2024
3c38a6a
Better typing
MichaelEstes Jan 20, 2024
ab41c92
onStart callback for setting initial metadata per type
MichaelEstes Jan 20, 2024
4b9351c
Update loaders to use abort controller signal
MichaelEstes Jan 22, 2024
dc2a8d2
license
MichaelEstes Jan 22, 2024
a8c2b3f
loader typings
MichaelEstes Jan 22, 2024
83f5b44
Portal texture resource hook
MichaelEstes Jan 22, 2024
781f43e
Media component resource hooks, potential on gpu callback
MichaelEstes Jan 22, 2024
3a9d192
EnvmapComponent resource hooks
MichaelEstes Jan 22, 2024
018cab7
rename abort controller function
MichaelEstes Jan 22, 2024
64358ed
fix ordering
MichaelEstes Jan 23, 2024
fc2ccf8
Envmap and Hyperspace to resource hooks
MichaelEstes Jan 23, 2024
4edc20b
seperate resources by type, start of tests
MichaelEstes Jan 23, 2024
ed96d01
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 23, 2024
e77f7f5
Update typing
MichaelEstes Jan 23, 2024
db6c2b5
Remove references to three FileLoader, Loader
MichaelEstes Jan 23, 2024
33f30af
Replace three loader reference
MichaelEstes Jan 23, 2024
3a6ec21
Parse function is optional
MichaelEstes Jan 23, 2024
eeed674
Custom loading manager with more hooks, tests
MichaelEstes Jan 24, 2024
53aa851
license
MichaelEstes Jan 24, 2024
fd19ef5
cleanup
MichaelEstes Jan 24, 2024
c5e60db
tests
MichaelEstes Jan 24, 2024
7f78ef8
tests
MichaelEstes Jan 24, 2024
a2a57db
tests
MichaelEstes Jan 24, 2024
a63b58c
Remove materials on model unload
MichaelEstes Jan 24, 2024
6d725a4
Model component fix wip
MichaelEstes Jan 25, 2024
48fdabe
Track textures loaded by the GLTFLoader
MichaelEstes Jan 26, 2024
3bca691
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 26, 2024
c1ac967
Update imports
MichaelEstes Jan 26, 2024
760f720
Resource manager track and manage materials, geometry
MichaelEstes Jan 26, 2024
e4d376c
Cleanup, no need to set asset to null when resources are disposed
MichaelEstes Jan 26, 2024
fd0355c
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 26, 2024
093893b
Track resources used by avatar preview, preliminary fixes for unloadi…
MichaelEstes Jan 27, 2024
b9656d0
unload correctly in avatar preview, envmapcomponent, model component
MichaelEstes Jan 29, 2024
92e0c56
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 29, 2024
16a304b
Update imports
MichaelEstes Jan 29, 2024
7aea3ba
Start of variant resource manager
MichaelEstes Jan 30, 2024
de97c1a
Resource manager loads variants if exists
MichaelEstes Jan 31, 2024
abb90a8
Merge branch 'dev' into ResourceManager
MichaelEstes Jan 31, 2024
588cbe2
hookstate fix
MichaelEstes Jan 31, 2024
30e3985
Component typing
MichaelEstes Jan 31, 2024
7779358
Merge branch 'dev' into ResourceManager
MichaelEstes Feb 1, 2024
fa66e8f
Merge branch 'dev' into ResourceManager
HexaField Feb 5, 2024
3478a38
Merge branch 'dev' into ResourceManager
HexaField Feb 6, 2024
208c4b6
Avatar preview animation fixes
MichaelEstes Feb 6, 2024
e2ec75b
Merge branch 'dev' into ResourceManager
MichaelEstes Feb 6, 2024
2892502
Mesh bvh hookstate and race condition fix, shadow system unmount fix
MichaelEstes Feb 6, 2024
f68e097
Mesh bvh fixes, object layer component fixes
MichaelEstes Feb 6, 2024
9c3fdcd
Preliminary fix for avatar switching
MichaelEstes Feb 7, 2024
7714d55
Fix errors when selecting a new avatar
MichaelEstes Feb 7, 2024
182d437
Merge branch 'dev' into ResourceManager
MichaelEstes Feb 7, 2024
2a02c32
track textures by source uuid, remove dispose calls, mesh bvh hooksta…
MichaelEstes Feb 8, 2024
2f3b46b
debug change
MichaelEstes Feb 8, 2024
b1330e1
Update mesh bvh
MichaelEstes Feb 8, 2024
0c5fade
Merge branch 'dev' into ResourceManager
MichaelEstes Feb 8, 2024
ba0aeb4
remove property
MichaelEstes Feb 8, 2024
d4a4906
test fix
MichaelEstes Feb 8, 2024
07f3d2a
Merge branch 'dev' into ResourceManager
MichaelEstes Feb 9, 2024
df00fb8
Merge branch 'dev' into ResourceManager
MichaelEstes Feb 12, 2024
1d5d89f
Fix for multiple avtars shown in avatar selection panel
MichaelEstes Feb 12, 2024
65874fe
Merge branch 'dev' into ResourceManager
MichaelEstes Feb 12, 2024
dae6e74
Avatar selection working
MichaelEstes Feb 13, 2024
91afb0b
Merge branch 'dev' into ResourceManager
MichaelEstes Feb 13, 2024
cc5cc1f
Merge branch 'dev' into ResourceManager
DanielBelmes Feb 14, 2024
04d1dc0
bug fixes
HexaField Feb 14, 2024
746388b
src ref
HexaField Feb 14, 2024
1d5b96d
Merge branch 'dev' into ResourceManager
HexaField Feb 14, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion packages/engine/src/assets/classes/AssetLoader.ts
Original file line number Diff line number Diff line change
Expand Up @@ -369,7 +369,7 @@ const assetLoadCallback =

const getAbsolutePath = (url) => (isAbsolutePath(url) ? url : getState(EngineState).publicPath + url)

type LoadingArgs = {
export type LoadingArgs = {
ignoreDisposeGeometry?: boolean
forceAssetType?: AssetType
assetRoot?: Entity
Expand Down
61 changes: 61 additions & 0 deletions packages/engine/src/assets/functions/resourceHooks.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
/*
CPAL-1.0 License

The contents of this file are subject to the Common Public Attribution License
Version 1.0. (the "License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
https://github.com/EtherealEngine/etherealengine/blob/dev/LICENSE.
The License is based on the Mozilla Public License Version 1.1, but Sections 14
and 15 have been added to cover use of software over a computer network and
provide for limited attribution for the Original Developer. In addition,
Exhibit A has been modified to be consistent with Exhibit B.

Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the
specific language governing rights and limitations under the License.

The Original Code is Ethereal Engine.

The Original Developer is the Initial Developer. The Initial Developer of the
Original Code is the Ethereal Engine team.

All portions of the code written by the Ethereal Engine team are Copyright © 2021-2023
Ethereal Engine. All Rights Reserved.
*/

import { State, useHookstate } from '@etherealengine/hyperflux'
import { useEffect } from 'react'
import { Entity, UndefinedEntity } from '../../ecs/classes/Entity'
import { LoadingArgs } from '../classes/AssetLoader'
import { GLTF } from '../loaders/gltf/GLTFLoader'
import { ResourceManager, ResourceType } from '../state/ResourceState'

export function useGLTF(
src: string,
params: LoadingArgs = {},
entity?: Entity
): [State<GLTF | null>, State<ProgressEvent<EventTarget> | null>, State<ErrorEvent | Error | null>] {
const gltf = useHookstate<GLTF | null>(null)
const progress = useHookstate<ProgressEvent<EventTarget> | null>(null)
const error = useHookstate<ErrorEvent | Error | null>(null)

useEffect(() => {
ResourceManager.load(
src,
ResourceType.GLTF,
entity || UndefinedEntity,
params,
(response) => {
gltf.set(response)
},
(request) => {
progress.set(request)
},
(err) => {
error.set(err)
}
)
}, [src])

return [gltf, progress, error]
}
150 changes: 150 additions & 0 deletions packages/engine/src/assets/state/ResourceState.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,150 @@
/*
CPAL-1.0 License

The contents of this file are subject to the Common Public Attribution License
Version 1.0. (the "License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
https://github.com/EtherealEngine/etherealengine/blob/dev/LICENSE.
The License is based on the Mozilla Public License Version 1.1, but Sections 14
and 15 have been added to cover use of software over a computer network and
provide for limited attribution for the Original Developer. In addition,
Exhibit A has been modified to be consistent with Exhibit B.

Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the
specific language governing rights and limitations under the License.

The Original Code is Ethereal Engine.

The Original Developer is the Initial Developer. The Initial Developer of the
Original Code is the Ethereal Engine team.

All portions of the code written by the Ethereal Engine team are Copyright © 2021-2023
Ethereal Engine. All Rights Reserved.
*/

import { defineState, getMutableState, getState } from '@etherealengine/hyperflux'
import { Entity } from '../../ecs/classes/Entity'
import { AssetLoader, LoadingArgs } from '../classes/AssetLoader'

//@ts-ignore
THREE.Cache.enabled

//@ts-ignore
THREE.DefaultLoadingManager.onLoad = () => {
const totalSize = getCurrentSizeOfResources()
console.log('Loaded: ' + totalSize + ' bytes of resources')
}

//Called when the item at the url passed in has completed loading
//@ts-ignore
THREE.DefaultLoadingManager.onProgress = (url: string, loaded: number, total: number) => {
console.log('On Progress', url, loaded, total)
}

//@ts-ignore
THREE.DefaultLoadingManager.onError = (url: string) => {
console.log('On Error', url)
}

//This doesn't work as you might imagine, it is only called once, the url parameter is pretty much useless
//@ts-ignore
THREE.DefaultLoadingManager.onStart = (url: string, loaded: number, total: number) => {
console.log('On Start', url, loaded, total)
}

enum ResourceStatus {
Unloaded,
Loading,
Loaded,
Error
}

export enum ResourceType {
GLTF,
Texture,
Geometry,
ECSData,
Audio,
Unknown
}

type Resource = {
status: ResourceStatus
type: ResourceType
references: Entity[]
assetRef?: any
metadata: {
size?: number
}
onGPU: boolean
}

export const ResourceState = defineState({
name: 'ResourceManagerState',
initial: () => ({
resources: {} as Record<string, Resource>
HexaField marked this conversation as resolved.
Show resolved Hide resolved
})
})

const getCurrentSizeOfResources = () => {
let size = 0
const resources = getState(ResourceState).resources
for (const key in resources) {
const resource = resources[key]
if (resource.metadata.size) size += resource.metadata.size
}

return size
}

const load = (
url: string,
resourceType: ResourceType,
entity: Entity,
args: LoadingArgs,
onLoad = (response: any) => {},
onProgress = (request: ProgressEvent) => {},
onError = (event: ErrorEvent | Error) => {}
) => {
const resourceState = getMutableState(ResourceState)
const resources = resourceState.nested('resources')
if (!resources[url].value) {
resources.merge({
[url]: {
status: ResourceStatus.Unloaded,
type: resourceType,
references: [entity],
metadata: {},
onGPU: false
}
})
} else {
resources[url].references.merge([entity])
}

const resource = resources[url]

AssetLoader.load(
url,
args,
(response) => {
resource.status.set(ResourceStatus.Loaded)
resource.assetRef.set(response)
onLoad(response)
},
(request) => {
resource.status.set(ResourceStatus.Loading)
resource.metadata.size.set(request.total)
onProgress(request)
},
(error) => {
resource.status.set(ResourceStatus.Error)
onError(error)
}
)
}

export const ResourceManager = {
load
}
101 changes: 56 additions & 45 deletions packages/engine/src/scene/components/ModelComponent.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -30,8 +30,8 @@ import { NO_PROXY, createState, getMutableState, getState, none, useHookstate }

import { VRM } from '@pixiv/three-vrm'
import React from 'react'
import { AssetLoader } from '../../assets/classes/AssetLoader'
import { AssetType } from '../../assets/enum/AssetType'
import { useGLTF } from '../../assets/functions/resourceHooks'
import { GLTF } from '../../assets/loaders/gltf/GLTFLoader'
import { AnimationComponent } from '../../avatar/components/AnimationComponent'
import { SkinnedMeshComponent } from '../../avatar/components/SkinnedMeshComponent'
Expand Down Expand Up @@ -146,6 +146,61 @@ function ModelReactor(): JSX.Element {
const modelComponent = useComponent(entity, ModelComponent)
const variantComponent = useOptionalComponent(entity, VariantComponent)

/** @todo this is a hack */
const override = !isAvaturn(modelComponent.value.src) ? undefined : AssetType.glB
const [gltf, progress, error] = useGLTF(
modelComponent.value.src,
{
forceAssetType: override,
ignoreDisposeGeometry: modelComponent.cameraOcclusion.value
},
entity
)

useEffect(() => {
const onprogress = progress.value
if (!onprogress) return
if (hasComponent(entity, SceneAssetPendingTagComponent))
SceneAssetPendingTagComponent.loadingProgress.merge({
[entity]: {
loadedAmount: onprogress.loaded,
totalAmount: onprogress.total
}
})
}, [progress])

useEffect(() => {
const loadedAsset = gltf.get(NO_PROXY)
if (!loadedAsset) return

if (variantComponent && !variantComponent.calculated.value) return
if (typeof loadedAsset !== 'object') {
addError(entity, ModelComponent, 'INVALID_SOURCE', 'Invalid URL')
return
}

const boneMatchedAsset = modelComponent.convertToVRM.value
? (autoconvertMixamoAvatar(loadedAsset) as GLTF)
: loadedAsset
/**if we've loaded or converted to vrm, create animation component whose mixer's root is the normalized rig */
if (boneMatchedAsset instanceof VRM)
setComponent(entity, AnimationComponent, {
animations: loadedAsset.animations,
mixer: new AnimationMixer(boneMatchedAsset.humanoid.normalizedHumanBones.hips.node)
})

modelComponent.asset.set(boneMatchedAsset)
}, [gltf])

useEffect(() => {
const err = error.value
if (!err) return

console.error(err)
addError(entity, ModelComponent, 'INVALID_SOURCE', err.message)
removeComponent(entity, SceneAssetPendingTagComponent)
}, [error])

useEffect(() => {
let aborted = false
if (variantComponent && !variantComponent.calculated.value) return
Expand All @@ -163,50 +218,6 @@ function ModelReactor(): JSX.Element {
proxifyParentChildRelationships(obj3d)
}

/** @todo this is a hack */
const override = !isAvaturn(model.src) ? undefined : AssetType.glB

AssetLoader.load(
modelComponent.src.value,
{
forceAssetType: override,
ignoreDisposeGeometry: modelComponent.cameraOcclusion.value
},
(loadedAsset) => {
if (variantComponent && !variantComponent.calculated.value) return
if (aborted) return
if (typeof loadedAsset !== 'object') {
addError(entity, ModelComponent, 'INVALID_SOURCE', 'Invalid URL')
return
}
const boneMatchedAsset = modelComponent.convertToVRM.value
? (autoconvertMixamoAvatar(loadedAsset) as GLTF)
: loadedAsset
/**if we've loaded or converted to vrm, create animation component whose mixer's root is the normalized rig */
if (boneMatchedAsset instanceof VRM)
setComponent(entity, AnimationComponent, {
animations: loadedAsset.animations,
mixer: new AnimationMixer(boneMatchedAsset.humanoid.normalizedHumanBones.hips.node)
})
modelComponent.asset.set(boneMatchedAsset)
},
(onprogress) => {
if (aborted) return
if (hasComponent(entity, SceneAssetPendingTagComponent))
SceneAssetPendingTagComponent.loadingProgress.merge({
[entity]: {
loadedAmount: onprogress.loaded,
totalAmount: onprogress.total
}
})
},
(err: Error) => {
if (aborted) return
console.error(err)
addError(entity, ModelComponent, 'INVALID_SOURCE', err.message)
removeComponent(entity, SceneAssetPendingTagComponent)
}
)
return () => {
aborted = true
}
Expand Down
Loading