Crear componentes personalizados
Si eres nuevo en scripting, te recomendamos encarecidamente leer primero las siguientes guías:
Si sabes lo que estás haciendo, siéntete libre de ir directamente a la documentación de la API de Needle Engine.
El código en tiempo de ejecución para Needle Engine está escrito en TypeScript (recomendado) o JavaScript. Generamos automáticamente componentes stub C# a partir de eso, que puedes añadir a GameObjects en el editor. Los componentes C# y sus datos son recreados por el tiempo de ejecución como componentes JavaScript con los mismos datos y adjuntados a objetos three.js.
Tanto los componentes personalizados como los componentes integrados de Unity pueden mapearse a componentes JavaScript de esta manera. Por ejemplo, las asignaciones para muchos componentes integrados relacionados con animación, renderizado o física ya están incluidas en Needle Engine.
Si deseas seguir los siguientes ejemplos sin tener que instalar nada, simplemente haz clic en el siguiente enlace:
Nuestro motor de tiempo de ejecución web adopta un modelo de componentes similar a Unity y, por lo tanto, proporciona mucha funcionalidad que resultará familiar. Los componentes adjuntos a los objetos Object3D de three tienen métodos de ciclo de vida como awake
, start
, onEnable
, onDisable
, update
y lateUpdate
que puedes implementar. También puedes usar Coroutines.
Cuando no necesitas escribir código
A menudo, las escenas interactivas se pueden realizar utilizando Events en Unity y llamando a métodos en componentes integrados. Un ejemplo típico es reproducir una animación al hacer clic en un botón: creas un botón, añades un evento Click en el inspector y haces que llame a Animator.SetTrigger o similar para reproducir una animación específica.
Needle Engine traduce Unity Events en llamadas a métodos JavaScript, lo que hace que este flujo de trabajo sea muy rápido y flexible: configura tus eventos como de costumbre y, cuando se activen, funcionarán igual que en Unity.
Un ejemplo de un Button Click Event que funciona de inmediato en Needle Engine — no se necesita código.
Crear un nuevo componente
Los scripts se escriben en TypeScript (recomendado) o JavaScript. Hay dos formas de añadir scripts personalizados a tu proyecto:
Simplemente añade un archivo con extensión
.ts
o.js
dentro desrc/scripts/
en el directorio de tu proyecto generado, por ejemplosrc/scripts/MyFirstScript.ts
Específico de Unity: Organiza tu código en NPM Definition Files (paquetes npm). Estos te ayudan a modularizar y reutilizar código entre proyectos y, si estás familiarizado con el desarrollo web, de hecho son paquetes npm regulares que se instalan localmente. En Unity puedes crear archivos NpmDef a través de
Create > NPM Definition
y luego añadir archivos TypeScript haciendo clic derecho en un archivo NpmDef y seleccionandoCreate > TypeScript
. Consulta este capítulo para más información.
En ambos enfoques, los directorios de origen son vigilados para detectar cambios y los componentes stub C# o paneles de Blender se regeneran cada vez que se detecta un cambio. Los cambios en los archivos de origen también resultan en una recarga en caliente del sitio web en ejecución – no tienes que esperar a que Unity recompile los componentes C#. Esto hace que iterar sobre el código sea prácticamente instantáneo.
Incluso puedes tener múltiples tipos de componentes dentro de un mismo archivo (por ejemplo, puedes declarar export class MyComponent1
y export class MyOtherComponent
en el mismo archivo Typescript).
Si eres nuevo escribiendo Javascript o Typescript, te recomendamos leer primero la guía Tipscript Essentials Guide antes de continuar con esta guía.
Ejemplo: Crear un Componente que rota un objeto
- Crear un componente que rota un objeto Crea
src/scripts/Rotate.ts
y añade el siguiente código:
import { Behaviour, serializable } from "@needle-tools/engine";
export class Rotate extends Behaviour
{
@serializable()
speed : number = 1;
start(){
// logging this is useful for debugging in the browser.
// You can open the developer console (F12) to see what data your component contains
console.log(this);
}
// update will be called every frame
update(){
this.gameObject.rotateY(this.context.time.deltaTime * this.speed);
}
}
Ahora, dentro de Unity, se generará automáticamente un nuevo script llamado Rotate.cs
. Añade el nuevo componente de Unity a un Cube y guarda la escena. El cubo ahora está rotando dentro del navegador. Abre la consola del desarrollador de chrome presionando F12
para inspeccionar el registro del método Rotate.start
. Esta es una práctica útil para aprender y depurar qué campos se exportan y se asignan actualmente. En general, se exportan todos los campos públicos y serializables y todas las propiedades públicas.
Ahora añade un nuevo campo public float speed = 5
a tu componente de Unity y guárdalo. El inspector del componente Rotate ahora muestra un campo speed
que puedes editar. Guarda la escena (o haz clic en el botón Build
) y observa que el componente javascript ahora tiene el valor speed
exportado asignado.
Crear componente con una función personalizada
Consulta la Typescript Essentials Guide para obtener más información sobre la sintaxis y el lenguaje.
import { Behaviour } from "@needle-tools/engine";
export class PrintNumberComponent extends Behaviour
{
start(){
this.printNumber(42);
}
private printNumber(myNumber : number){
console.log("My Number is: " + myNumber);
}
}
Control de Versiones y Unity
Aunque los componentes C# generados utilizan el nombre del tipo para producir GUIDs estables, recomendamos incluir los componentes generados en el control de versiones como una buena práctica.
Arquitectura de Componentes
Los componentes se añaden a los Object3Ds
de three.js. Esto es similar a cómo se añaden los Componentes en Unity a los GameObjects
. Por lo tanto, cuando queremos acceder a un Object3D de three.js, podemos acceder a él como this.gameObject
, que devuelve el Object3D
al que está adjunto el componente.
Nota: Establecer visible
en falso en un Object3D actuará como SetActive(false)
en Unity, lo que significa que también deshabilitará todos los componentes actuales en este objeto y sus hijos. Los eventos de actualización para componentes inactivos no se llaman hasta que visible
se establece nuevamente en verdadero. Si quieres ocultar un objeto sin afectar a los componentes, simplemente puedes deshabilitar el componente Renderer
de Needle Engine.
Métodos del ciclo de vida
Ten en cuenta que los métodos del ciclo de vida solo se llaman cuando se declaran. Por lo tanto, solo declara los métodos del ciclo de vida update
cuando sean realmente necesarios, de lo contrario, puede perjudicar el rendimiento si tienes muchos componentes con bucles de actualización que no hacen nada.
Nombre del método | Descripción |
---|---|
awake() | Primer método llamado cuando se crea un nuevo componente |
onEnable() | Llamado cuando un componente está habilitado (por ejemplo, cuando enabled cambia de falso a verdadero) |
onDisable() | Llamado cuando un componente está deshabilitado (por ejemplo, cuando enabled cambia de verdadero a falso) |
onDestroy() | llamado cuando el Object3D o el componente está siendo destruido |
start() | Llamado al inicio del primer frame después de que se creó el componente |
earlyUpdate() | Primer evento de actualización |
update() | Evento de actualización por defecto |
lateUpdate() | Llamado después de update |
onBeforeRender() | Último evento de actualización antes de la llamada de renderizado |
onAfterRender() | Llamado después del evento de renderizado |
Métodos de eventos de física
Nombre del método | Descripción |
---|---|
onCollisionEnter(col : Collision) | |
onCollisionStay(col : Collision) | |
onCollisionExit(col : Collision) | |
onTriggerEnter(col : Collision) | |
onTriggerStay(col : Collision) | |
onTriggerExit(col : Collision) |
Métodos de eventos de entrada
Nombre del método | Descripción |
---|---|
onPointerEnter(args : PointerEventData) | Llamado cuando un cursor comienza a pasar sobre un objeto (o cualquiera de sus hijos) |
onPointerMove(args : PointerEventData) | Llamado cuando un cursor se mueve sobre un objeto (o cualquiera de sus hijos) |
onPointerExit(args : PointerEventData) | Llamado cuando un cursor sale (deja de pasar sobre) un objeto |
onPointerDown(args : PointerEventData) | Llamado cuando se presiona un cursor sobre un objeto |
onPointerUp(args : PointerEventData) | Llamado cuando se suelta un cursor sobre un objeto |
onPointerClick(args : PointerEventData) | Llamado cuando se hace clic con un cursor sobre un objeto |
Métodos de eventos XR
requiere Needle Engine >= 3.32.0
Nombre del método | Descripción |
---|---|
supportsXR(mode: XRSessionMode) | Implementa opcionalmente si solo quieres recibir callbacks de XR para modos XR específicos como immersive-vr o immersive-ar . Retorna true para notificar al sistema que quieres callbacks para el modo pasado |
onBeforeXR(mode: XRSessionMode, init: XRSessionInit) | Llamado justo antes de que se solicite una XRSession y se puede usar para modificar el objeto XRSessionInit |
onEnterXR(args: NeedleXREventArgs) | Callback cuando este componente se une a una sesión xr (o se activa en una sesión XR en ejecución) |
onUpdateXR(args: NeedleXREventArgs) | Callback cuando una sesión xr se actualiza (mientras todavía está activa en la sesión XR) |
onLeaveXR(args: NeedleXREventArgs) | Callback cuando este componente sale de una sesión xr (o cuando se vuelve inactivo en una sesión XR en ejecución) |
onControllerAdded(args: NeedleXRControllerEventArgs) | Callback cuando se conecta/añade un controlador mientras está en una sesión XR O cuando el componente se une a una sesión XR en ejecución que ya tiene controladores conectados O cuando el componente se activa durante una sesión XR en ejecución que ya tiene controladores conectados |
onControllerRemoved(args: NeedleXRControllerEventArgs) | Callback cuando se elimina un controlador mientras está en una sesión XR O cuando el componente se vuelve inactivo durante una sesión XR en ejecución |
Eventos XR adicionales
Nombre del método | Descripción |
---|---|
window.addEventListener("needle-xrsession-start") | CustomEvent que se invoca cuando comienza una XRSession. details contiene la NeedleXRSession |
window.addEventListener("needle-xrsession-end") | CustomEvent que se invoca cuando comienza una XRSession. details contiene la NeedleXRSession |
onXRSessionStart(args: { session:NeedleXRSession } ) | Hook de evento global. Para desuscribirse, usa offXRSessionStart |
Coroutines
Las Coroutines se pueden declarar utilizando la Sintaxis de Generadores de JavaScript. Para iniciar una coroutine, llama a this.startCoroutine(this.myRoutineName());
Ejemplo
import { Behaviour, FrameEvent } from "@needle-tools/engine";
export class Rotate extends Behaviour {
start() {
// the second argument is optional and allows you to specifiy
// when it should be called in the current frame loop
// coroutine events are called after regular component events of the same name
// for example: Update coroutine events are called after component.update() functions
this.startCoroutine(this.rotate(), FrameEvent.Update);
}
// this method is called every frame until the component is disabled
*rotate() {
// keep looping forever
while (true) {
yield;
}
}
}
Para detener una coroutine, sal de la rutina retornando de ella, o guarda en caché el valor de retorno de startCoroutine
y llama a this.stopCoroutine(<...>)
. Todas las Coroutines se detienen en onDisable
/ al deshabilitar un componente.
Hooks especiales del ciclo de vida
Needle Engine también expone algunos hooks del ciclo de vida que puedes usar para engancharte al bucle de actualización sin tener que escribir un componente completo. Estos hooks pueden insertarse en cualquier punto de tu aplicación web (por ejemplo, en el ámbito de nivel superior o en un componente svelte)
Nombre del método | Descripción |
---|---|
onInitialized(cb, options) | Llamado cuando se inicializa un nuevo contexto (antes del primer frame) |
onClear(cb, options) | Registra un callback antes de que se limpie el contexto del motor |
onDestroy(cb, options) | Registra un callback en el motor antes de que se destruya el contexto |
onStart(cb, options) | Llamado directamente después de que los componentes start al principio de un frame |
onUpdate(cb, options) | Llamado directamente después de que los componentes update |
onBeforeRender(cb, options) | llamado antes de llamar a render |
onAfterRender(cb, options) | llamado antes de llamar a render |
Por ejemplo (Ver ejemplo en stackblitz)
// this can be put into e.g. main.ts or a svelte component (similar to onMount)
import { onStart, onUpdate, onBeforeRender, onAfterRender } from "@needle-tools/engine"
onStart(ctx => console.log("Hello Scene", ctx.scene));
onUpdate(ctx => {
// do something... e.g. access the frame # or deltatime via ctx.time
console.log("UPDATE", ctx.time.frame);
});
onBeforeRender(ctx => {
// this event is only called once because of the { once: true } argument
console.log("ON BEFORE RENDER", ctx.time.frame);
}, { once: true } );
// Every event hook returns a method to unsubscribe from the event
const unsubscribe = onAfterRender(ctx => {
console.log("ON AFTER RENDER", ctx.time.frame);
});
// Unsubscribe from the event at any time
setTimeout(()=> unsubscribe(), 1000);
Encontrar, añadir y eliminar componentes
Para acceder a otros componentes, utiliza los métodos estáticos en GameObject
o los métodos this.gameObject
. Por ejemplo, para acceder a un componente Renderer
en el padre, utiliza GameObject.getComponentInParent(this.gameObject, Renderer)
o this.gameObject.getComponentInParent(Renderer)
.
Ejemplo:
import { Behaviour, GameObject, Renderer } from "@needle-tools/engine";
export class MyComponent extends Behaviour {
start() {
const renderer = GameObject.getComponentInParent(this.gameObject, Renderer);
console.log(renderer);
}
}
Algunos de los métodos disponibles:
Método | |
---|---|
GameObject.instantiate(Object3D, InstantiateOptions) | crea una nueva instancia de este objeto incluyendo nuevas instancias de todos sus componentes |
GameObject.destroy(Object3D | Component) | destruye un componente o Object3D (y sus componentes) |
GameObject.addNewComponent(Object3D, Type) | añade (y crea) un nuevo componente para un tipo al objeto proporcionado. Ten en cuenta que awake y onEnable ya se llaman cuando se devuelve el componente |
GameObject.addComponent(Object3D, Component) | mueve una instancia de componente al objeto proporcionado. Es útil si ya tienes una instancia, por ejemplo, cuando creas un componente con new MyComponent() y luego lo adjuntas a un objeto. |
GameObject.removeComponent(Component) | elimina un componente de un gameObject |
GameObject.getComponent(Object3D, Type) | devuelve el primer componente que coincide con un tipo en el objeto proporcionado. |
GameObject.getComponents(Object3D, Type) | devuelve todos los componentes que coinciden con un tipo en el objeto proporcionado. |
GameObject.getComponentInChildren | igual que getComponent pero también busca en objetos hijos. |
GameObject.getComponentsInChildren | igual que getComponents pero también busca en objetos hijos. |
GameObject.getComponentInParent | igual que getComponent pero también busca en objetos padre. |
GameObject.getComponentsInParent | igual que getComponents pero también busca en objetos padre. |
GameObject.findObjectOfType | busca en toda la escena un tipo. |
GameObject.findObjectsOfType | busca en toda la escena todos los tipos que coinciden. |
Three.js y el HTML DOM
El contexto se refiere al tiempo de ejecución dentro de un web component. La escena de three.js vive dentro de un componente HTML personalizado llamado <needle-engine>
(consulta el index.html en tu proyecto). Puedes acceder al web component <needle-engine>
utilizando this.context.domElement
.
Esta arquitectura permite tener potencialmente múltiples escenas WebGL de Needle en la misma página web, que pueden ejecutarse por sí solas o comunicarse entre sí como partes de tu página web.
Acceder a la escena
Para acceder a la escena actual desde un componente, utilizas this.scene
, que es equivalente a this.context.scene
; esto te da el objeto raíz de la escena de three.js.
Para recorrer la jerarquía desde un componente, puedes iterar sobre los hijos de un objeto con un bucle for:
for(let i = 0; i < this.gameObject.children; i++)
console.log(this.gameObject.children[i]);
o puedes iterar usando el equivalente a foreach
:
for(const child of this.gameObject.children) {
console.log(child);
}
También puedes usar métodos específicos de three.js para iterar rápidamente todos los objetos recursivamente usando el método traverse
:
import { Object3D } from "three";
this.gameObject.traverse((obj: Object3D) => console.log(obj));
o para recorrer solo objetos visibles, utiliza traverseVisible
en su lugar.
Otra opción que es bastante útil cuando solo quieres iterar objetos que son renderizables es consultar todos los componentes renderer e iterar sobre ellos de la siguiente manera:
import { Renderer } from "@needle-tools/engine";
for(const renderer of this.gameObject.getComponentsInChildren(Renderer))
console.log(renderer);
Para más información sobre cómo obtener componentes, consulta la siguiente sección.
Tiempo
Usa this.context.time
para acceder a los datos de tiempo:
this.context.time.time
es el tiempo transcurrido desde que la aplicación comenzó a ejecutarsethis.context.time.deltaTime
es el tiempo que ha pasado desde el último framethis.context.time.frameCount
es el número de frames que han pasado desde que la aplicación comenzóthis.context.time.realtimeSinceStartup
es el tiempo sin escalar desde que la aplicación comenzó a ejecutarse
También es posible usar this.context.time.timeScale
para ralentizar deliberadamente el tiempo para efectos de cámara lenta, por ejemplo.
Entrada
Recibe datos de entrada para el objeto en el que se encuentra el componente:
import { Behaviour } from "@needle-tools/engine";
export class MyScript extends Behaviour
{
onPointerDown() {
console.log("POINTER DOWN on " + this.gameObject.name);
}
}
También puedes suscribirte a eventos globales en la enumeración InputEvents
de la siguiente manera:
import { Behaviour, InputEvents, NEPointerEvent } from "@needle-tools/engine";
export class MyScript extends Behaviour
{
onEnable() {
this.context.input.addEventListener(InputEvents.PointerDown, this.inputPointerDown);
}
onDisable() {
// it is recommended to also unsubscribe from events when your component becomes inactive
this.context.input.removeEventListener(InputEvents.PointerDown, this.inputPointerDown);
}
// @nonSerialized
inputPointerDown = (evt: NEPointerEvent) => { console.log("POINTER DOWN anywhere on the <needle-engine> element"); }
}
O usa this.context.input
si quieres sondear el estado de entrada en cada frame:
import { Behaviour } from "@needle-tools/engine";
export class MyScript extends Behaviour
{
update() {
if(this.context.input.getPointerDown(0)){
console.log("POINTER DOWN anywhere")
}
}
}
Si quieres manejar las entradas tú mismo, también puedes suscribirte a todos los eventos que proporciona el navegador (hay muchísimos). Por ejemplo, para suscribirte al evento click del navegador, puedes escribir:
import { Behaviour } from "@needle-tools/engine";
export class MyScript extends Behaviour
{
onEnable() {
window.addEventListener("click", this.windowClick);
}
onDisable() {
// unsubscribe again when the component is disabled
window.removeEventListener("click", this.windowClick);
}
windowClick = () => { console.log("CLICK anywhere on the page, not just on <needle-engine>"); }
}
Ten en cuenta que en este caso tienes que manejar todos los casos tú mismo. Por ejemplo, es posible que necesites usar diferentes eventos si tu usuario está visitando tu sitio web en un escritorio, móvil o un dispositivo VR. Needle Engine maneja automáticamente estos casos con los eventos de entrada (por ejemplo, PointerDown
se dispara tanto para el clic del ratón, el toque en pantalla y, en el caso de VR, al presionar un botón del controlador).
Raycasting
Utiliza this.context.physics.raycast()
para realizar un raycast y obtener una lista de intersecciones. Si no pasas ninguna opción, el raycast se realiza desde la posición del ratón (o la primera posición táctil) en espacio de pantalla utilizando la mainCamera
actualmente activa. También puedes pasar un objeto RaycastOptions
que tiene varias configuraciones como maxDistance
, la cámara a usar o las capas contra las que se debe probar.
Utiliza this.context.physics.raycastFromRay(your_ray)
para realizar un raycast utilizando un three.js ray.
Nota: Este tipo de raycast lanza un rayo contra todos los objetos visibles en la escena. No se necesita un motor de física, lo cual es diferente al comportamiento en Unity, donde siempre necesitas colliders para golpear objetos. Si quieres lanzar solo contra colliders de física, utiliza los métodos
physics.engine.raycast
descritos a continuación.
Consideraciones de rendimiento
Cuando se utilizan las configuraciones de compresión predeterminadas de Needle, se crean y utilizan automáticamente versiones simplificadas de las mallas para el raycasting también. Aun así, algunos tipos de mallas son lentas; por ejemplo, las mallas con skinning o las mallas con blendshapes requieren cálculos costosos para determinar los hits exactos. Considera establecer esos objetos en la capa Ignore Raycast
en Unity para evitar el raycasting contra ellos.
Raycasting basado en física
Otra opción es utilizar los métodos de raycast de física que solo devolverán hits con colliders en la escena.
const hit = this.context.physics.engine?.raycast();
Aquí tienes un ejemplo editable de raycast de física
Networking
Se puede acceder a los métodos de Networking a través de this.context.connection
. Consulta la documentación de networking para más información.
Acceder a Needle Engine y componentes desde cualquier lugar
Es posible acceder a toda la funcionalidad descrita anteriormente utilizando código JavaScript regular que no esté dentro de los componentes y que se encuentre en otro lugar. Todos los componentes y la funcionalidad del tiempo de ejecución de Needle son accesibles a través del namespace global Needle
(puedes escribir console.log(Needle)
para obtener una visión general)
Puedes encontrar componentes utilizando Needle.findObjectOfType(Needle.AudioSource)
, por ejemplo. Se recomienda guardar en caché estas referencias, ya que buscar en toda la escena repetidamente es costoso. Consulta la lista para encontrar, añadir y eliminar componentes arriba.
Para obtener callbacks para la carga inicial de la escena, consulta el siguiente ejemplo:
<needle-engine loadstart="loadingStarted" progress="loadingProgress" loadfinished="loadingFinished"></needle-engine>
<script type="text/javascript">
function loadingStarted() { console.log("START") }
function loadingProgress() { console.log("LOADING...") }
function loadingFinished() { console.log("FINISHED!") }
</script>
También puedes suscribirte al NeedleEngine
global (a veces también conocido como ContextRegistry) para recibir un callback cuando se haya creado un contexto de Needle Engine o para acceder a todos los contextos disponibles:
class YourComponentType extends Behaviour {}
//---cut---
import { NeedleEngine, GameObject, Behaviour } from "@needle-tools/engine";
NeedleEngine.addContextCreatedCallback((args) => {
const context = args.context;
const scene = context.scene;
const myInstance = GameObject.getComponentInChildren(scene, YourComponentType);
});
Otra opción es usar el hook del ciclo de vida onInitialized(ctx => {})
También puedes acceder a todos los contextos disponibles a través de NeedleEngine.Registered
, que devuelve el array interno. (Ten en cuenta que este array no debe modificarse, pero se puede usar para iterar sobre todos los contextos activos para modificar configuraciones, por ejemplo, establecer todos los contextos en context.isPaused = true
)
A continuación, encontrarás una lista de eventos disponibles en el tipo estático NeedleEngine
. Puedes suscribirte a estos eventos a través de NeedleEngine.registerCallback(ContextEvent.ContextCreated, (args) => {})
Opciones de ContextEvent | |
---|---|
ContextEvent.ContextRegistered | Llamado cuando el contexto se registra en el registro. |
ContextEvent.ContextCreationStart | Llamado antes de que se cargue el primer glb y se puede usar para inicializar el motor de física. Puede devolver una promesa |
ContextEvent.ContextCreated | Llamado cuando el contexto ha sido creado antes del primer frame |
ContextEvent.ContextDestroyed | Llamado cuando el contexto ha sido destruido |
ContextEvent.MissingCamera | Llamado cuando el contexto no pudo encontrar una cámara, actualmente solo se llama durante la creación |
ContextEvent.ContextClearing | Llamado cuando el contexto se está limpiando: todos los objetos en la escena se están destruyendo y el estado interno se restablece |
ContextEvent.ContextCleared | Llamado después de que el contexto ha sido limpiado |
Gizmos
La clase estática Gizmos
se puede usar para dibujar líneas, formas y texto, lo cual es principalmente útil para la depuración. Todas las funciones de Gizmos tienen múltiples opciones para, por ejemplo, colores o por cuánto tiempo deben mostrarse en la escena. Internamente se almacenan en caché y se reutilizan.
Gizmos | |
---|---|
Gizmos.DrawLabel | Dibuja una etiqueta con fondo opcionalmente. Puede adjuntarse a un objeto. Devuelve un identificador Label que se puede usar para actualizar el texto. |
Gizmos.DrawRay | Toma un origen y una dirección en espacio de mundo para dibujar una línea de rayo infinita |
Gizmos.DrawDirection | Toma un origen y una dirección para dibujar una dirección en espacio de mundo |
Gizmos.DrawLine | Toma dos puntos vec3 en espacio de mundo para dibujar una línea |
Gizmos.DrawWireSphere | Dibuja una esfera de estructura alámbrica en espacio de mundo |
Gizmos.DrawSphere | Dibuja una esfera sólida en espacio de mundo |
Gizmos.DrawWireBox | Dibuja una caja de estructura alámbrica en espacio de mundo |
Gizmos.DrawWireBox3 | Dibuja una box3 de estructura alámbrica |
Gizmos.DrawArrow | Dibuja una flecha tomando dos puntos en espacio de mundo |
Serialización / Components en archivos glTF
Para incrustar componentes y recrear componentes con sus tipos correctos en glTF, también necesitamos guardar tipos no primitivos (todo lo que no sea un Number
, Boolean
o String
). Puedes hacerlo añadiendo un decorador @serializable(<type>)
encima de tu campo o propiedad.
Ejemplo:
import { Behaviour, serializable, Camera } from "@needle-tools/engine";
import { Object3D } from "three"
export class MyClass extends Behaviour {
// this will be a "Transform" field in Unity
@serializable(Object3D)
myObjectReference: Object3D | null = null;
// this will be a "Transform" array field in Unity
// Note that the @serializable decorator contains the array content type! (Object3D and not Object3D[])
@serializable(Object3D)
myObjectReferenceList: Object3D[] | null = null;
// for component or other objects use the object's type
@serializable(Camera)
myCameraComponent: Camera | null = null;
}
Para serializar desde y hacia formatos personalizados, es posible extender la clase TypeSerializer
y crear una instancia. Usa super()
en el constructor para registrar los tipos soportados.
Nota: Además de los campos que coinciden, las propiedades que coinciden también se exportarán cuando coincidan con campos en el archivo typescript.
Cargando Escenas
En Unity, los Prefabs, SceneAssets y AssetReferences (el sistema Addressable de Unity) referenciados se exportarán automáticamente como archivos glTF (consulta la documentación Export Prefabs).
Estos archivos glTF exportados se serializarán como URIs de cadena plana. Para simplificar la carga de estos desde componentes TypeScript, añadimos el concepto de tipos AssetReference
. Se pueden cargar en tiempo de ejecución y, por lo tanto, permiten diferir la carga de partes de tu aplicación o cargar contenido externo.
Ejemplo:
import { Behaviour, serializable, AssetReference } from "@needle-tools/engine";
export class MyClass extends Behaviour {
// if you export a prefab or scene as a reference from Unity you'll get a path to that asset
// which you can de-serialize to AssetReference for convenient loading
@serializable(AssetReference)
myPrefab?: AssetReference;
async start() {
// directly instantiate
const myInstance = await this.myPrefab?.instantiate();
// you can also just load and instantiate later
// const myInstance = await this.myPrefab.loadAssetAsync();
// this.gameObject.add(myInstance)
// this is useful if you know that you want to load this asset only once because it will not create a copy
// since ``instantiate()`` does create a copy of the asset after loading it
}
}
Los AssetReferences se almacenan en caché por URI, por lo que si referencias el mismo glTF/Prefab exportado en múltiples componentes/scripts, solo se cargará una vez y luego se reutilizará.
Próximos pasos
Página traducida automáticamente usando IA