@needle-tools/engine 5.1.0-alpha.7 → 5.1.0-alpha.8

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.
Files changed (52) hide show
  1. package/dist/{needle-engine.bundle-BwS5IsLn.js → needle-engine.bundle-3LOy9tj8.js} +6545 -6525
  2. package/dist/{needle-engine.bundle-BNj2FJS7.min.js → needle-engine.bundle-BuI2pjv1.min.js} +144 -144
  3. package/dist/{needle-engine.bundle-tHGdwr9a.umd.cjs → needle-engine.bundle-DQy9vQdJ.umd.cjs} +144 -144
  4. package/dist/needle-engine.d.ts +12 -6
  5. package/dist/needle-engine.js +400 -400
  6. package/dist/needle-engine.min.js +1 -1
  7. package/dist/needle-engine.umd.cjs +1 -1
  8. package/lib/engine/api.d.ts +1 -1
  9. package/lib/engine/api.js +1 -1
  10. package/lib/engine/api.js.map +1 -1
  11. package/lib/engine/engine_init.js +2 -2
  12. package/lib/engine/engine_init.js.map +1 -1
  13. package/lib/engine/engine_instantiate_resolve.js +40 -10
  14. package/lib/engine/engine_instantiate_resolve.js.map +1 -1
  15. package/lib/engine/engine_license.d.ts +7 -7
  16. package/lib/engine/engine_license.js +72 -72
  17. package/lib/engine/engine_license.js.map +1 -1
  18. package/lib/engine/engine_networking_blob.js +3 -3
  19. package/lib/engine/engine_networking_blob.js.map +1 -1
  20. package/lib/engine/engine_utils_qrcode.js +2 -2
  21. package/lib/engine/engine_utils_qrcode.js.map +1 -1
  22. package/lib/engine/webcomponents/needle menu/needle-menu-spatial.js +2 -2
  23. package/lib/engine/webcomponents/needle menu/needle-menu-spatial.js.map +1 -1
  24. package/lib/engine/webcomponents/needle menu/needle-menu.js +5 -5
  25. package/lib/engine/webcomponents/needle menu/needle-menu.js.map +1 -1
  26. package/lib/engine/webcomponents/needle-engine.js +2 -2
  27. package/lib/engine/webcomponents/needle-engine.js.map +1 -1
  28. package/lib/engine/webcomponents/needle-engine.loading.js +2 -2
  29. package/lib/engine/webcomponents/needle-engine.loading.js.map +1 -1
  30. package/lib/engine/xr/TempXRContext.js +2 -2
  31. package/lib/engine/xr/TempXRContext.js.map +1 -1
  32. package/lib/engine-components/export/usdz/USDZExporter.js +4 -4
  33. package/lib/engine-components/export/usdz/USDZExporter.js.map +1 -1
  34. package/lib/engine-components/timeline/PlayableDirector.d.ts +6 -0
  35. package/lib/engine-components/timeline/PlayableDirector.js +24 -11
  36. package/lib/engine-components/timeline/PlayableDirector.js.map +1 -1
  37. package/package.json +1 -1
  38. package/plugins/common/license.js +4 -4
  39. package/plugins/vite/license.js +4 -4
  40. package/src/engine/api.ts +1 -1
  41. package/src/engine/engine_init.ts +2 -2
  42. package/src/engine/engine_instantiate_resolve.ts +37 -10
  43. package/src/engine/engine_license.ts +69 -69
  44. package/src/engine/engine_networking_blob.ts +3 -3
  45. package/src/engine/engine_utils_qrcode.ts +2 -2
  46. package/src/engine/webcomponents/needle menu/needle-menu-spatial.ts +2 -2
  47. package/src/engine/webcomponents/needle menu/needle-menu.ts +5 -5
  48. package/src/engine/webcomponents/needle-engine.loading.ts +6 -6
  49. package/src/engine/webcomponents/needle-engine.ts +2 -2
  50. package/src/engine/xr/TempXRContext.ts +2 -2
  51. package/src/engine-components/export/usdz/USDZExporter.ts +4 -4
  52. package/src/engine-components/timeline/PlayableDirector.ts +22 -11
@@ -9,19 +9,19 @@ import { getParam } from "./engine_utils.js";
9
9
 
10
10
  const debug = getParam("__debuglic__");
11
11
 
12
- const $rsfEzjO: ((result: boolean) => void)[] = [];
12
+ const _$zbtJJVy: ((result: boolean) => void)[] = [];
13
13
 
14
14
  // DO NOT EDIT MANUALLY
15
- let __nNJ: string = "";
15
+ let miOmwPp: string = "";
16
16
  // eslint-disable-next-line prefer-const
17
- let _ldXlow: string = "";
17
+ let _$ALiqbpo: string = "";
18
18
 
