@colyseus/schema 4.0.5 → 4.0.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,10 +1,10 @@
1
1
  import { OPERATION } from "../encoding/spec.js";
2
2
  import { Schema } from "../Schema.js";
3
- import type { Ref } from "../encoder/ChangeTree.js";
3
+ import type { IRef, Ref } from "../encoder/ChangeTree.js";
4
4
  import type { Decoder } from "./Decoder.js";
5
5
  import { Iterator } from "../encoding/decode.js";
6
6
  export interface DataChange<T = any, F = string> {
7
- ref: Ref;
7
+ ref: IRef;
8
8
  refId: number;
9
9
  op: OPERATION;
10
10
  field: F;
@@ -13,7 +13,7 @@ export interface DataChange<T = any, F = string> {
13
13
  previousValue: T;
14
14
  }
15
15
  export declare const DEFINITION_MISMATCH = -1;
16
- export type DecodeOperation<T extends Schema = any> = (decoder: Decoder<T>, bytes: Uint8Array, it: Iterator, ref: Ref, allChanges: DataChange[]) => number | void;
16
+ export type DecodeOperation<T extends Schema = any> = (decoder: Decoder<T>, bytes: Uint8Array, it: Iterator, ref: IRef, allChanges: DataChange[]) => number | void;
17
17
  export declare function decodeValue<T extends Ref>(decoder: Decoder, operation: OPERATION, ref: T, index: number, type: any, bytes: Uint8Array, it: Iterator, allChanges: DataChange[]): {
18
18
  value: any;
19
19
  previousValue: T;
@@ -1,11 +1,11 @@
1
1
  import { TypeContext } from "../types/TypeContext.js";
2
2
  import { Schema } from "../Schema.js";
3
- import type { Ref } from "../encoder/ChangeTree.js";
3
+ import type { IRef } from "../encoder/ChangeTree.js";
4
4
  import type { Iterator } from "../encoding/decode.js";
5
5
  import { ReferenceTracker } from "./ReferenceTracker.js";
6
6
  import { type DataChange } from "./DecodeOperation.js";
7
7
  import { Collection } from "../types/HelperTypes.js";
8
- export declare class Decoder<T extends Schema = any> {
8
+ export declare class Decoder<T extends IRef = any> {
9
9
  context: TypeContext;
10
10
  state: T;
11
11
  root: ReferenceTracker;
@@ -13,7 +13,7 @@ export declare class Decoder<T extends Schema = any> {
13
13
  triggerChanges?: (allChanges: DataChange[]) => void;
14
14
  constructor(root: T, context?: TypeContext);
15
15
  protected setState(root: T): void;
16
- decode(bytes: Uint8Array, it?: Iterator, ref?: Ref): DataChange<any, string>[];
16
+ decode(bytes: Uint8Array, it?: Iterator, ref?: IRef): DataChange<any, string>[];
17
17
  skipCurrentStructure(bytes: Uint8Array, it: Iterator, totalBytes: number): void;
18
18
  getInstanceType(bytes: Uint8Array, it: Iterator, defaultType: typeof Schema): typeof Schema;
19
19
  createInstanceOfType(type: typeof Schema): Schema;
@@ -1,4 +1,4 @@
1
- import { Ref } from "../encoder/ChangeTree.js";
1
+ import type { IRef } from "../encoder/ChangeTree.js";
2
2
  /**
3
3
  * Used for decoding only.
4
4
  */
@@ -6,7 +6,7 @@ export type SchemaCallbacks = {
6
6
  [field: string | number]: Function[];
7
7
  };
8
8
  export declare class ReferenceTracker {
9
- refs: Map<number, Ref>;
9
+ refs: Map<number, IRef>;
10
10
  refCount: {
11
11
  [refId: number]: number;
12
12
  };
@@ -16,7 +16,7 @@ export declare class ReferenceTracker {
16
16
  };
17
17
  protected nextUniqueId: number;
18
18
  getNextUniqueId(): number;
19
- addRef(refId: number, ref: Ref, incrementCount?: boolean): void;
19
+ addRef(refId: number, ref: IRef, incrementCount?: boolean): void;
20
20
  removeRef(refId: number): void;
21
21
  clearRefs(): void;
22
22
  garbageCollectDeletedRefs(): void;
@@ -1,5 +1,5 @@
1
1
  import { Collection, NonFunctionPropNames } from "../../types/HelperTypes.js";
2
- import { Ref } from "../../encoder/ChangeTree.js";
2
+ import type { IRef, Ref } from "../../encoder/ChangeTree.js";
3
3
  import { Decoder } from "../Decoder.js";
4
4
  import { DataChange } from "../DecodeOperation.js";
5
5
  import { OPERATION } from "../../encoding/spec.js";
@@ -18,7 +18,7 @@ type CollectionPropNames<T> = Exclude<{
18
18
  }[keyof T] & string, typeof $refId>;
19
19
  type CollectionValueType<T, K extends keyof T> = T[K] extends MapSchema<infer V, any> ? V : T[K] extends ArraySchema<infer V> ? V : T[K] extends Collection<any, infer V, any> ? V : never;
20
20
  type CollectionKeyType<T, K extends keyof T> = T[K] extends MapSchema<any, infer Key> ? Key : T[K] extends ArraySchema<any> ? number : T[K] extends Collection<infer Key, any, any> ? Key : never;
21
- export declare class StateCallbackStrategy<TState extends Schema> {
21
+ export declare class StateCallbackStrategy<TState extends IRef> {
22
22
  protected decoder: Decoder<TState>;
23
23
  protected uniqueRefIds: Set<number>;
24
24
  protected isTriggering: boolean;
@@ -28,7 +28,7 @@ export declare class StateCallbackStrategy<TState extends Schema> {
28
28
  };
29
29
  protected get state(): TState;
30
30
  protected addCallback(refId: number, operationOrProperty: OPERATION | string, handler: Function): () => void;
31
- protected addCallbackOrWaitCollectionAvailable<TInstance extends Schema, TReturn extends Ref>(instance: TInstance, propertyName: string, operation: OPERATION, handler: Function, immediate?: boolean): () => void;
31
+ protected addCallbackOrWaitCollectionAvailable<TInstance extends IRef, TReturn extends Ref>(instance: TInstance, propertyName: string, operation: OPERATION, handler: Function, immediate?: boolean): () => void;
32
32
  /**
33
33
  * Listen to property changes on the root state.
34
34
  */
@@ -37,7 +37,7 @@ export declare class StateCallbackStrategy<TState extends Schema> {
37
37
  * Listen to property changes on a nested instance.
38
38
  */
39
39
  listen<TInstance extends Schema, K extends PublicPropNames<TInstance>>(instance: TInstance, property: K, handler: PropertyChangeCallback<TInstance[K]>, immediate?: boolean): () => void;
40
- protected listenInstance<TInstance extends Schema>(instance: TInstance, propertyName: string, handler: PropertyChangeCallback<any>, immediate?: boolean): () => void;
40
+ protected listenInstance<TInstance extends IRef>(instance: TInstance, propertyName: string, handler: PropertyChangeCallback<any>, immediate?: boolean): () => void;
41
41
  /**
42
42
  * Listen to any property change on an instance.
43
43
  */
@@ -106,7 +106,7 @@ export declare const Callbacks: {
106
106
  * @param roomOrDecoder - Room or Decoder instance to get the callbacks for.
107
107
  * @returns the new callbacks standard API.
108
108
  */
109
- get<T extends Schema>(roomOrDecoder: Decoder<T> | {
109
+ get<T extends IRef>(roomOrDecoder: Decoder<T> | {
110
110
  serializer: {
111
111
  decoder: Decoder<T>;
112
112
  };
package/build/index.cjs CHANGED
@@ -929,7 +929,11 @@ const Metadata = {
929
929
  });
930
930
  }
931
931
  }
932
- constructor[Symbol.metadata] = metadata;
932
+ Object.defineProperty(constructor, Symbol.metadata, {
933
+ value: metadata,
934
+ writable: false,
935
+ configurable: true
936
+ });
933
937
  return metadata;
934
938
  },
935
939
  isValidInstance(klass) {
@@ -3551,7 +3555,12 @@ function schema(fieldsAndMethods, name, inherits = Schema) {
3551
3555
  }
3552
3556
  else if (value['type'] !== undefined && Schema.is(value['type'])) {
3553
3557
  // Direct Schema type: Type → new Type()
3554
- defaultValues[fieldName] = new value['type']();
3558
+ if (!value['type'].prototype.initialize || value['type'].prototype.initialize.length === 0) {
3559
+ // only auto-initialize Schema instances if:
3560
+ // - they don't have an initialize method
3561
+ // - or initialize method doesn't accept any parameters
3562
+ defaultValues[fieldName] = new value['type']();
3563
+ }
3555
3564
  }
3556
3565
  }
3557
3566
  else {
@@ -3561,7 +3570,12 @@ function schema(fieldsAndMethods, name, inherits = Schema) {
3561
3570
  else if (typeof (value) === "function") {
3562
3571
  if (Schema.is(value)) {
3563
3572
  // Direct Schema type: Type → new Type()
3564
- defaultValues[fieldName] = new value();
3573
+ if (!value.prototype.initialize || value.prototype.initialize.length === 0) {
3574
+ // only auto-initialize Schema instances if:
3575
+ // - they don't have an initialize method
3576
+ // - or initialize method doesn't accept any parameters
3577
+ defaultValues[fieldName] = new value();
3578
+ }
3565
3579
  fields[fieldName] = getNormalizedType(value);
3566
3580
  }
3567
3581
  else {
@@ -3588,13 +3602,32 @@ function schema(fieldsAndMethods, name, inherits = Schema) {
3588
3602
  }
3589
3603
  return defaults;
3590
3604
  };
3605
+ const getParentProps = (props) => {
3606
+ const fieldNames = Object.keys(fields);
3607
+ const parentProps = {};
3608
+ for (const key in props) {
3609
+ if (!fieldNames.includes(key)) {
3610
+ parentProps[key] = props[key];
3611
+ }
3612
+ }
3613
+ return parentProps;
3614
+ };
3591
3615
  /** @codegen-ignore */
3592
3616
  const klass = Metadata.setFields(class extends inherits {
3593
3617
  constructor(...args) {
3594
- super(Object.assign({}, getDefaultValues(), args[0] || {}));
3595
3618
  // call initialize method
3596
3619
  if (methods.initialize && typeof methods.initialize === 'function') {
3597
- methods.initialize.apply(this, args);
3620
+ super(Object.assign({}, getDefaultValues(), getParentProps(args[0] || {})));
3621
+ /**
3622
+ * only call initialize() in the current class, not the parent ones.
3623
+ * see "should not call initialize automatically when creating an instance of inherited Schema"
3624
+ */
3625
+ if (new.target === klass) {
3626
+ methods.initialize.apply(this, args);
3627
+ }
3628
+ }
3629
+ else {
3630
+ super(Object.assign({}, getDefaultValues(), args[0] || {}));
3598
3631
  }
3599
3632
  }
3600
3633
  }, fields);