GameObject.findObjectOfType with NPN def class

Using needleEngine 4.3.2-beta.4, unity 2022.3.43f1 and Node 18.20.3. iā€™m having an issue with GameObject.findObjectOfType, using it with an NPN def class like this:
_camController = GameObject.findObjectOfType(CameraController);
Iā€™m getting these 2 errors:

Type ā€˜IComponent | nullā€™ is not assignable to type ā€˜CameraController | nullā€™.
Type ā€˜IComponentā€™ is missing the following properties from type ā€˜CameraControllerā€™: viewPositions, _onEnterView, selectView, tweenToGameObject, and 24 more.

Argument of type ā€˜typeof CameraControllerā€™ is not assignable to parameter of type ā€˜Constructorā€™.
Types of construct signatures are incompatible.
Type ā€˜new (init?: Partial<NoInternalNeedleEngineState<FilterStartingWith<FilterTypes<Component, Function | null | undefined>, ā€œ">>> | undefined) => CameraControllerā€™ is not assignable to type ā€˜abstract new (ā€¦args: any[]) => IComponentā€™.
Construct signature return types ā€˜CameraControllerā€™ and ā€˜IComponentā€™ are incompatible.
The types of ā€˜gameObject.addNewComponentā€™ are incompatible between these types.
Type '(type: ConstructorConcrete, init?: Partial<NoInternalNeedleEngineState<FilterStartingWith<FilterTypes<T, Function | null | undefined>, "
ā€>>> | undefined) => Tā€™ is not assignable to type ā€˜(type: Constructor, init?: Partial<NoInternalNeedleEngineState<FilterStartingWith<FilterTypes<T, Function | null | undefined>, ā€œ_ā€>>> | undefined) => Tā€™.
Types of parameters ā€˜typeā€™ and ā€˜typeā€™ are incompatible.
Type ā€˜Constructorā€™ is not assignable to type ā€˜ConstructorConcreteā€™.
Cannot assign an abstract constructor type to a non-abstract constructor type.

It works without forcing the implementation but i gotta use @ts-ignore

Hi TomƔs,

The issue youā€™re facing seems to be related to TypeScriptā€™s type system and how it interprets the types involved in your usage of GameObject.findObjectOfType.

Plan

  1. Ensure youā€™re using the correct import statement for findObjectOfType.
  2. Create a custom type guard to safely assert that the found object is indeed a CameraController.
  3. Use this type guard to avoid requiring @ts-ignore.

Code

Hereā€™s how you can implement it:

import { GameObject, IComponent } from "@needle-tools/engine";
import { CameraController } from "./path/to/CameraController"; // Adjust the import path accordingly

// Type guard to check if the object is a CameraController
function isCameraController(object: IComponent | null): object is CameraController {
    return object !== null && (object as CameraController).viewPositions !== undefined; // Check for a specific property from CameraController
}

// Usage
const foundObject = GameObject.findObjectOfType(CameraController);
if (isCameraController(foundObject)) {
    _camController = foundObject; // Now TypeScript knows this is a CameraController
} else {
    _camController = null; // Handle the null case appropriately
}

Explanation

  1. Type Guard: The isCameraController function checks if the object is not null and has a property specific to CameraController. This allows TypeScript to infer the correct type when you set _camController.

  2. Error Handling: By checking the type before assignment, you can avoid the need for @ts-ignore.

With this approach, you maintain type safety without suppressing the TypeScript errors. If you have further questions or need additional help, feel free to ask!

1 replies left - Get a PRO license for more AI replies.
thread_0PSVH3OMd55mukDjPcK1gPzE
Reply to this message to continue the conversation