19
19
  if (debug) {
20
- console.log("License Type: " + __nNJ);
21
- if (_ldXlow) {
22
- console.log("License JWT: " + _ldXlow);
20
+ console.log("License Type: " + miOmwPp);
21
+ if (_$ALiqbpo) {
22
+ console.log("License JWT: " + _$ALiqbpo);
23
23
  try {
24
- const payload = JSON.parse(atob(_ldXlow.split(".")[1].replace(/-/g, '+').replace(/_/g, '/')));
24
+ const payload = JSON.parse(atob(_$ALiqbpo.split(".")[1].replace(/-/g, '+').replace(/_/g, '/')));
25
25
  console.log("License JWT payload:", payload);
26
26
  }
27
27
  catch { console.log("License JWT payload: (failed to decode)"); }
@@ -32,8 +32,8 @@ if (debug) {
32
32
  }
33
33
 
34
34
  /** @internal */
35
- export function $HDGDr() {
36
- switch (__nNJ) {
35
+ export function _$QyV() {
36
+ switch (miOmwPp) {
37
37
  case "pro":
38
38
  case "enterprise":
39
39
  return true;
@@ -42,8 +42,8 @@ export function $HDGDr() {
42
42
  }
43
43
 
44
44
  /** @internal */
45
- export function _LMtzpq() {
46
- switch (__nNJ) {
45
+ export function __Carg() {
46
+ switch (miOmwPp) {
47
47
  case "indie":
48
48
  return true;
49
49
  }
@@ -51,8 +51,8 @@ export function _LMtzpq() {
51
51
  }
52
52
 
53
53
  /** @internal */
54
- export function Mwrl() {
55
- switch (__nNJ) {
54
+ export function NVzibd() {
55
+ switch (miOmwPp) {
56
56
  case "edu":
57
57
  return true;
58
58
  }
@@ -60,20 +60,20 @@ export function Mwrl() {
60
60
  }
61
61
 
62
62
  /** @internal */
63
- export function _IrDl() {
64
- return $HDGDr() || _LMtzpq() || Mwrl();
63
+ export function bLW() {
64
+ return _$QyV() || __Carg() || NVzibd();
65
65
  }
66
66
 
67
67
 
68
68
  /** @internal */
69
- export function _CCKlL(cb: (result: boolean) => void) {
70
- if ($HDGDr() || _LMtzpq() || Mwrl())
69
+ export function $bxGE(cb: (result: boolean) => void) {
70
+ if (_$QyV() || __Carg() || NVzibd())
71
71
  return cb(true);
72
- $rsfEzjO.push(cb);
72
+ _$zbtJJVy.push(cb);
73
73
  }
74
74
 
75
- function _vYQmlKI(result: boolean) {
76
- for (const cb of $rsfEzjO) {
75
+ function __JogRhlxZ(result: boolean) {
76
+ for (const cb of _$zbtJJVy) {
77
77
  try {
78
78
  cb(result);
79
79
  }
@@ -88,7 +88,7 @@ function _vYQmlKI(result: boolean) {
88
88
 
89
89
  // ECDSA P-256 public key for verifying license JWTs (verification-only, safe to ship)
90
90
  /* eslint-disable no-secrets/no-secrets -- public key, not a secret */
91
- const _UbM = {
91
+ const CcFRcciC = {
92
92
  kty: "EC",
93
93
  crv: "P-256",
94
94
  x: "A34nyKMjhQYVgzeE4tyLUYdx34TAKogDa7v7PRaO9Lg",
@@ -112,7 +112,7 @@ function base64urlDecode(str: string): Uint8Array {
112
112
  * Verify a JWT license token and return the `type` claim if valid.
113
113
  * Returns null if the JWT is missing, malformed, or has an invalid signature.
114
114
  */
115
- async function _Dyc(jwt: string): Promise<string | null> {
115
+ async function ZhJ(jwt: string): Promise<string | null> {
116
116
  if (!jwt) return null;
117
117
  try {
118
118
  const parts = jwt.split(".");
@@ -122,7 +122,7 @@ async function _Dyc(jwt: string): Promise<string | null> {
122
122
 
123
123
  const key = await crypto.subtle.importKey(
124
124
  "jwk",
125
- _UbM,
125
+ CcFRcciC,
126
126
  { name: "ECDSA", namedCurve: "P-256" },
127
127
  false,
128
128
  ["verify"]
@@ -182,24 +182,24 @@ async function _Dyc(jwt: string): Promise<string | null> {
182
182
  }
183
183
 
184
184
  /** Verify the injected JWT and update the license type if valid.
185
- * The engine ONLY trusts the JWT — the plain __nNJ string is ignored. */
185
+ * The engine ONLY trusts the JWT — the plain miOmwPp string is ignored. */
186
186
  let _jwtVerificationPromise: Promise<void> | undefined = undefined;
187
- async function _vxB(): Promise<void> {
188
- // NOTE: do NOT add an `if (!_ldXlow) return` guard here.
187
+ async function $ESIFccD(): Promise<void> {
188
+ // NOTE: do NOT add an `if (!_$ALiqbpo) return` guard here.
189
189
  // esbuild (Vite dependency pre-bundling) would see the variable as constant ""
190
190
  // and tree-shake the entire JWT verification code path.
191
- const jwt = _ldXlow;
191
+ const jwt = _$ALiqbpo;
192
192
  // Clear after reading — this reassignment also prevents esbuild from
193
193
  // constant-folding the variable (it now has multiple assignment sites).
194
- _ldXlow = "";
195
- const verifiedType = await _Dyc(jwt);
194
+ _$ALiqbpo = "";
195
+ const verifiedType = await ZhJ(jwt);
196
196
  if (verifiedType) {
197
- __nNJ = verifiedType;
197
+ miOmwPp = verifiedType;
198
198
  if (debug) console.log("License type set from verified JWT: " + verifiedType);
199
- _vYQmlKI(_IrDl());
199
+ __JogRhlxZ(bLW());
200
200
  }
201
201
  else {
202
- __nNJ = "basic";
202
+ miOmwPp = "basic";
203
203
  if (debug && jwt) console.warn("JWT verification failed — license reset to basic");
204
204
  }
205
205
  }
@@ -271,7 +271,7 @@ export namespace Telemetry {
271
271
 
272
272
  const attribute = domElement?.getAttribute("no-telemetry");
273
273
  if (attribute === "" || attribute === "true" || attribute === "1") {
274
- if (__nNJ === "pro" || __nNJ === "enterprise") {
274
+ if (miOmwPp === "pro" || miOmwPp === "enterprise") {
275
275
  if (debug) console.debug("Telemetry is disabled via no-telemetry attribute");
276
276
  return false;
277
277
  }
@@ -378,32 +378,32 @@ export namespace Telemetry {
378
378
  }
379
379
 
380
380
 
381
- export function _$zdhiyteo() {
381
+ export function $cLf() {
382
382
 
383
- if(__nNJ === "") __nNJ = "basic";
383
+ if(miOmwPp === "") miOmwPp = "basic";
384
384
 
385
385
  // Start JWT verification — must be here (not top-level) to avoid tree-shaking
386
- _jwtVerificationPromise = _vxB();
386
+ _jwtVerificationPromise = $ESIFccD();
387
387
 
388
388
  Telemetry.init();
389
389
  ContextRegistry.registerCallback(ContextEvent.ContextRegistered, evt => {
390
- __OFIRedHS(evt.context);
391
- $LMcMC(evt.context);
392
- setTimeout(() => _PVj(evt.context), 2000);
390
+ _JTSOMz(evt.context);
391
+ __rlGEVK(evt.context);
392
+ setTimeout(() => __OmIq(evt.context), 2000);
393
393
  });
394
394
  }
395
395
 
396
- export let _$pva: Promise<void> | undefined = undefined;
396
+ export let _MHgSj: Promise<void> | undefined = undefined;
397
397
  let applicationIsForbidden = false;
398
398
  let applicationForbiddenText = "";
399
- async function $SMhOeOfj() {
399
+ async function $pek() {
400
400
  // Only perform the runtime license check once
401
- if (_$pva) return _$pva;
401
+ if (_MHgSj) return _MHgSj;
402
402
  // Wait for JWT verification to complete first (if running)
403
403
  if (_jwtVerificationPromise) {
404
404
  await _jwtVerificationPromise;
405
405
  }
406
- if (__nNJ === "basic") {
406
+ if (miOmwPp === "basic") {
407
407
  try {
408
408
  const licenseUrl = "https://needle.tools/api/v1/needle-engine/check?location=" + encodeURIComponent(window.location.href) + "&version=" + VERSION + "&generator=" + encodeURIComponent(GENERATOR);
409
409
  const res = await fetch(licenseUrl, {
@@ -415,29 +415,29 @@ async function $SMhOeOfj() {
415
415
  if (res?.status === 200) {
416
416
  applicationIsForbidden = false;
417
417
  if (debug) console.log("License check succeeded");
418
- __nNJ = "pro";
419
- _vYQmlKI(true);
418
+ miOmwPp = "pro";
419
+ __JogRhlxZ(true);
420
420
  }
421
421
  else if (res?.status === 403) {
422
- _vYQmlKI(false);
422
+ __JogRhlxZ(false);
423
423
  applicationIsForbidden = true;
424
424
  applicationForbiddenText = await res.text();
425
425
  }
426
426
  else {
427
- _vYQmlKI(false);
427
+ __JogRhlxZ(false);
428
428
  if (debug) console.log("License check failed with status " + res?.status);
429
429
  }
430
430
  }
431
431
  catch (err) {
432
- _vYQmlKI(false);
432
+ __JogRhlxZ(false);
433
433
  if (debug) console.error("License check failed", err);
434
434
  }
435
435
  }
436
- else if (debug) console.log("Runtime license check is skipped because license is already applied as \"" + __nNJ + "\"");
436
+ else if (debug) console.log("Runtime license check is skipped because license is already applied as \"" + miOmwPp + "\"");
437
437
  }
438
- _$pva = $SMhOeOfj();
438
+ _MHgSj = $pek();
439
439
 
440
- async function $LMcMC(ctx: IContext) {
440
+ async function __rlGEVK(ctx: IContext) {
441
441
  function createForbiddenElement() {
442
442
  const div = document.createElement("div");
443
443
  div.className = "needle-forbidden";
@@ -502,40 +502,40 @@ async function $LMcMC(ctx: IContext) {
502
502
  }, 500)
503
503
  }
504
504
 
505
- async function __OFIRedHS(ctx: IContext) {
505
+ async function _JTSOMz(ctx: IContext) {
506
506
  try {
507
- if (!$HDGDr() && !_LMtzpq()) {
508
- return __qFJUQ(ctx);
507
+ if (!_$QyV() && !__Carg()) {
508
+ return dtvHGhBg(ctx);
509
509
  }
510
510
  }
511
511
  catch (err) {
512
512
  if (debug) console.log("License check failed", err)
513
- return __qFJUQ(ctx)
513
+ return dtvHGhBg(ctx)
514
514
  }
515
- if (debug) __qFJUQ(ctx)
515
+ if (debug) dtvHGhBg(ctx)
516
516
  }
517
517
 
518
518
 
519
519
 
520
- async function __qFJUQ(ctx: IContext) {
520
+ async function dtvHGhBg(ctx: IContext) {
521
521
 
522
522
  // if the engine loads faster than the license check, we need to capture the ready event here
523
523
  let isReady = false;
524
524
  ctx.domElement.addEventListener("ready", () => isReady = true);
525
525
 
526
- await _$pva?.catch(() => { });
526
+ await _MHgSj?.catch(() => { });
527
527
 
528
528
 
529
- if ($HDGDr() || _LMtzpq()) return;
530
- if (_IrDl() === false) _RKZHfk();
529
+ if (_$QyV() || __Carg()) return;
530
+ if (bLW() === false) __PJeisZrd();
531
531
 
532
532
  // check if the engine is already ready (meaning has finished loading)
533
533
  if (isReady) {
534
- osmhoV(ctx);
534
+ __CRBs(ctx);
535
535
  }
536
536
  else {
537
537
  ctx.domElement.addEventListener("ready", () => {
538
- osmhoV(ctx);
538
+ __CRBs(ctx);
539
539
  });
540
540
  }
541
541
  }
@@ -543,7 +543,7 @@ async function __qFJUQ(ctx: IContext) {
543
543
  // const licenseElementIdentifier = "needle-license-element";
544
544
  // const licenseDuration = 10000;
545
545
  // const licenseDelay = 1200;
546
- function osmhoV(ctx: IContext) {
546
+ function __CRBs(ctx: IContext) {
547
547
 
548
548
  const style = `
549
549
  position: relative;
@@ -556,7 +556,7 @@ function osmhoV(ctx: IContext) {
556
556
  padding: 10px;
557
557
  padding-left: 30px;
558
558
  `;
559
- if (__nNJ === "edu") {
559
+ if (miOmwPp === "edu") {
560
560
  if (navigator.webdriver) {
561
561
  console.log("This project is supported by Needle for Education – https://needle.tools");
562
562
  }
@@ -613,7 +613,7 @@ function osmhoV(ctx: IContext) {
613
613
  }
614
614
  }, 1000);
615
615
 
616
- if (Mwrl()) {
616
+ if (NVzibd()) {
617
617
  const removeDelay = 20_000;
618
618
  setTimeout(() => {
619
619
  clearInterval(interval);
@@ -622,7 +622,7 @@ function osmhoV(ctx: IContext) {
622
622
  const intervalInMinutes = 5;
623
623
  setTimeout(() => {
624
624
  if (ctx.domElement.parentNode)
625
- osmhoV(ctx);
625
+ __CRBs(ctx);
626
626
  }, 1000 * 60 * intervalInMinutes)
627
627
  }, removeDelay);
628
628
  }
@@ -633,7 +633,7 @@ function osmhoV(ctx: IContext) {
633
633
  const base64Logo = "data:image/webp;base64,UklGRrABAABXRUJQVlA4WAoAAAAQAAAAHwAAHwAAQUxQSKEAAAARN6CmbSM4WR7vdARON11EBDq3fLiNbVtVzpMCPlKAEzsx0Y/x+Ovuv4dn0EFE/ydAvz6YggXzgh5sVgXM/zOC/4sii7qgGvB5N7hmuQYwkvazWAu1JPW41FXSHq6pnaQWvqYH18Fc0j1hO/BFTtIeSBlJi5w6qIIO7IOrwhFsB2Yxukif0FTRLpXswHR8MxbslKe9VZsn/Ub5C7YFOpqSTABWUDgg6AAAAFAGAJ0BKiAAIAA+7VyoTqmkpCI3+qgBMB2JbACdMt69DwMIQBLhkTO6XwY00UEDK6cNIDnuNibPf0EgAP7Y1myuiQHLDsF/0h5unrGh6WAbv7aegg2ZMd3uRKfT/3SJztcaujYfTvMXspfCTmYcoO6a+vhC3ss4M8uM58t4siiu59I4aOl59e9Sr6xoxYlHf2v+NnBNpJYeJf8jABQAId/PXuBkLEFkiCucgSGEcfhvajql/j3reCGl0M5/9gQWy7ayNPs+wlvIxFnNfSlfuND4CZOCyxOHhRqOmHN4ULHo3tCSrUNvgAA=";
634
634
 
635
635
  let lastLogTime = 0;
636
- async function _RKZHfk(_logo?: string) {
636
+ async function __PJeisZrd(_logo?: string) {
637
637
  const now = Date.now();
638
638
  if (now - lastLogTime < 2000) return;
639
639
  lastLogTime = now;
@@ -666,7 +666,7 @@ async function _RKZHfk(_logo?: string) {
666
666
  }
667
667
 
668
668
 
669
- async function _PVj(context: IContext) {
669
+ async function __OmIq(context: IContext) {
670
670
  // We can't send beacons from cross-origin isolated pages
671
671
  if (window.crossOriginIsolated) return;
672
672
 
@@ -681,7 +681,7 @@ async function _PVj(context: IContext) {
681
681
 
682
682
  // current url without query parameters
683
683
  const currentUrl = window.location.href.split("?")[0];
684
- const license = __nNJ;
684
+ const license = miOmwPp;
685
685
 
686
686
  const beaconData = {
687
687
  license,
@@ -1,7 +1,7 @@
1
1
  import { FileLoader } from "three";
2
2
 
3
3
  import { showBalloonWarning } from "./debug/index.js";
4
- import { _IrDl } from "./engine_license.js";
4
+ import { bLW } from "./engine_license.js";
5
5
  import { delay } from "./engine_utils.js";
6
6
  import { md5AsBytes, md5Hex, sha256Base64 } from "./engine_utils_hash.js";
7
7
 
@@ -45,7 +45,7 @@ export namespace BlobStorage {
45
45
  */
46
46
  export function canUpload(info: { filesize: number }) {
47
47
  const sizeInMB = info.filesize / 1024 / 1024;
48
- if (_IrDl()) {
48
+ if (bLW()) {
49
49
  return sizeInMB < maxSizeInMB;
50
50
  }
51
51
  return sizeInMB < maxFreeSizeInMB;
@@ -100,7 +100,7 @@ export namespace BlobStorage {
100
100
  console.warn(`Your file is too large for uploading (${filesizeInMB.toFixed(1)}MB). Max allowed size is ${maxSizeInMB}MB`);
101
101
  return null;
102
102
  }
103
- else if (!_IrDl() && filesizeInMB > maxFreeSizeInMB) {
103
+ else if (!bLW() && filesizeInMB > maxFreeSizeInMB) {
104
104
  if (opts?.silent !== true) showBalloonWarning(`File is too large for uploading. Please get a <a href=\"https://needle.tools/pricing\" target=\"_blank\">commercial license</a> to upload files larger than 5MB`);
105
105
  console.warn(`Your file is too large for uploading (${filesizeInMB.toFixed(1)}MB). Max size is 5MB for non-commercial users. Please get a commercial license at https://needle.tools/pricing for larger files (up to 50MB)`);
106
106
  return null;
@@ -4,7 +4,7 @@ import { Quaternion, Vector2, Vector3, Vector4 } from "three";
4
4
 
5
5
  import { needleLogoOnlySVG } from "./assets/index.js";
6
6
  import { isDevEnvironment } from "./debug/debug.js";
7
- import { _IrDl } from "./engine_license.js";
7
+ import { bLW } from "./engine_license.js";
8
8
  import { InternalAttributeUtils } from "./engine_utils_attributes.js";
9
9
  import type { NeedleEngineWebComponent } from "./webcomponents/needle-engine.js";
10
10
 
@@ -135,7 +135,7 @@ async function internalRenderQRCodeOverlays(canvas: HTMLCanvasElement, args: { s
135
135
  console.debug("[QR Code] No web component found")
136
136
  }
137
137
 
138
- const canUseCustomLogo = _IrDl();
138
+ const canUseCustomLogo = bLW();
139
139
 
140
140
  // Query logo src from needle-engine attribute.
141
141
  // For any supported attribute it's possible to use "falsey" values (e.g. "0" or "false" to disable the logo in the QR code)
@@ -2,7 +2,7 @@ import { Mesh, Object3D, TextureLoader, Vector4 } from "three";
2
2
  import ThreeMeshUI from "three-mesh-ui";
3
3
 
4
4
  import { addNewComponent } from "../../engine_components.js";
5
- import { $HDGDr } from "../../engine_license.js";
5
+ import { _$QyV } from "../../engine_license.js";
6
6
  import { OneEuroFilterXYZ } from "../../engine_math.js";
7
7
  import type { Context } from "../../engine_setup.js";
8
8
  import { lookAtObject } from "../../engine_three_utils.js";
@@ -321,7 +321,7 @@ export class NeedleSpatialMenu {
321
321
  }
322
322
  if (this.menu) {
323
323
  const index = this.menu.children.indexOf(this._poweredByNeedleElement as any);
324
- if (!this._showNeedleLogo && $HDGDr()) {
324
+ if (!this._showNeedleLogo && _$QyV()) {
325
325
  if (index >= 0) {
326
326
  this._poweredByNeedleElement.removeFromParent();
327
327
  this.markDirty();
@@ -1,6 +1,6 @@
1
1
  import { showBalloonMessage } from "../../debug/debug.js";
2
2
  import type { Context } from "../../engine_context.js";
3
- import { _IrDl, _CCKlL, Telemetry } from "../../engine_license.js";
3
+ import { bLW, $bxGE, Telemetry } from "../../engine_license.js";
4
4
  import { isLocalNetwork } from "../../engine_networking_utils.js";
5
5
  import { HTMLElementBase } from "../../engine_ssr.js";
6
6
  import { DeviceUtilities, getParam } from "../../engine_utils.js";
@@ -778,8 +778,8 @@ export class NeedleMenuElement extends HTMLElementBase {
778
778
  try {
779
779
  // if the user has a license then we CAN hide the needle logo
780
780
  // calling this method immediately will cause an issue with vite bundling tho
781
- window.requestAnimationFrame(() => _CCKlL(res => {
782
- if (res == true && _IrDl() && !debugNonCommercial) {
781
+ window.requestAnimationFrame(() => $bxGE(res => {
782
+ if (res == true && bLW() && !debugNonCommercial) {
783
783
  let visible = this._userRequestedLogoVisible;
784
784
  if (visible === undefined) visible = false;
785
785
  this.___onSetLogoVisible(visible);
@@ -830,7 +830,7 @@ export class NeedleMenuElement extends HTMLElementBase {
830
830
  // ensure the menu is not hidden or removed
831
831
  const requiredParent = this?.parentNode;
832
832
  if (this.style.display != "flex" || this.style.visibility != "visible" || this.style.opacity != "1" || requiredParent != this._domElement?.shadowRoot) {
833
- if (!_IrDl()) {
833
+ if (!bLW()) {
834
834
  const change = changeEventCounter++;
835
835
  // if a user doesn't have a local pro license *but* for development the menu is hidden then we show a warning
836
836
  if (isLocalNetwork() && this._userRequestedMenuVisible === false) {
@@ -918,7 +918,7 @@ export class NeedleMenuElement extends HTMLElementBase {
918
918
  showNeedleLogo(visible: boolean) {
919
919
  this._userRequestedLogoVisible = visible;
920
920
  if (!visible) {
921
- if (!_IrDl() || debugNonCommercial) {
921
+ if (!bLW() || debugNonCommercial) {
922
922
  console.warn("[Needle Engine] You need a PRO license to hide the Needle Engine logo in production.");
923
923
  const localNetwork = isLocalNetwork()
924
924
  if (!localNetwork) return;
@@ -1,6 +1,6 @@
1
1
  import { needleLogoOnlySVG } from "../assets/index.js"
2
2
  import { isDevEnvironment, showBalloonWarning } from "../debug/index.js";
3
- import { _IrDl, $HDGDr, _$pva } from "../engine_license.js";
3
+ import { bLW, _$QyV, _MHgSj } from "../engine_license.js";
4
4
  import { Mathf } from "../engine_math.js";
5
5
  import { LoadingProgressArgs } from "../engine_setup.js";
6
6
  import { getParam } from "../engine_utils.js";
@@ -205,7 +205,7 @@ export class EngineLoadingView implements ILoadingViewHandler {
205
205
  }
206
206
 
207
207
 
208
- const hasLicense = $HDGDr();
208
+ const hasLicense = _$QyV();
209
209
  if (!existing) {
210
210
  this._loadingElement.style.position = "absolute";
211
211
  this._loadingElement.style.width = "100%";
@@ -373,7 +373,7 @@ export class EngineLoadingView implements ILoadingViewHandler {
373
373
 
374
374
  // private async handleRuntimeLicense(loadingElement: HTMLElement) {
375
375
  // // First check if we have a commercial license
376
- // let commercialLicense = _IrDl();
376
+ // let commercialLicense = bLW();
377
377
  // // if it's the case then we don't need to perform a runtime check
378
378
  // if (commercialLicense) return;
379
379
 
@@ -393,10 +393,10 @@ export class EngineLoadingView implements ILoadingViewHandler {
393
393
  // loadingElement.appendChild(nonCommercialContainer);
394
394
 
395
395
  // // Use the runtime license check
396
- // if (!isDevEnvironment() && _$pva) {
396
+ // if (!isDevEnvironment() && _MHgSj) {
397
397
  // if (debugLicense) console.log("Waiting for runtime license check");
398
- // await _$pva;
399
- // commercialLicense = _IrDl();
398
+ // await _MHgSj;
399
+ // commercialLicense = bLW();
400
400
  // }
401
401
  // if (commercialLicense) return;
402
402
  // nonCommercialContainer.style.transition = "opacity .5s ease-in-out";
@@ -3,7 +3,7 @@ import type { Texture } from "three";
3
3
  import { isDevEnvironment, showBalloonWarning } from "../debug/index.js";
4
4
  import { PUBLIC_KEY, VERSION } from "../engine_constants.js";
5
5
  import { ContextEvent, ContextRegistry } from "../engine_context_registry.js";
6
- import { _IrDl } from "../engine_license.js";
6
+ import { bLW } from "../engine_license.js";
7
7
  import { onStart } from "../engine_lifecycle_api.js";
8
8
  import { setDracoDecoderPath, setDracoDecoderType, setKtx2TranscoderPath } from "../engine_loaders.gltf.js";
9
9
  import { Context, ContextCreateArgs } from "../engine_setup.js";
@@ -579,7 +579,7 @@ export class NeedleEngineWebComponent extends HTMLElementBase implements INeedle
579
579
 
580
580
 
581
581
  // Loading start events
582
- const allowOverridingDefaultLoading = _IrDl();
582
+ const allowOverridingDefaultLoading = bLW();
583
583
  // default loading can be overriden by calling preventDefault in the onload start event
584
584
  this.ensureLoadStartIsRegistered();
585
585
  let useDefaultLoading = this.dispatchEvent(new CustomEvent("loadstart", {
@@ -3,7 +3,7 @@ import { ArrayCamera, AxesHelper, Camera, Color, DirectionalLight, Fog, GridHelp
3
3
  import { needleLogoOnlySVG } from "../assets/index.js";
4
4
  import { isDevEnvironment } from "../debug/index.js";
5
5
  import { ObjectUtils, PrimitiveType } from "../engine_create_objects.js";
6
- import { _IrDl } from "../engine_license.js";
6
+ import { bLW } from "../engine_license.js";
7
7
  import { Mathf } from "../engine_math.js";
8
8
  import { delay, DeviceUtilities } from "../engine_utils.js";
9
9
 
@@ -215,7 +215,7 @@ export class TemporaryXRContext {
215
215
  this._scene.background = new Color(0x000000);
216
216
 
217
217
  let logoSrc = needleLogoOnlySVG;
218
- if (_IrDl()) {
218
+ if (bLW()) {
219
219
  const htmlComponent = document.querySelector("needle-engine");
220
220
  if (htmlComponent) {
221
221
  const licenseLogo = htmlComponent.getAttribute("logo-src");
@@ -3,7 +3,7 @@ import { Euler, Material, Matrix4, Mesh, Object3D, Quaternion, Vector3 } from "t
3
3
 
4
4
  import { isDevEnvironment, showBalloonMessage, showBalloonWarning } from "../../../engine/debug/index.js";
5
5
  import { findObjectOfType } from "../../../engine/engine_components.js";
6
- import { $HDGDr } from "../../../engine/engine_license.js";
6
+ import { _$QyV } from "../../../engine/engine_license.js";
7
7
  import { serializable } from "../../../engine/engine_serialization.js";
8
8
  import { getFormattedDate, Progress } from "../../../engine/engine_time_utils.js";
9
9
  import { DeviceUtilities, getParam } from "../../../engine/engine_utils.js";
@@ -277,7 +277,7 @@ export class USDZExporter extends Behaviour {
277
277
  let name = this.exportFileName ?? this.objectToExport?.name ?? this.name;
278
278
  name += "-" + getFormattedDate(); // seems iOS caches the file in some cases, this ensures we always have a fresh file
279
279
 
280
- if (!$HDGDr()) {
280
+ if (!_$QyV()) {
281
281
  if (name !== "") name += "-";
282
282
  name += "MadeWithNeedle";
283
283
  }
@@ -682,7 +682,7 @@ export class USDZExporter extends Behaviour {
682
682
  if (debug)
683
683
  showBalloonMessage("Quicklook url: " + callToActionURL);
684
684
  if (callToActionURL) {
685
- if (!$HDGDr()) {
685
+ if (!_$QyV()) {
686
686
  console.warn("Quicklook closed: custom redirects require a Needle Engine Pro license: https://needle.tools/pricing", callToActionURL)
687
687
  }
688
688
  else {
@@ -697,7 +697,7 @@ export class USDZExporter extends Behaviour {
697
697
  private buildQuicklookOverlay(): CustomBranding {
698
698
  const obj: CustomBranding = {};
699
699
  if (this.customBranding) Object.assign(obj, this.customBranding);
700
- if (!$HDGDr()) {
700
+ if (!_$QyV()) {
701
701
  console.log("Custom Quicklook banner text requires pro license: https://needle.tools/pricing");
702
702
  obj.callToAction = "Close";
703
703
  obj.checkoutTitle = "🌵 Made with Needle";
@@ -120,9 +120,11 @@ export class PlayableDirector extends Behaviour {
120
120
  if (this._playableAsset !== value) {
121
121
  this._playableAsset = value;
122
122
  this._needsGraphRebuild = true;
123
+ this._didWarnInvalid = false;
123
124
  }
124
125
  }
125
126
  private _playableAsset?: Models.TimelineAssetModel;
127
+ private _didWarnInvalid = false;
126
128
 
127
129
  /**
128
130
  * When true, the timeline starts playing automatically when the component awakens.
@@ -195,19 +197,27 @@ export class PlayableDirector extends Behaviour {
195
197
  awake(): void {
196
198
  if (debug) console.log(`[Timeline] Awake '${this.name}'`, this);
197
199
 
200
+ // Note: we intentionally do NOT warn here if there are no tracks yet. A playableAsset can
201
+ // legitimately be assigned after awake (e.g. via TimelineBuilder.install() or deserialization),
202
+ // see rebuildGraph() in play()/evaluate(). The validity warning lives in play()/evaluate(),
203
+ // which are the points where a still-empty director actually indicates a misconfiguration.
198
204
  this.rebuildGraph();
205
+ }
199
206
 
200
- if (!this.isValid() && (debug || isDevEnvironment())) {
201
- if (debug) {
202
- console.warn("PlayableDirector is not valid", "Asset?", this.playableAsset, "Tracks:", this.playableAsset?.tracks, "IsArray?", Array.isArray(this.playableAsset?.tracks), this);
203
- }
204
- else if (!this.playableAsset?.tracks?.length) {
205
- console.warn("PlayableDirector has no tracks");
206
- }
207
- else {
208
- console.warn("PlayableDirector is not valid");
209
- }
207
+ /**
208
+ * Returns true if the director has a valid playable asset. When invalid, warns once (dev only)
209
+ * that the director has no tracks called from the user-facing play()/evaluate() entry points.
210
+ */
211
+ private warnIfInvalid(method: string): boolean {
212
+ if (this.isValid()) return true;
213
+ if (!this._didWarnInvalid && (debug || isDevEnvironment())) {
214
+ this._didWarnInvalid = true;
215
+ if (!this.playableAsset?.tracks?.length)
216
+ console.warn(`PlayableDirector '${this.name}' has no tracks — assign a playableAsset (or build one via TimelineBuilder) before calling ${method}().`, this);
217
+ else
218
+ console.warn(`PlayableDirector '${this.name}' is not valid`, this);
210
219
  }
220
+ return false;
211
221
  }
212
222
 
213
223
  /** @internal */
@@ -276,7 +286,7 @@ export class PlayableDirector extends Behaviour {
276
286
  * If the timeline is already playing this method does nothing.
277
287
  */
278
288
  async play() {
279
- if (!this.isValid()) return;
289
+ if (!this.warnIfInvalid("play")) return;
280
290
  // Ensure graph is built (handles the case where playableAsset is assigned after awake)
281
291
  if (this._needsGraphRebuild) this.rebuildGraph();
282
292
  const pauseChanged = this._isPaused == true;
@@ -354,6 +364,7 @@ export class PlayableDirector extends Behaviour {
354
364
  * Evaluate the timeline at the current time. This is useful when you want to manually update the timeline e.g. when the timeline is paused and you set `time` to a new value.
355
365
  */
356
366
  evaluate() {
367
+ if (!this.warnIfInvalid("evaluate")) return;
357
368
  if (this._needsGraphRebuild) this.rebuildGraph();
358
369
  this.internalEvaluate(true);
359
370
  }